From 4bac535eabb83eda7b590660b7cb2e9fc724bc97 Mon Sep 17 00:00:00 2001 From: Xavier Barrachina Civera Date: Wed, 17 Jul 2024 17:46:02 +0200 Subject: [PATCH] =?UTF-8?q?Initial=20population=20of=20repo.=20Co-authored?= =?UTF-8?q?-by:=20Arfima=20Dev=20=20Co-authored-by:=20Gonz?= =?UTF-8?q?alo=20=C3=81lvarez=20=20Co-authored-by:=20?= =?UTF-8?q?Jonathan=20Y=C3=A1nez=20=20Co-authored-by:?= =?UTF-8?q?=20V=C3=ADctor=20de=20Luna=20=20Co-authored?= =?UTF-8?q?-by:=20Virginia=20Morales=20?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Xavier Barrachina Civera --- .coveragerc | 30 +- .flake8 | 3 - .markdownlint.yaml | 3 + .readthedocs.yml | 25 +- AUTHORS.rst | 7 +- CHANGELOG.rst | 4 +- CONTRIBUTING.rst | 84 +- LICENSE | 1 - LICENSES/Apache-2.0.txt => LICENSE.txt | 0 README.md | 33 + README.rst | 54 - docs/assets.rst | 7 + docs/conf.py | 86 +- docs/dynamics.rst | 7 + docs/functions.rst | 7 + docs/images/OS-Climate-Logo.png | Bin 0 -> 109264 bytes docs/index.rst | 100 +- docs/license.rst | 3 +- docs/modules.rst | 46 + docs/random_variables.rst | 7 + docs/readme.rst | 31 +- docs/requirements.txt | 5 - pyproject.toml | 129 +- setup.sh | 139 --- src/osc_physrisk_financial/__init__.py | 17 +- src/osc_physrisk_financial/assets.py | 342 ++++++ src/osc_physrisk_financial/dynamics.py | 121 ++ src/osc_physrisk_financial/functions.py | 172 +++ .../random_variables.py | 1033 +++++++++++++++++ src/osc_physrisk_financial/skeleton.py | 149 --- tests/conftest.py | 8 +- tests/test_dynamics.py | 25 + tests/test_functions.py | 205 ++++ tests/test_powerPlant.py | 68 ++ tests/test_random_variables.py | 295 +++++ tests/test_realstate.py | 321 +++++ tests/test_skeleton.py | 25 - tox.ini | 93 -- 38 files changed, 3019 insertions(+), 666 deletions(-) delete mode 100644 .flake8 delete mode 120000 LICENSE rename LICENSES/Apache-2.0.txt => LICENSE.txt (100%) create mode 100644 README.md delete mode 100644 README.rst create mode 100644 docs/assets.rst create mode 100644 docs/dynamics.rst create mode 100644 docs/functions.rst create mode 100644 docs/images/OS-Climate-Logo.png create mode 100644 docs/modules.rst create mode 100644 docs/random_variables.rst delete mode 100644 docs/requirements.txt delete mode 100755 setup.sh create mode 100644 src/osc_physrisk_financial/assets.py create mode 100644 src/osc_physrisk_financial/dynamics.py create mode 100644 src/osc_physrisk_financial/functions.py create mode 100644 src/osc_physrisk_financial/random_variables.py delete mode 100644 src/osc_physrisk_financial/skeleton.py create mode 100644 tests/test_dynamics.py create mode 100644 tests/test_functions.py create mode 100644 tests/test_powerPlant.py create mode 100644 tests/test_random_variables.py create mode 100644 tests/test_realstate.py delete mode 100644 tests/test_skeleton.py delete mode 100644 tox.ini diff --git a/.coveragerc b/.coveragerc index 43d93a7..e6ee9d6 100644 --- a/.coveragerc +++ b/.coveragerc @@ -1,28 +1,2 @@ -# .coveragerc to control coverage.py -[run] -branch = True -source = osc_physrisk_financial -# omit = bad_file.py - -[paths] -source = - src/ - */site-packages/ - -[report] -# Regexes for lines to exclude from consideration -exclude_lines = - # Have to re-enable the standard pragma - pragma: no cover - - # Don't complain about missing debug-only code: - def __repr__ - if self\.debug - - # Don't complain if tests don't hit defensive assertion code: - raise AssertionError - raise NotImplementedError - - # Don't complain if non-runnable code isn't run: - if 0: - if __name__ == .__main__.: +[coverage:report] +skip_empty = true diff --git a/.flake8 b/.flake8 deleted file mode 100644 index cb23f32..0000000 --- a/.flake8 +++ /dev/null @@ -1,3 +0,0 @@ -[flake8] -max-line-length = 160 -extend-ignore = E203, E501 diff --git a/.markdownlint.yaml b/.markdownlint.yaml index 34efb59..91b14f4 100644 --- a/.markdownlint.yaml +++ b/.markdownlint.yaml @@ -8,3 +8,6 @@ default: true extends: null MD013: false +MD033: { + "allowed_elements": ["img"] +} diff --git a/.readthedocs.yml b/.readthedocs.yml index a2bcab3..1d2c166 100644 --- a/.readthedocs.yml +++ b/.readthedocs.yml @@ -1,27 +1,24 @@ +--- # Read the Docs configuration file # See https://docs.readthedocs.io/en/stable/config-file/v2.html for details # Required version: 2 +# Set the OS, Python version and other tools you might need +build: + os: ubuntu-22.04 + tools: + python: "3.11" + jobs: + post_install: + - pip install --upgrade pdm + - VIRTUAL_ENV=$(dirname $(dirname $(which python))) pdm install -dG docs + # Build documentation in the docs/ directory with Sphinx sphinx: configuration: docs/conf.py -# Build documentation with MkDocs -#mkdocs: -# configuration: mkdocs.yml - # Optionally build your docs in additional formats such as PDF formats: - pdf - -build: - os: ubuntu-22.04 - tools: - python: "3.11" - -python: - install: - - requirements: docs/requirements.txt - - {path: ., method: pip} diff --git a/AUTHORS.rst b/AUTHORS.rst index 5281c92..8308d5e 100644 --- a/AUTHORS.rst +++ b/AUTHORS.rst @@ -2,4 +2,9 @@ Contributors ============ -* github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> +* Arfima Dev +* Gonzalo Álvarez +* Jonathan Yánez +* Víctor de Luna +* Virginia Morales +* Xavier Barrachina diff --git a/CHANGELOG.rst b/CHANGELOG.rst index 226e6f5..6fcf78a 100644 --- a/CHANGELOG.rst +++ b/CHANGELOG.rst @@ -5,6 +5,4 @@ Changelog Version 0.1 =========== -- Feature A added -- FIX: nasty bug #1729 fixed -- add your changes here! +- Initial version! diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index a4d1ae7..ec4b46f 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -1,28 +1,3 @@ -.. todo:: THIS IS SUPPOSED TO BE AN EXAMPLE. MODIFY IT ACCORDING TO YOUR NEEDS! - - The document assumes you are using a source repository service that promotes a - contribution model similar to `GitHub's fork and pull request workflow`_. - While this is true for the majority of services (like GitHub, GitLab, - BitBucket), it might not be the case for private repositories (e.g., when - using Gerrit). - - Also notice that the code examples might refer to GitHub URLs or the text - might use GitHub specific terminology (e.g., *Pull Request* instead of *Merge - Request*). - - Please make sure to check the document having these assumptions in mind - and update things accordingly. - -.. todo:: Provide the correct links/replacements at the bottom of the document. - -.. todo:: You might want to have a look on `PyScaffold's contributor's guide`_, - - especially if your project is open source. The text should be very similar to - this template, but there are a few extra contents that you might decide to - also include, like mentioning labels of your issue tracker or automated - releases. - - ============ Contributing ============ @@ -72,25 +47,20 @@ by adding missing information and correcting mistakes. This means that the docs are kept in the same repository as the project code, and that any documentation update is done in the same way was a code contribution. -.. todo:: Don't forget to mention which markup language you are using. - - e.g., reStructuredText_ or CommonMark_ with MyST_ extensions. -.. todo:: If your project is hosted on GitHub, you can also mention the following tip: - - .. tip:: - Please notice that the `GitHub web interface`_ provides a quick way of - propose changes in ``osc-physrisk-financial``'s files. While this mechanism can - be tricky for normal code contributions, it works perfectly fine for - contributing to the docs, and can be quite handy. - - If you are interested in trying this method out, please navigate to - the ``docs`` folder in the source repository_, find which file you - would like to propose changes and click in the little pencil icon at the - top, to open `GitHub's code editor`_. Once you finish editing the file, - please write a message in the form at the bottom of the page describing - which changes have you made and what are the motivations behind them and - submit your proposal. +.. tip:: + Please notice that the `GitHub web interface`_ provides a quick way of + propose changes in ``osc-physrisk-financial``'s files. While this mechanism can + be tricky for normal code contributions, it works perfectly fine for + contributing to the docs, and can be quite handy. + + If you are interested in trying this method out, please navigate to + the ``docs`` folder in the source repository_, find which file you + would like to propose changes and click in the little pencil icon at the + top, to open `GitHub's code editor`_. Once you finish editing the file, + please write a message in the form at the bottom of the page describing + which changes have you made and what are the motivations behind them and + submit your proposal. When working on documentation changes in your local machine, you can compile them using |tox|_:: @@ -151,8 +121,6 @@ Clone the repository to be able to import the package under development in the Python REPL. - .. todo:: if you are not using pre-commit, please remove the following item: - #. Install |pre-commit|_:: pip install pre-commit @@ -182,11 +150,9 @@ Implement your changes to record your changes in git_. - .. todo:: if you are not using pre-commit, please remove the following item: - Please make sure to see the validation messages from |pre-commit|_ and fix any eventual issues. - This should automatically use flake8_/black_ to check/fix the code style + This should automatically use ruff_ to check/fix the code style in a way that is compatible with the project. .. important:: Don't forget to add unit tests and documentation in case your @@ -218,11 +184,9 @@ Submit your contribution #. Go to the web page of your fork and click |contribute button| to send your changes for review. - .. todo:: if you are using GitHub, you can uncomment the following paragraph - - Find more detailed information in `creating a PR`_. You might also want to open - the PR as a draft first and mark it as ready for review after the feedbacks - from the continuous integration (CI) system or any required fixes. + Find more detailed information in `creating a PR`_. You might also want to open + the PR as a draft first and mark it as ready for review after the feedbacks + from the continuous integration (CI) system or any required fixes. Troubleshooting @@ -278,11 +242,6 @@ Maintainer tasks Releases -------- -.. todo:: This section assumes you are using PyPI to publicly release your package. - - If instead you are using a different/private package index, please update - the instructions accordingly. - If you are part of the group of maintainers and have correct user permissions on PyPI_, the following steps can be used to release a new version for ``osc-physrisk-financial``: @@ -308,15 +267,12 @@ on PyPI_, the following steps can be used to release a new version for of environments, including private companies and proprietary code bases. -.. <-- start --> -.. todo:: Please review and change the following definitions: .. |the repository service| replace:: GitHub .. |contribute button| replace:: "Create pull request" -.. _repository: https://github.com//osc-physrisk-financial -.. _issue tracker: https://github.com//osc-physrisk-financial/issues -.. <-- end --> +.. _repository: https://github.com/os-climate/osc-physrisk-financial +.. _issue tracker: https://github.com/os-climate/osc-physrisk-financial/issues .. |virtualenv| replace:: ``virtualenv`` @@ -331,7 +287,7 @@ on PyPI_, the following steps can be used to release a new version for .. _descriptive commit message: https://chris.beams.io/posts/git-commit .. _docstrings: https://www.sphinx-doc.org/en/master/usage/extensions/napoleon.html .. _first-contributions tutorial: https://github.com/firstcontributions/first-contributions -.. _flake8: https://flake8.pycqa.org/en/stable/ +.. _ruff: https://docs.astral.sh/ruff/ .. _git: https://git-scm.com .. _GitHub's fork and pull request workflow: https://guides.github.com/activities/forking/ .. _guide created by FreeCodeCamp: https://github.com/FreeCodeCamp/how-to-contribute-to-open-source diff --git a/LICENSE b/LICENSE deleted file mode 120000 index 5431dc1..0000000 --- a/LICENSE +++ /dev/null @@ -1 +0,0 @@ -LICENSES/Apache-2.0.txt \ No newline at end of file diff --git a/LICENSES/Apache-2.0.txt b/LICENSE.txt similarity index 100% rename from LICENSES/Apache-2.0.txt rename to LICENSE.txt diff --git a/README.md b/README.md new file mode 100644 index 0000000..08ccf17 --- /dev/null +++ b/README.md @@ -0,0 +1,33 @@ + + +> [!IMPORTANT] +> On June 26 2024, Linux Foundation announced the merger of its financial services umbrella, the Fintech Open Source Foundation ([FINOS](https://finos.org)), with OS-Climate, an open source community dedicated to building data technologies, modeling, and analytic tools that will drive global capital flows into climate change mitigation and resilience; OS-Climate projects are in the process of transitioning to the [FINOS governance framework](https://community.finos.org/docs/governance); read more on [finos.org/press/finos-join-forces-os-open-source-climate-sustainability-esg](https://finos.org/press/finos-join-forces-os-open-source-climate-sustainability-esg) + + + +# osc-physrisk-financial + +Physical climate risk financial valuation + +drawing + +## About osc-physrisk-financial + +An [OS-Climate](https://os-climate.org) project, osc-physrisk-financial +is a library for valuating assets under different climate risk scenarios. + +## Using the library + +The library can be run locally and is installed via: + +```shell +pip install osc-physrisk-financial +``` + +The library uses the output generated by the +[physrisk](https://github.com/os-climate/physrisk) library + +### Note + +This is the first stage of development, where the models are intentionally +simple, focusing on setting up the proper structure of the library. diff --git a/README.rst b/README.rst deleted file mode 100644 index 83f5efa..0000000 --- a/README.rst +++ /dev/null @@ -1,54 +0,0 @@ -💬 Important - -On June 26 2024, Linux Foundation announced the merger of its financial services umbrella, the Fintech Open Source Foundation (`FINOS `_), with OS-Climate, an open source community dedicated to building data technologies, modelling, and analytic tools that will drive global capital flows into climate change mitigation and resilience; OS-Climate projects are in the process of transitioning to the `FINOS governance framework `_; read more on `finos.org/press/finos-join-forces-os-open-source-climate-sustainability-esg `_ - - -.. These are examples of badges you might want to add to your README: - please update the URLs accordingly - - .. image:: https://api.cirrus-ci.com/github//osc-physrisk-financial.svg?branch=main - :alt: Built Status - :target: https://cirrus-ci.com/github//osc-physrisk-financial - .. image:: https://readthedocs.org/projects/osc-physrisk-financial/badge/?version=latest - :alt: ReadTheDocs - :target: https://osc-physrisk-financial.readthedocs.io/en/stable/ - .. image:: https://img.shields.io/coveralls/github//osc-physrisk-financial/main.svg - :alt: Coveralls - :target: https://coveralls.io/r//osc-physrisk-financial - .. image:: https://img.shields.io/pypi/v/osc-physrisk-financial.svg - :alt: PyPI-Server - :target: https://pypi.org/project/osc-physrisk-financial/ - .. image:: https://img.shields.io/conda/vn/conda-forge/osc-physrisk-financial.svg - :alt: Conda-Forge - :target: https://anaconda.org/conda-forge/osc-physrisk-financial - .. image:: https://pepy.tech/badge/osc-physrisk-financial/month - :alt: Monthly Downloads - :target: https://pepy.tech/project/osc-physrisk-financial - .. image:: https://img.shields.io/twitter/url/http/shields.io.svg?style=social&label=Twitter - :alt: Twitter - :target: https://twitter.com/osc-physrisk-financial - -.. image:: https://img.shields.io/badge/-PyScaffold-005CA0?logo=pyscaffold - :alt: Project generated with PyScaffold - :target: https://pyscaffold.org/ - -| - -====================== -osc-physrisk-financial -====================== - - - OS-Climate Python Project - - -A longer description of your project goes here... - - -.. _pyscaffold-notes: - -Note -==== - -This project has been set up using PyScaffold 4.5. For details and usage -information on PyScaffold see https://pyscaffold.org/. diff --git a/docs/assets.rst b/docs/assets.rst new file mode 100644 index 0000000..2fb9a81 --- /dev/null +++ b/docs/assets.rst @@ -0,0 +1,7 @@ +Assets +====== + +.. automodule:: osc_physrisk_financial.assets + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/conf.py b/docs/conf.py index a798c7b..3aaf850 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -9,7 +9,6 @@ import os import sys -import shutil # -- Path setup -------------------------------------------------------------- @@ -28,31 +27,31 @@ # setup.py install" in the RTD Advanced Settings. # Additionally it helps us to avoid running apidoc manually -try: # for Sphinx >= 1.7 - from sphinx.ext import apidoc -except ImportError: - from sphinx import apidoc +# try: # for Sphinx >= 1.7 +# from sphinx.ext import apidoc +# except ImportError: +# from sphinx import apidoc -output_dir = os.path.join(__location__, "api") -module_dir = os.path.join(__location__, "../src/osc_physrisk_financial") -try: - shutil.rmtree(output_dir) -except FileNotFoundError: - pass +# # output_dir = os.path.join(__location__, "api") +# module_dir = os.path.join(__location__, "../src/osc_physrisk_financial") +# try: +# shutil.rmtree(output_dir) +# except FileNotFoundError: +# pass -try: - import sphinx +# try: +# import sphinx - cmd_line = f"sphinx-apidoc --implicit-namespaces -f -o {output_dir} {module_dir}" +# cmd_line = f"sphinx-apidoc --implicit-namespaces -f -o {output_dir} {module_dir}" - args = cmd_line.split(" ") - if tuple(sphinx.__version__.split(".")) >= ("1", "7"): - # This is a rudimentary parse_version to avoid external dependencies - args = args[1:] +# args = cmd_line.split(" ") +# if tuple(sphinx.__version__.split(".")) >= ("1", "7"): +# # This is a rudimentary parse_version to avoid external dependencies +# args = args[1:] - apidoc.main(args) -except Exception as e: - print("Running `sphinx-apidoc` failed!\n{}".format(e)) +# apidoc.main(args) +# except Exception as e: +# print("Running `sphinx-apidoc` failed!\n{}".format(e)) # -- General configuration --------------------------------------------------- @@ -63,7 +62,6 @@ # coming with Sphinx (named 'sphinx.ext.*') or your custom ones. extensions = [ "sphinx.ext.autodoc", - "sphinx.ext.intersphinx", "sphinx.ext.todo", "sphinx.ext.autosummary", "sphinx.ext.viewcode", @@ -72,6 +70,8 @@ "sphinx.ext.ifconfig", "sphinx.ext.mathjax", "sphinx.ext.napoleon", + "sphinx_design", + "myst_parser", ] # Add any paths that contain templates here, relative to this directory. @@ -88,7 +88,18 @@ # General information about the project. project = "osc-physrisk-financial" -copyright = "2024, github-actions[bot]" +copyright = "2024, Arfima Dev" +author = "Arfima Dev" + +# Summary +autosummary_generate = True + +# Docstrings of private methods +autodoc_default_options = { + "members": True, + "undoc-members": True, + "private-members": False, +} # The version info for the project you're documenting, acts as replacement for # |version| and |release|, also used in various other places throughout the @@ -153,14 +164,15 @@ # The theme to use for HTML and HTML Help pages. See the documentation for # a list of builtin themes. -html_theme = "alabaster" - +# html_theme = "alabaster" +html_theme = "pydata_sphinx_theme" # Theme options are theme-specific and customize the look and feel of a theme # further. For a list of options available for each theme, see the # documentation. html_theme_options = { - "sidebar_width": "300px", - "page_width": "1200px" + # "sidebar_width": "300px", + # "page_width": "1200px", + "logo": {"text": "PhysRisk Financial"} } # Add any paths that contain custom themes here, relative to this directory. @@ -168,14 +180,14 @@ # The name for this set of Sphinx documents. If None, it defaults to # " v documentation". -# html_title = None +html_title = "Physrisk Financial Documentation" # A shorter title for the navigation bar. Default is the same as html_title. # html_short_title = None # The name of an image file (relative to this directory) to place at the top # of the sidebar. -# html_logo = "" +html_logo = "images/OS-Climate-Logo.png" # The name of an image file (within the static path) to use as favicon of the # docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32 @@ -196,7 +208,13 @@ # html_use_smartypants = True # Custom sidebar templates, maps document names to template names. -# html_sidebars = {} +html_sidebars = { + "readme": [], + "changelog": [], + "authors": [], + "contributing": [], + "license": [], +} # Additional templates that should be rendered to pages, maps page names to # template names. @@ -246,7 +264,13 @@ # Grouping the document tree into LaTeX files. List of tuples # (source start file, target name, title, author, documentclass [howto/manual]). latex_documents = [ - ("index", "user_guide.tex", "osc-physrisk-financial Documentation", "github-actions[bot]", "manual") + ( + "index", + "user_guide.tex", + "osc-physrisk-financial Documentation", + "Arfima Dev", + "manual", + ) ] # The name of an image file (relative to this directory) to place at the top of diff --git a/docs/dynamics.rst b/docs/dynamics.rst new file mode 100644 index 0000000..f8cb418 --- /dev/null +++ b/docs/dynamics.rst @@ -0,0 +1,7 @@ +Dynamics +======== + +.. automodule:: osc_physrisk_financial.dynamics + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/functions.rst b/docs/functions.rst new file mode 100644 index 0000000..c134705 --- /dev/null +++ b/docs/functions.rst @@ -0,0 +1,7 @@ +Functions +========= + +.. automodule:: osc_physrisk_financial.functions + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/images/OS-Climate-Logo.png b/docs/images/OS-Climate-Logo.png new file mode 100644 index 0000000000000000000000000000000000000000..36c887813b32182fed583ced2537fd869b4d1b70 GIT binary patch literal 109264 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 literal 0 HcmV?d00001 diff --git a/docs/index.rst b/docs/index.rst index 521f362..70f1038 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -1,40 +1,98 @@ +Physrisk Financial ====================== -osc-physrisk-financial -====================== -This is the documentation of **osc-physrisk-financial**. +- **version**: 0.1 (See `Changelog `_) +- **date**: July 4th 2024 + +An OS-Climate project, **osc-physrisk-financial** is a library for valuating assets under different climate risk scenarios. + +.. _cards-clickable: + +.. + list with all the possible icons for the grid + https://sphinx-design.readthedocs.io/en/latest/badges_buttons.html + +.. raw:: html + + + + +.. grid:: 2 + :gutter: 1 + + .. grid-item-card:: Overview + :link: readme.html + :text-align: center + + :octicon:`book;5em;sd-text-info` + ^^^ + Check the getting started guides and tutorials to learn how to install and use **osc-physrisk-financial**. + + .. grid-item-card:: Code documentation + :link: modules.html + :text-align: center -.. note:: + :octicon:`code;5em;sd-text-info` + ^^^ + Check the documentation of the code used in **osc-physrisk-financial**. - This is the main page of your project's `Sphinx`_ documentation. - It is formatted in `reStructuredText`_. Add additional pages - by creating rst-files in ``docs`` and adding them to the `toctree`_ below. - Use then `references`_ in order to link them from this page, e.g. - :ref:`authors` and :ref:`changes`. +.. grid:: 2 + :gutter: 1 - It is also possible to refer to the documentation of other Python packages - with the `Python domain syntax`_. By default you can reference the - documentation of `Sphinx`_, `Python`_, `NumPy`_, `SciPy`_, `matplotlib`_, - `Pandas`_, `Scikit-Learn`_. You can add more by extending the - ``intersphinx_mapping`` in your Sphinx's ``conf.py``. + .. grid-item-card:: Changelog + :link: changelog.html + :text-align: center - The pretty useful extension `autodoc`_ is activated by default and lets - you include documentation from docstrings. Docstrings can be written in - `Google style`_ (recommended!), `NumPy style`_ and `classical style`_. + :octicon:`list-ordered;5em;sd-text-info` + ^^^ + Check the history of the evolution of the code. + + .. grid-item-card:: Contributions & Help + :link: contributing.html + :text-align: center + + :octicon:`code-of-conduct;5em;sd-text-info` + ^^^ + If you want to contribute to the development take a look to the development guidelines first. + +.. grid:: 2 + :gutter: 1 + + .. grid-item-card:: Authors + :link: authors.html + :text-align: center + + :octicon:`people;5em;sd-text-info` + ^^^ + + .. grid-item-card:: License + :link: license.html + :text-align: center + + :octicon:`file-badge;5em;sd-text-info` + ^^^ Contents -======== +========== .. toctree:: :maxdepth: 2 Overview + Code documentation + Changelog Contributions & Help - License Authors - Changelog - Module Reference + License Indices and tables diff --git a/docs/license.rst b/docs/license.rst index 3989c51..725c6af 100644 --- a/docs/license.rst +++ b/docs/license.rst @@ -4,4 +4,5 @@ License ======= -.. include:: ../LICENSE.txt +.. include:: ../LICENSES/Apache-2.0.txt + :parser: myst_parser.sphinx_ diff --git a/docs/modules.rst b/docs/modules.rst new file mode 100644 index 0000000..fa53cac --- /dev/null +++ b/docs/modules.rst @@ -0,0 +1,46 @@ +Code documentation +================== + +.. grid:: 2 + :gutter: 1 + + .. grid-item-card:: Assets + :link: assets.html + :text-align: center + + :octicon:`list-unordered;5em;sd-text-info` + ^^^ + + .. grid-item-card:: Dynamics + :link: dynamics.html + :text-align: center + + :octicon:`pulse;5em;sd-text-info` + ^^^ + +.. grid:: 2 + :gutter: 1 + + .. grid-item-card:: Functions + :link: functions.html + :text-align: center + + :octicon:`tools;5em;sd-text-info` + ^^^ + + .. grid-item-card:: Random Variables + :link: random_variables.html + :text-align: center + + :octicon:`graph;5em;sd-text-info` + ^^^ + + +.. toctree:: + :maxdepth: 1 + :hidden: + + assets + dynamics + functions + random_variables diff --git a/docs/random_variables.rst b/docs/random_variables.rst new file mode 100644 index 0000000..bc9efd6 --- /dev/null +++ b/docs/random_variables.rst @@ -0,0 +1,7 @@ +Random Variables +================ + +.. automodule:: osc_physrisk_financial.random_variables + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/readme.rst b/docs/readme.rst index 81995ef..6a59a1a 100644 --- a/docs/readme.rst +++ b/docs/readme.rst @@ -1,2 +1,29 @@ -.. _readme: -.. include:: ../README.rst +Overview +====================== + +osc-physrisk-financial +---------------------- +Physical climate risk financial valuation + +.. image:: images/OS-Climate-Logo.png + :alt: drawing + :width: 150 + +About osc-physrisk-financial +---------------------------- + +An `OS-Climate `_ project, osc-physrisk-financial is a library for valuating assets under different climate risk scenarios. + +Using the library +----------------- + +The library can be run locally and is installed via:: + + pip install osc-physrisk-financial + +The library uses the output generated by the `physrisk `_ library. + +Note +---- + +This is the first stage of development, where the models are intentionally simple, focusing on setting up the proper structure of the library. diff --git a/docs/requirements.txt b/docs/requirements.txt deleted file mode 100644 index 2ddf98a..0000000 --- a/docs/requirements.txt +++ /dev/null @@ -1,5 +0,0 @@ -# Requirements file for ReadTheDocs, check .readthedocs.yml. -# To build the module reference correctly, make sure every external package -# under `install_requires` in `setup.cfg` is also listed here! -sphinx>=3.2.1 -# sphinx_rtd_theme diff --git a/pyproject.toml b/pyproject.toml index b4025d7..c998c0a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -2,38 +2,141 @@ requires = ["pdm-backend"] build-backend = "pdm.backend" -[tool.setuptools_scm] -# For smarter version schemes and other configuration options, -# check out https://github.com/pypa/setuptools_scm -version_scheme = "no-guess-dev" - [tool.pdm] package-dir = "src" [project] name = "osc-physrisk-financial" -description = "OS-Climate Python Project" -readme = "README.rst" +description = "Physical climate risk financial valuation" +readme = "README.md" +dynamic = ["version"] +keywords = ["Financial risk", "climate", "Physical risk"] authors = [ - {name = "github-actions[bot]", email = "41898282+github-actions[bot]@users.noreply.github.com"}, + {name = "Arfima Dev", email = "dev@arfima.com"}, ] classifiers = [ "Development Status :: 4 - Beta", "Programming Language :: Python", ] dependencies = [ - "importlib-metadata; python_version<\"3.8\"", + "scipy>=1.10.1", + "pandas>=2.0.3", + "plotly>=5.15", + "numpy>=1.24", ] license = {text = "Apache-2.0"} requires-python = ">=3.10" [project.urls] -Homepage = "https://github.com/pyscaffold/pyscaffold/" -Documentation = "https://pyscaffold.org/" +"Homepage" = "https://github.com/os-climate/osc-physrisk-financial" +"Documentation" = "https://github.com/os-climate/osc-physrisk-financial" +"Bug Tracker" = "https://github.com/os-climate/osc-physrisk-financial/issues" + [project.optional-dependencies] -testing = [ +docs = [ + "myst_parser>=3.0.1", + "pydata_sphinx_theme>=0.15.4", + "sphinx>=7.3.7", + "sphinx_design>=0.6.0" +] +lint = ["pre-commit"] +test = [ + "pdm[pytest]", "pytest", "pytest-cov", - "setuptools", ] +tox = ["tox", "tox-pdm"] + +[tool.pdm.version] +source = "scm" +write_to = "osc_physrisk_financial/_version.py" +write_template = "version: str\n__version__: str\n__version__ = version = '{}'\n" +# Semver like tag, ignore after + or - +# tag_regex = '^(?:[\w-]+-)?(?P[vV]?\d+(?:\.\d+){0,2}[^\+-]*?)(?:\-.*)?(?:\+.*)?(?:\-.*)?$' +# PyPa-compliant (Removing trailing (?:\.dev(?:\d+))? as it would be handled by PDM) +tag_regex = '^(?P[vV]??(?:(?:\d+)(?:\.(?:\d+))*)(?:(?:a|b|rc|alpha|beta|c)(?:\d+)?)?(?:\.post(?:\d+))?)$' + +[tool.pytest.ini_options] +testpaths = [ + "tests/", +] +addopts = "--cov=src --cov-report html --cov-report term-missing --cov-fail-under 65" + +[tool.coverage.run] +source = ["src"] + +[tool.mypy] +ignore_missing_imports = true + + +[tool.ruff] +# Exclude a variety of commonly ignored directories. +exclude = [ + ".bzr", + ".direnv", + ".eggs", + ".git", + ".git-rewrite", + ".hg", + ".ipynb_checkpoints", + ".mypy_cache", + ".nox", + ".pants.d", + ".pyenv", + ".pytest_cache", + ".pytype", + ".ruff_cache", + ".svn", + ".tox", + ".venv", + ".vscode", + "__pypackages__", + "_build", + "buck-out", + "build", + "dist", + "node_modules", + "site-packages", + "venv", +] + +[tool.ruff.lint] +extend-fixable = [ + # Instead of trailing-whitespace + "W291", "W293" + ] + +extend-select = [ + # Instead of pydocstyle + "D", + #Instead of flake8 + "E", "F","B", + # Instead of pep8-naming + "N", + # Instead of flake8-debugger or debug-statements + "T10", +] + +ignore = [ + "E203", + "E501", + # Avoid incompatible rules + "D203", + "D213", +] + +[tool.ruff.lint.extend-per-file-ignores] +# Ignore `D` rules everywhere except for the `src/` directory. +"!src/**.py" = ["D"] + +[tool.ruff.lint.pycodestyle] +max-line-length = 160 + +[tool.ruff.format] +quote-style = "double" +indent-style = "space" +skip-magic-trailing-comma = false +line-ending = "auto" +docstring-code-format = false +docstring-code-line-length = "dynamic" diff --git a/setup.sh b/setup.sh deleted file mode 100755 index 8c5eeac..0000000 --- a/setup.sh +++ /dev/null @@ -1,139 +0,0 @@ -#!/usr/bin/env bash - -# SPDX-License-Identifier: Apache-2.0 -# Copyright 2024 The Linux Foundation - -### Script to bootstrap the OS-Climate DevOps environment ### - -set -eu -o pipefail -# set -xv - -### Variables ### - -SOURCE_FILE="bootstrap.yaml" -WGET_URL="https://raw.githubusercontent.com/os-climate/devops-toolkit/main/.github/workflows/$SOURCE_FILE" -AUTOMATION_BRANCH="update-devops-tooling" -DEVOPS_DIR=".devops" -FETCH_MODE="wget" - -### Checks ### - -GIT_CMD=$(which git) -if [ ! -x "$GIT_CMD" ]; then - echo "GIT command was NOT found in PATH"; exit 1 -fi - -WGET_CMD=$(which wget) -if [ ! -x "$WGET_CMD" ]; then - echo "WGET command was NOT found in PATH; using CURL" - FETCH_MODE="curl" -fi - -MKTEMP_CMD=$(which mktemp) -if [ ! -x "$MKTEMP_CMD" ]; then - echo "MKTEMP command was NOT found in PATH"; exit 1 -fi - - - -SHELL_SCRIPT=$(mktemp -t script-XXXXXXXX.sh) - -### Functions ### - -change_dir_error() { - echo "Could not change directory"; exit 1 -} - -check_for_local_branch() { - BRANCH="$1" - git show-ref --quiet refs/heads/"$BRANCH" - return $? -} - -check_for_remote_branch() { - BRANCH="$1" - git ls-remote --exit-code --heads origin "$BRANCH" - return $? -} - -cleanup_on_exit() { - # Remove PR branch, if it exists - echo "Cleaning up on exit: bootstrap.sh" - echo "Swapping from temporary branch to: $HEAD_BRANCH" - git checkout main > /dev/null 2>&1 - if (check_for_local_branch "$AUTOMATION_BRANCH"); then - echo "Removing temporary local branch: $AUTOMATION_BRANCH" - git branch -d "$AUTOMATION_BRANCH" > /dev/null 2>&1 - fi - if [ -f "$SHELL_SCRIPT" ]; then - echo "Removing temporary shell code" - rm "$SHELL_SCRIPT" - fi - if [ -d "$DEVOPS_DIR" ]; then - echo "Removed local copy of devops repository" - rm -Rf "$DEVOPS_DIR" - fi -} -trap cleanup_on_exit EXIT - -### Main script entry point - -# Get organisation and repository name -# git config --get remote.origin.url -# git@github.com:ModeSevenIndustrialSolutions/test-bootstrap.git -URL=$(git config --get remote.origin.url) - -# Take the above and store it converted as ORG_AND_REPO -# e.g. ModeSevenIndustrialSolutions/test-bootstrap -ORG_AND_REPO=${URL/%.git} -ORG_AND_REPO=${ORG_AND_REPO//:/ } -ORG_AND_REPO=$(echo "$ORG_AND_REPO" | awk '{ print $2 }') -HEAD_BRANCH=$("$GIT_CMD" rev-parse --abbrev-ref HEAD) -REPO_DIR=$(git rev-parse --show-toplevel) -# Change to top-level of GIT repository -CURRENT_DIR=$(pwd) -if [ "$REPO_DIR" != "$CURRENT_DIR" ]; then - echo "Changing directory to: $REPO_DIR" - cd "$REPO_DIR" || change_dir_error -fi - -# Get latest copy of bootstrap workflow -if [ -f "$SOURCE_FILE" ]; then - echo "Removing existing copy of: $SOURCE_FILE" - rm "$SOURCE_FILE" -fi -echo "Pulling latest DevOps bootstrap YAML from:" -echo " $WGET_URL" -if [ "$FETCH_MODE" = "wget" ]; then - "$WGET_CMD" -q "$WGET_URL" > /dev/null 2>&1 -fi -if [ ! -f "$SOURCE_FILE" ]; then - echo "Attempting to retrieve YAML file with CURL" - curl "$WGET_URL" > "$SOURCE_FILE" -fi - -# The section below extracts shell code from the YAML file -echo "Extracting shell code from: $SOURCE_FILE" -EXTRACT="false" -while read -r LINE; do - if [ "$LINE" = "#SHELLCODESTART" ]; then - EXTRACT="true" - SHELL_SCRIPT=$(mktemp -t script-XXXXXXXX.sh) - touch "$SHELL_SCRIPT" - chmod a+x "$SHELL_SCRIPT" - echo "Creating shell script: $SHELL_SCRIPT" - echo "#!/bin/sh" > "$SHELL_SCRIPT" - fi - if [ "$EXTRACT" = "true" ]; then - echo "$LINE" >> "$SHELL_SCRIPT" - if [ "$LINE" = "#SHELLCODEEND" ]; then - break - fi - fi -done < "$SOURCE_FILE" - -echo "Running extracted shell script code" -# https://www.shellcheck.net/wiki/SC1090 -# Shell code executed is temporary and cannot be checked by linting -# shellcheck disable=SC1090 -. "$SHELL_SCRIPT" diff --git a/src/osc_physrisk_financial/__init__.py b/src/osc_physrisk_financial/__init__.py index a65eb6f..25547bc 100644 --- a/src/osc_physrisk_financial/__init__.py +++ b/src/osc_physrisk_financial/__init__.py @@ -1,16 +1 @@ -import sys - -if sys.version_info[:2] >= (3, 8): - # TODO: Import directly (no need for conditional) when `python_requires = >= 3.8` - from importlib.metadata import PackageNotFoundError, version # pragma: no cover -else: - from importlib_metadata import PackageNotFoundError, version # pragma: no cover - -try: - # Change here if project is renamed and does not equal the package name - dist_name = "osc-physrisk-financial" - __version__ = version(dist_name) -except PackageNotFoundError: # pragma: no cover - __version__ = "unknown" -finally: - del version, PackageNotFoundError +"""Init for osc-physrisk.""" diff --git a/src/osc_physrisk_financial/assets.py b/src/osc_physrisk_financial/assets.py new file mode 100644 index 0000000..84d3a7e --- /dev/null +++ b/src/osc_physrisk_financial/assets.py @@ -0,0 +1,342 @@ +"""Assets definitions.""" + +from typing import Optional, Sequence, Union + +import numpy as np +import pandas as pd + +import osc_physrisk_financial.functions as afsfun +from osc_physrisk_financial.dynamics import Dynamic +from osc_physrisk_financial.random_variables import DiscreteRandomVariable + + +class Asset(object): + """Class for instantiating a general Asset. + + Parameters + ---------- + value_0 : float + Value of the asset at 0, :math:`V_{0}` + + dynamics : dynamics.Dynamic + Dynamics assumed for the asset value. + + name : string, optional + Name for identification. + + cash_flows: Sequence, optional + Sequence of the associated cash flows (for cash flow generating assets only). + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + + # TODO: This is not the final parameters list. Check OS-C (assets.py) + # TODO: we should include latitude: float, longitude: float. + + def __init__( + self, + value_0: float, + dynamics: Optional[Dynamic] = None, + name: Optional[str] = None, + cash_flows: Optional[Sequence] = None, + ): + """Initialize the AssetClass with dynamics and name. + + `dynamics` or `name` are optional. 'value_0 must be provided' + + Parameters + ---------- + value_0 : float + Initial value + dynamics : Optional[Dynamic] = None + Asset value dynamics. + name : Optional[str] = None + Asset name. + cash_flows: Optional[Sequence] = None + Cash flows. + + """ + self.value_0 = value_0 + self.dynamics = dynamics + self.name = name # TODO: Not sure if this is useful. + self.cash_flows = cash_flows + + # TODO: Maybe here we can use OS-C standard: + # class Asset: + # def __init__(self, latitude: float, longitude: float, **kwargs): + # self.latitude = latitude + # self.longitude = longitude + # self.__dict__.update(kwargs) + + +class RealAsset(Asset): + """Class for instantiating a Real Asset. + + Parameters + ---------- + value_0 : float + Value of the asset at 0, :math:`V_{0}` + + dynamics : dynamics.Dynamic + Dynamics assumed for the asset value. + + name : string, optional + Name for identification. + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + + def __init__(self, value_0: float, dynamics: Dynamic, name: Optional[str] = None): + """Initialize the RealAssetClass with dynamics and name. + + `dynamics` or `name` are optional. 'value_0 must be provided' + + Parameters + ---------- + value_0 : float + Initial value + dynamics : Optional[Dynamic] = None + Asset value dynamics. + name : Optional[str] = None + Asset name. + + """ + super().__init__(value_0=value_0, dynamics=dynamics, name=None) + + def financial_losses( + self, dates: Union[pd.DatetimeIndex, list], damage: DiscreteRandomVariable + ): + """Compute financial losses for a real asset. + + Parameters + ---------- + dates : pandas.DatetimeIndex, list of strings, pandas.Timestamp, or string + Dates for which we want to compute :math:`X_t` of [Methodology]. TODO: Do we want to include t_0 here? + damage : random_variables.RandomVariable + Damage caused to the asset. + + Returns + ------- + random_variables.RandomVariable + Random variable representing :math:`X_{t}` [Methodology]. + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + if self.dynamics is None: + raise ValueError("Dynamics must be provided.") + dates = afsfun.dates_formatting(dates) + value_t = self.dynamics.compute_value(dates) + losses = value_t * damage + return losses + + def ltv( + self, + dates: Union[pd.DatetimeIndex, list], + damages: Sequence[DiscreteRandomVariable], + loan_amounts: Sequence[float], + ): + r"""Compute Loan To Value (LTV) for a real asset. + + Parameters + ---------- + dates : pandas.DatetimeIndex, list of strings, pandas.Timestamp, or string + Dates for which we want to compute :math:`X_t` of [Methodology]. + Note that :math: `t_0` should be included here. # TODO: Do we want to include t_0 here? + damages : Sequence[DiscreteRandomVariable] + Sequence of DiscreteRandomVariable instances representing damage for each asset. + loan_amounts : Sequence[float] + Sequence of floats representing loan amount for each asset. + + Returns + ------- + random_variables.RandomVariable + Random variable representing LTV of [Methodology]. + It returns a numpy.ndarray of 2 dimensions and shape :math:`(\\# dates, \\# assets)`. + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + # Define a function to apply the check to an array of DiscreteRandomVariable instances + + def validate_values(drvs: Sequence[DiscreteRandomVariable]): + # Vectorize check_values method + vec_check = np.vectorize(lambda drv: drv.check_values()) + values_valid = vec_check(drvs) + + if not np.all(values_valid): + raise ValueError( + "One or more damages have values outside the 0 to 1 range." + ) + + validate_values(damages) + + if len(damages) != len(loan_amounts): + raise ValueError( + "The lengths of 'damage' and 'loan_amount' (number of assets) must match." + ) + # We reshape for allowing broadcasting + if self.dynamics is None: + raise ValueError("Dynamics must be provided.") + valuet = self.dynamics.compute_value(dates=dates).reshape((len(dates), 1)) + damages_np = np.array(damages) + damages_mod = (1 + (-1) * damages_np).reshape( + 1, len(damages_np) + ) # Note that __sub__ is not needed in class DiscreteRandomVariable. + valuet_sc = valuet * damages_mod + return loan_amounts / valuet_sc + + # TODO: Maybe it is interesting to vectorize the computation of the mean and variance of the LTVs computed by leveraging numpy. + # Check impact_distrib.py from OS-C. As it stands, we can do it using np.vectorize (see methods means_vectorized and means_vectorized) + # but we know it is not efficient (essentially a for loop) + # https://numpy.org/doc/stable/reference/generated/numpy.vectorize.html#:~:text=returns%20a%20ufunc-,Notes,-The%20vectorize%20function + + +class PowerPlants(Asset): + """Class for instantiating a PowerPlant Asset. + + Either `production` or both `capacity` and `av_rate` must be provided. If not directly provided, `production`, is calculated as: + `production` = `capacity` * `av_rate` * 8760. + + Parameters + ---------- + dynamics : dynamics.Dynamic + Dynamics assumed for the asset value. + + name : string, optional + Name for identification. + + production : float, optional + Real annual production of a power plant in Wh. + + capacity : float, optional + Capacity of the power plant in W. + + av_rate : float, optional + Availability factor of production. + + References + ---------- + `Canonical_Example_Power_Generation_Plants_Floods` (Overleaf). + + Notes + ----- + In this case the cash flows are defined through production. + + """ + + def __init__( + self, dynamics: Optional[Dynamic] = None, name: Optional[str] = None, **kwargs + ): + """Initialize the PowerPlantsClass with dynamics, name and a variable number of arguments. + + `dynamics` or `name` are optional. 'value_0 must be provided' + + Parameters + ---------- + dynamics : dynamics.Dynamic + Dynamics assumed for the asset value. + name : string, optional + Name for identification. + **kwargs : dict + Variable number of arguments. + + """ + if "production" in kwargs: + production = kwargs["production"] + # If not, check if capacity and av_rate are both provided + elif "capacity" in kwargs and "av_rate" in kwargs: + production = ( + kwargs["capacity"] * kwargs["av_rate"] * 8760 + ) # Number of hours in a year + else: + raise ValueError( + "Must provide either 'production' or both 'capacity' and 'av_rate'." + ) + super().__init__( + value_0=production, dynamics=dynamics, name=None, cash_flows=None + ) + + @staticmethod + def discount(r: Sequence[float], n: Optional[int] = 1) -> float: + r"""Compute discount for a given annual evolution of interest rates. + + Parameters + ---------- + r : Sequence[float] + An array or sequence including the yearly interest rate for the required period. + + n : int, optional + By default r is a list containing the yearly interest rates. + To consider a constant interest rate, introduce the value of the + interest rate in r and n = number of years to be discounted. + + Returns + ------- + float + Float containing the discounting factor calculated as + :math:`\prod_{i} 1/(1+r_i)^n`. + + """ + if n is not None and n < 1: + raise ValueError("Discounting cash flows in negative number of year") + + if len(r) > 1 and n != 1: + raise ValueError("Discounting cash flows has a wrong format") + + aux = np.array(r) + 1 + disc = 1 / np.prod(aux) ** n + + return disc + + def financial_losses( + self, + damages: DiscreteRandomVariable, + energy_price: float, + r: Sequence[float], + n: Optional[int] = 1, + ) -> DiscreteRandomVariable: + r"""Compute financial losses for a PowerPlant asset. + + Parameters + ---------- + damages : DiscreteRandomVariable + Random Variable with the production loss expressed as a decimal (50% :math:`\equiv` 0.5) for each plant. + + energy_price : float + Average price in €/Wh of the energy production. + + r : list[float] + An array or sequence containing the annual interest rates. + + n : int, optional + Number of years to discount. + + Returns + ------- + DiscreteRandomVariable + Random Variable containing the financial losses for the asset. + + Notes + ----- + The use of `r` and `n` follows the same convention as in `discount` method. + + + """ + scaled_values = ( + damages.values * self.value_0 * energy_price * self.discount(r, n) + ) + res = DiscreteRandomVariable( + values=scaled_values, probabilities=damages.probabilities.tolist() + ) + return res diff --git a/src/osc_physrisk_financial/dynamics.py b/src/osc_physrisk_financial/dynamics.py new file mode 100644 index 0000000..25a9a2f --- /dev/null +++ b/src/osc_physrisk_financial/dynamics.py @@ -0,0 +1,121 @@ +"""Dynamics.""" + +from abc import ABC, abstractmethod +from typing import Optional, Union + +import numpy as np +import pandas as pd + +import osc_physrisk_financial.functions as afsfun + + +class Dynamic(ABC): + """A base class for simulating asset value dynamics. + + Notes + ----- + This base class is based on Underlying from pypricing. + + """ + + def __init__(self, name: Optional[str] = None): + """Initialize a new instance of Dynamic. + + Attributes + ---------- + name : string, optional + Name for identification. + + """ + self.name = name + self.data = pd.DataFrame() + + @abstractmethod + def compute_value(self, dates: Union[pd.DatetimeIndex, list]): + """Abstract method for computing the asset value at future dates. + + Attributes + ---------- + dates : pandas.DatetimeIndex, list of strings, pandas.Timestamp, or string + Future dates for which the asset value wants to be computed. + + Notes + ----- + This base class is based on Underlying from pypricing. + + """ + + # TODO: Maybe we can use methods like set_data, get_data, get_value, get_dates, get_arithmetic_return, get_return from Underlying. + # TODO: We have to think about this while developing the code. + + +class ConstantGrowth(Dynamic): + r"""Class representing a constant growth model: :math:`V_t = V_0 \\times (1 + \mu)^t.`. + + Parameters + ---------- + growth_rate : float + Constant growth rate :math:`\mu.` + + name : string, optional + Name for identification. + + value0 : float + :math:`V_0` in [Methodology] + + Examples + -------- + >>> cg = ConstantGrowth(growth_rate=0.02, name='RealAsset') + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + + def __init__(self, growth_rate: float, value0: float, name: Optional[str] = None): + r"""Initialize a new instance of ConstantGrowth. + + Attributes + ---------- + growth_rate : float + Constant growth rate :math:`\mu.` + + value0 : float + :math:`V_0` in [Methodology] + + name : string, optional + Name for identification. + + """ + super().__init__(name=name) + self.growth_rate = growth_rate + self.value0 = value0 + + def compute_value(self, dates: Union[pd.DatetimeIndex, list]): + """Compute the asset value at future dates. + + Attributes + ---------- + dates : pandas.DatetimeIndex, list of strings, pandas.Timestamp, or string + Dates for which the value wants to be computed. Note that in this model we are only + interested in the years, so we only extract that part. The initial date is also included + here ( :math:`t_{0}` such that :math:`V_{t_0} = V_0` of [Methodology]. + + Returns + ------- + np.ndarray + :math:`V_t` in [Methodology] for the different dates. It includes the value :math:`V_0`. + Note that the dates have been sorted and the output is returned with the dates sorted. + + References + ---------- + Methodology, Chapter 4 of Methodology survey (Overleaf). + + """ + dates = afsfun.dates_formatting(dates) + dates = pd.to_datetime(dates) + years = dates.year + years = years - years[0] + valuet = self.value0 * (1 + self.growth_rate) ** years + return np.array(valuet) diff --git a/src/osc_physrisk_financial/functions.py b/src/osc_physrisk_financial/functions.py new file mode 100644 index 0000000..ed9c862 --- /dev/null +++ b/src/osc_physrisk_financial/functions.py @@ -0,0 +1,172 @@ +"""Auxiliary functions.""" + +import math + +import numpy as np +import pandas as pd +import plotly.graph_objects as go +from scipy import optimize + +# TODO: We should make pypricing library installable so we can import pypricing.pricing.functions.py. +# TODO: Meanwhile we have copied this file in this repository. + + +def check_all_nonnumeric(arr): + """Check if all elements in a numpy array or Python list are non-numeric. + + This function tries to convert each element in the array or list to a float. If the + conversion raises a ValueError or TypeError, or if the value is nan, it means the + element is non-numeric, so the function continues to the next element. If the + conversion does not raise an exception and the value is not nan, it means the element + is numeric, so the function immediately returns False. If the function finishes + checking all elements without finding a numeric one, it returns True. + + Parameters + ---------- + arr : numpy.ndarray or list + The array or list to check. + + Returns + ------- + bool + True if all elements are non-numeric, False otherwise. + + """ + for i in arr: + try: + val = float(i) + if not math.isnan(val): + return False + except (ValueError, TypeError): + continue + return True + + +def find_root(func, x0, interval, tolerance=10**-8, fprime=None): + """Find the root of a given function, using several methods. + + Each method is tried in turn until one succeeds. + + If none succeeds, we plot the function in interval. + + Parameters + ---------- + func : callable + The function for which the root is to be computed. + x0 : float + Initial guess for the root. + interval : list + Interval [a,b] for ridder, bisecction and brentq. + tolerance : float + If func(solution)>tolerance an exception is raised. + fprime : callable, optional + The derivative of the function. If not provided, the Newton method will use the secant method. + + Returns + ------- + float + The root found by the successful method (unless all methods failed). + + """ + methods = [ + ("fixed_point", optimize.fixed_point), + ("newton (Secant)", optimize.newton), + ("newton (Newton-Raphson)", optimize.newton), + ("bisection", optimize.bisect), + ("brentq", optimize.brentq), + ("ridder", optimize.ridder), + ] + + root = None + + for name, method in methods: + try: + if name == "fixed_point": + root = method(lambda x: x - func(x), x0) + elif name == "newton (Secant)": + root = method(func, x0, fprime=None) + elif name == "newton (Newton-Raphson)": + root = method(func, x0, fprime=fprime) + else: + root = method(func, interval[0], interval[1]) + # print(f"Method {name} succeeded with root {root}") + break # if method succeeded, stop trying the rest + except Exception: + pass + if root is None: + x_vals = np.linspace(interval[0], interval[1], 200) + y_vals = [ + func(x) for x in x_vals + ] # Note that this code snippet is intentionally not vectorized. + fig = go.Figure(data=go.Scatter(x=x_vals, y=y_vals)) + fig.update_layout(title="Plot of func", xaxis_title="x", yaxis_title="y") + fig.show() + raise Exception("All methods failed") + else: + if np.abs(func(root)) > tolerance: # Maybe another tolerance can be chosen. + raise Exception("The numerical error is too large.") + else: + return root + + +def dates_formatting(*date_sets): + """Convert dates to a consistent format and sort them in ascending order. + + Parameters + ---------- + date_sets : pandas.DatetimeIndex,list of strings, pandas.Timestamp, or string + Dates to be formatted. It can be a single date (as a pandas.Timestamp or its string representation) + or an array-like object (as a pandas.DatetimeIndex or a list of its string representation) containing dates. + + Returns + ------- + sorted_dates : pandas.DatetimeIndex + A pandas DatetimeIndex object containing the formatted dates in ascending order. + + Examples + -------- + >>> dates_formatting('2022-01-01') + DatetimeIndex(['2022-01-01'], dtype='datetime64[ns]', freq=None) + + >>> dates_formatting(['2022-01-03', '2022-01-01', '2022-01-02']) + DatetimeIndex(['2022-01-01', '2022-01-02', '2022-01-03'], dtype='datetime64[ns]', freq=None) + + >>> dates_formatting(['2022-01-03', '2022-01-01', '2022-01-02'], '2022-01-01') + [DatetimeIndex(['2022-01-01', '2022-01-02', '2022-01-03'], dtype='datetime64[ns]', freq=None), DatetimeIndex(['2022-01-01'], dtype='datetime64[ns]', freq=None)] + + """ + formatted_dates = [] + for date_set in date_sets: + if np.asarray(date_set).shape == (): + dates = [date_set] + else: + dates = date_set + formatted_dates.append(pd.to_datetime(dates).sort_values()) + if len(formatted_dates) == 1: + formatted_dates = formatted_dates[0] + + return formatted_dates + + +def contains_word(string_list, word): + """Check if strings in the given list contain the specified word. Words in each string are separated by underscores. + + Parameters + ---------- + string_list : list of str + The list of strings where each string has words separated by underscores. + word : str + The word to search for within the strings. + + Returns + ------- + list of str + A list of strings from the input `string_list` that contain the specified `word`. + + Examples + -------- + >>> contains_word(['word1_word2', 'word3_word4', 'word2_word5'], 'word2') + ['word1_word2', 'word2_word5'] + + """ + return [s for s in string_list if word in s.split("_")] diff --git a/src/osc_physrisk_financial/random_variables.py b/src/osc_physrisk_financial/random_variables.py new file mode 100644 index 0000000..90f0edf --- /dev/null +++ b/src/osc_physrisk_financial/random_variables.py @@ -0,0 +1,1033 @@ +"""functions for random and discrete random variables.""" + +from abc import ABC, abstractmethod +from typing import Optional, Union, Sequence, Any + +import numpy as np +import plotly.graph_objects as go + + +class RandomVariable(ABC): + """Abstract class with the common methods and attributes of discrete and continuous random variables. + + Ideally, we wouldn't have to implement this class from scratch, but an initial search seems to indicate + that what we want doesn't exist in another libraries (like SciPy). + """ + + @abstractmethod + def __init__(self): + """Initialize a RandomVariable.""" + + @abstractmethod + def __mul__(self, other: Union[float, int]): + """Multiply the random variable by a real number. Case RandomVariable * real number. + + This method scales the pdf or pmf of the random variable by a given scalar + while keeping the probabilities unchanged. + + Parameters + ---------- + other : float, or int + The scalar by which to multiply the pdf or pmf of the random variable. + + Returns + ------- + RandomVariable + A new instance of DiscreteRandomVariable with scaled pdf or pmf. + + Notes + ----- + We define this class since operations like the ones defined are not implemented in scipy. + For instance: TypeError: unsupported operand type(s) for *: 'int' and 'rv_sample'. + + """ + + def __rmul__(self, other: Union[float, int]): + """Multiply the random variable by a real number. Case real number * RandomVariable. + + This method delegates to `__mul__`, assuming commutativity of the operation. + + Parameters + ---------- + other : float, or int + The real number by which to multiply the random variable. + + Returns + ------- + RandomVariable + A new instance of DiscreteRandomVariable with scaled pdf or pmf. + + """ + return self.__mul__(other) + + def __neg__(self): + """Negate the random variable.""" + return self.__mul__(-1) + + @abstractmethod + def __add__(self, other: Union[float, int]): + """Add a real number to the random variable. Case RandomVariable + real number. + + This method shifts the pdf or pmf of the random variable by a given number + while keeping the probabilities unchanged. + + Parameters + ---------- + other : float, or int + The real number to add to the pdf or pmf of the random variable. + + Returns + ------- + RandomVariable + A new instance of DiscreteRandomVariable with shifted pdf or pmf. + + """ + + def __radd__(self, other): + """Add a real number from the random variable. Case real number + RandomVariable. + + This method is called if the first operand does not support addition + or returns NotImplemented. It allows commutative addition where the scalar + is on the left side of the `+`. + + Parameters are the same as __add__. + """ + # __add__ handles the actual operation, so we just delegate to it. + return self.__add__(other) + + def __sub__(self, other): + """Subtract a real number to the random variable. Case RandomVariable - real number. + + __add__ handles the actual operation, so we just delegate to it. + + Parameters are the same as __add__. + """ + return self.__add__(-other) + + def __rsub__(self, other): + """Subtract the random variable from a real number. Case real number - RandomVariable. + + __add__ and __mul__ handle the actual operation, so we just delegate to them. + + Parameters are the same as __add__. + """ + return self.__mul__(-1).__add__(other) + + @abstractmethod + def __rtruediv__(self, other): + """Implement division where a real number is divided by a DiscreteRandomVariable. + + Parameters + ---------- + other : float, or int + The real number numerator. + + Returns + ------- + RandomVariable: A new instance representing the result. + + Raises + ------ + ValueError: If division by any value of the DiscreteRandomVariable is not possible. + + """ + + @abstractmethod + def __eq__(self, other: Any) -> bool: + """Check if the current instance equals another instance of a RandomVariable. + + Parameters + ---------- + other : Any + The object to compare against. + + Returns + ------- + bool + True if the objects are considered equal, False otherwise. + + """ + + @abstractmethod + def mean(self): + """Calculate the mean of the random variable. + + Returns + ------- + float + The mean of the random variable. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + @staticmethod + @abstractmethod + def means_vectorized(rvs: Sequence["RandomVariable"]) -> np.ndarray: + """Abstract static method to compute means for an array of RandomVariable instances using a vectorized approach. + + Parameters + ---------- + rvs : Sequence[RandomVariable] + An array or sequence of RandomVariable instances. + + Returns + ------- + np.ndarray + An array of floats representing the means of the random variables. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + @abstractmethod + def var(self): + """Calculate the variance of the random variable. + + Returns + ------- + float + The variance of the discrete random variable. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + @staticmethod + @abstractmethod + def vars_vectorized(rvs: Sequence["RandomVariable"]) -> np.ndarray: + """Abstract static method to compute variances for an array of RandomVariable instances using a vectorized approach. + + Parameters + ---------- + rvs : Sequence[RandomVariable] + An array or sequence of RandomVariable instances. + + Returns + ------- + np.ndarray + An array of floats representing the variances of the random variables. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + @abstractmethod + def compute_cdf(self): + """Compute the Cumulative Distribution Function (CDF) for the random variable.""" + + @abstractmethod + def compute_var(self, percentile=95): + r"""Compute the Value at Risk :math:`V^{p}_{X}` for a random variable :math:`X`. + + The Value at Risk (:math:`V^{p}_{X}`) of a discrete random variable :math:`X` at the level + :math:`p \in (0, 1)` is the p-quantile of :math:`X` defined by the condition that the cumulative + distribution function :math:`F_{X}(x)` is greater than or equal to :math:`p`. Formally, + :math:`V^{p}_{X}` is given by: + + .. math:: V^{p}_{X} := \inf\{x \in \mathbb{R} : P(X \leq x) \geq p\}. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + @staticmethod + @abstractmethod + def compute_var_vectorized(rvs): + """Compute VaRs for an array of RandomVariable instances using a vectorized approach. + + Parameters + ---------- + rvs : Sequence[RandomVariable] + An array or sequence of RandomVariable instances. + + Returns + ------- + np.ndarray + An array of floats representing the VaRs of the random variables. + + Notes + ----- + This is an abstract method and must be implemented by subclasses. + + """ + + +class DiscreteRandomVariable(RandomVariable): + """A class to represent a discrete random variable derived from observed data. + + Parameters + ---------- + probabilities : array like + The probabilities associated with each interval or value in the histogram. + values : array like, optional + The specific values representing the discrete random variable. Required if `intervals` is not provided. + intervals : array like, optional + The intervals (bins) of the histogram representing the discrete random variable. Required if `values` is not provided. + convert_to_osc_format : bool, optional + If True, it ensures that the probabilities sum to 1 by adjusting the zero-impact bin. + This is needed for `ImpactDistrib` from OS-C. Default, False. + + Examples + -------- + Values Example: + + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] # This should sum up to 1 + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + + Intervals Example: + + >>> intervals = [0, 0.2, 0.4, 0.6, 0.8, 1.0] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] # This should sum up to 1 + >>> drv = DiscreteRandomVariable(intervals=intervals, probabilities=probabilities) + + Notes + ----- + - We use intervals following OS-C convention. Internally, we work with the midpoints of each interval. + - We define this class since classes like rv_discrete from scipy do not support some important operations like multiplication + by scalar or adding a scalar to the random variable. However, it would be nice to have these features since they seem standard. + Maybe from another library outside Scipy. + - When the probabilities do not sum to one, as in the case of the ImpactDistrib class from OS-C, we add the missing value to zero + to make the sum equal to one. In this way, we create a "mass point" at zero, meaning that we take the mean value for each interval + except for zero, where we assign the remaining the probability. + TODO: We need to check the output (methodology implemented in code) of OS-C impact distribution so we are sure the constructor of + this class is properly defined. That is to say, verify that methodologically this is what we want given OS-C code. + + """ + + def __init__( + self, + probabilities: Sequence[Union[float, int]], + values: Optional[Sequence[Union[float, int]]] = None, + intervals: Optional[Sequence[Union[float, int]]] = None, + convert_to_osc_format: Optional[bool] = False, + ): + """Initialize the ExampleClass with probabilities, and either values or intervals. + + Exactly one of `values` or `intervals` must be provided. + + Parameters + ---------- + probabilities : Sequence[Union[float, int]] + A sequence of probabilities which can be float or int. + values : Optional[Sequence[Union[float, int]]], optional + An optional sequence of values corresponding to the probabilities, by default None. + intervals : Optional[Sequence[Union[float, int]]], optional + An optional sequence of intervals, by default None. + convert_to_osc_format : Optional[bool] + Ensures that the probabilities sum to 1 by adjusting the zero-impact bin. False by default. + + Raises + ------ + ValueError: If both `values` and `intervals` are provided, or if neither is provided. + + """ + if intervals is None and values is None: + raise ValueError("Either intervals or values must be provided.") + if intervals is not None and values is not None: + raise ValueError( + "Only one of intervals or values should be provided, not both." + ) + + self.probabilities = np.array(probabilities) + if intervals is not None: + probabilities_np = np.array(probabilities) + intervals_np = np.array(intervals) + if convert_to_osc_format: + if not np.all((0 <= probabilities_np) & (probabilities_np <= 1)): + raise ValueError("All probabilities must be between 0 and 1.") + + if not np.all(np.diff(intervals_np) >= 0): + raise ValueError( + "Impact bins must be sorted in non-decreasing order." + ) + total_prob = np.sum(probabilities_np) + print(total_prob) + if not np.isclose(total_prob, 1): + if 0 in intervals_np: + zero_index = np.where(intervals_np == 0)[0][0] + # Adjust the zero-impact probability + probabilities_np[zero_index] += 1 - total_prob + else: + intervals_np = np.insert(intervals_np, 0, 0) + probabilities_np = np.insert( + probabilities_np, 0, 1 - total_prob + ) + self.intervals = intervals_np + self.probabilities = probabilities_np + self.values = (self.intervals[1:-1] + self.intervals[2:]) / 2 + self.values = np.insert(self.values, 0, 0) + else: + self.intervals = intervals_np + if not (self.intervals == np.sort(self.intervals)).all(): + raise ValueError("The intervals must be sorted increasingly.") + if len(self.intervals) != len(probabilities_np) + 1: + raise ValueError( + "The number of intervals must be one more than the number of probabilities." + ) + self.values = (self.intervals[:-1] + self.intervals[1:]) / 2 + self.probabilities = probabilities_np + else: + values_np = np.array(values) + probabilities_np = np.array(probabilities) + if len(values_np) != len(probabilities_np): + raise ValueError( + "The number of values must match the number of probabilities." + ) + sorted_indices = np.argsort(values_np) + self.values = values_np[sorted_indices] + self.probabilities = probabilities_np[sorted_indices] + + # Ensure probabilities sum up to 1 + if not np.isclose(self.probabilities.sum(), 1): + raise ValueError("The probabilities must sum up to 1.") + + def __mul__(self, other: Union[float, int]): + """Multiply the discrete random variable by a scalar. + + This method scales the values of the random variable by a given scalar + while keeping the probabilities unchanged. + + Parameters + ---------- + other : float, or int + The scalar by which to multiply the values of the random variable. + + Returns + ------- + DiscreteRandomVariable + A new instance of DiscreteRandomVariable with scaled values. + + """ + if isinstance(other, (int, float)): + scaled_values = self.values * other + return DiscreteRandomVariable( + values=scaled_values, probabilities=self.probabilities.tolist() + ) + else: + return NotImplemented + + def __add__(self, other: Union[float, int]): + """Add a scalar to the discrete random variable. + + This method shifts the values of the random variable by a given scalar + while keeping the probabilities unchanged. + + Parameters + ---------- + other : float, or int + The scalar to add to the values of the random variable. + + Returns + ------- + DiscreteRandomVariable + A new instance of DiscreteRandomVariable with shifted values. + + """ + if isinstance(other, (int, float)): + shifted_values = self.values + other + return DiscreteRandomVariable( + values=shifted_values, probabilities=self.probabilities.tolist() + ) + else: + return NotImplemented + + def __rtruediv__(self, other: Union[float, int]): + r"""Implement division where a real number is divided by a DiscreteRandomVariable. + + :math:`a / X` where :math:`a, \\ X` are a Real number and a Discrete Random Variable, respectively. + + Parameters + ---------- + other : float, or int + The scalar to add to the values of the random variable. + + Returns + ------- + DiscreteRandomVariable + A new instance representing the result. + + Raises + ------ + ValueError: If division by any value of the DiscreteRandomVariable is not possible. + + Notes + ----- + We don't really need to define :math:`a / X` but rather :math:`1 / X` since __mul__ and __rmul__ + could be used. For convenience, we have done so, although it wasn't strictly necessary. + + """ + if not isinstance(other, (int, float)): + raise TypeError("Numerator must be a real number") + + # Check for zeros in self.values to avoid division by zero + if np.any(self.values == 0): + raise ValueError( + "Division by zero encountered in DiscreteRandomVariable values" + ) + + # Calculate new values as the real number divided by each value of the DiscreteRandomVariable + new_values = other / self.values + + return DiscreteRandomVariable( + values=new_values, probabilities=self.probabilities.tolist() + ) + + def __eq__(self, other: Any) -> bool: + """Determine if two DiscreteRandomVariable instances are equal based on their values and probabilities. + + Parameters + ---------- + other : Any + The other DiscreteRandomVariable instance to compare against. + + Returns + ------- + bool + Returns True if both the values and probabilities match, False otherwise. + + """ + if not isinstance(other, DiscreteRandomVariable): + return False + return np.allclose(self.values, other.values) and np.allclose( + self.probabilities, other.probabilities + ) + + def mean(self): + """Calculate the mean of the discrete random variable. + + Returns + ------- + float + The mean of the discrete random variable. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.mean() + 0.48000000000000004 + + """ + return np.sum(self.values * self.probabilities) + + @staticmethod + def means_vectorized(drvs): + """Compute means for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + + Returns + ------- + np.ndarray + An array of floats representing the means of the discrete random variables. + + Notes + ----- + This method utilizes np.vectorize to apply the mean calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> DiscreteRandomVariable.means_vectorized(drvs) + array([0.48 , 2.9968254]) + + """ + # TODO: CHeck https://github.com/os-climate/physrisk/blob/main/src/physrisk/kernel/impact_distrib.py#L40 + compute_mean = np.vectorize(lambda drv: drv.mean()) + return compute_mean(drvs) + + def var(self): + """Calculate the variance of the discrete random variable. + + Returns + ------- + float + The variance of the discrete random variable. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.var() + 0.05160000000000001 + + """ + mean = self.mean() + variance = np.sum(((self.values - mean) ** 2) * self.probabilities) + return variance + + @staticmethod + def vars_vectorized(drvs): + """Compute variances for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + + Returns + ------- + np.ndarray + An array of floats representing the means of the discrete random variables. + + Notes + ----- + This method utilizes np.vectorize to apply the variance calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> DiscreteRandomVariable.vars_vectorized(drvs) + array([0.0516 , 6.08399093]) + + """ + compute_var = np.vectorize(lambda drv: drv.var()) + return compute_var(drvs) + + def plot_pmf(self): + """Plot an interactive histogram representing the probability mass function (PMF) of the discrete random variable. + + This method uses Plotly to create an interactive histogram that provides a visual representation of how + probabilities are distributed across different intervals. + """ + # Bar chart with Plotly + fig = go.Figure( + data=[ + go.Bar( + x=self.values, + y=self.probabilities, + marker=dict(line=dict(color="black", width=1)), + ) + ] + ) + fig.update_layout( + title="Histogram of the Discrete random variable", + xaxis_title="Value", + yaxis_title="Probability", + bargap=0.2, + ) + fig.show() + + def check_values(self, min_value: float = 0, max_value: float = 1) -> bool: + """Check if all values of the DiscreteRandomVariable instance fall within a specified range. + + This method verifies that each value defined in the DiscreteRandomVariable instance is + between a specified minimum value and maximum value, inclusive. By default, it checks + whether the values are between 0 and 1. + + Parameters + ---------- + min_value : float, optional + The minimum allowable value for the values. This value is inclusive, meaning that + values can be equal to this minimum value. The default is 0. + max_value : float, optional + The maximum allowable value for the values. This value is inclusive, meaning that + values can be equal to this maximum value. The default is 1. + + Returns + ------- + bool + Returns True if all values are within the specified range (min_value to max_value, inclusive). + Otherwise, returns False. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.check_values() + True + >>> drv.check_values(0,0.5) + False + + Notes + ----- + The method utilizes numpy's vectorized operations to efficiently check all values + against the provided bounds. This approach is effective for instances with a large + number of values. + + """ + return bool(np.all((min_value <= self.values) & (self.values <= max_value))) + + def sample(self, n: Optional[int] = 1): + """Generate `n` random samples from the discrete random variable. + + Parameters + ---------- + n : int, optional + The number of samples to generate. The default is 1. + + Returns + ------- + np.ndarray + An array of sampled values. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> sample = drv.sample(5) + + """ + return np.random.choice(self.values, size=n, p=self.probabilities) + + def compute_cdf(self): + r"""Compute the Cumulative Distribution Function (CDF) for the discrete random variable. + + The CDF is defined as the probability that the variable takes a value less than or equal to `x`. + Formally, for a discrete random variable `X` with values `x_i` and corresponding probabilities `p_i`, + the CDF at a point `x` is given by: + + .. math:: F(x) = P(X \leq x) = \sum_{x_i \leq x} p_i + + Returns + ------- + cdf : np.ndarray + An array representing the cumulative probabilities corresponding to the values of the random variable. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.compute_cdf() + array([0.1, 0.4, 0.7, 0.9, 1. ]) + + """ + # Compute the cumulative distribution function (CDF) + cdf = np.cumsum(self.probabilities) + + return cdf + + def compute_exceedance_probability(self): + """Compute the exceedance probability for a given threshold. + + The exceedance probability is the probability that the discrete random variable exceeds a certain value `x`. + Formally: + + .. math:: F_X^c(x) = P(X > x) = 1 - F_X(x) + + Returns + ------- + exceed_prob : np.ndarray + An array representing the exceedance probabilities corresponding to the values of the random variable. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.compute_exceedance_probability() + array([9.00000000e-01, 6.00000000e-01, 3.00000000e-01, 1.00000000e-01, + 1.11022302e-16]) + + """ + cdf = self.compute_cdf() + exceed_prob = 1 - cdf + return exceed_prob + + @staticmethod + def compute_exceedance_probability_vectorized(drvs, x): + """Compute the exceedance probabilities for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + x : float + Value at which to evaluate the exceedance probability function. + + Returns + ------- + np.ndarray + An array of floats representing the exceedance probabilities of the discrete random variables evaluated at `x`. + + Notes + ----- + This method utilizes np.vectorize to apply the exceedance probability calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> DiscreteRandomVariable.compute_exceedance_probability_vectorized(drvs, 2) + array([1.11022302e-16, 4.00000000e-01]) + + """ + compute_exceedance = np.vectorize( + lambda drv, x: 1 - np.sum(drv.probabilities[np.where(drv.values <= x)[0]]) + ) + return compute_exceedance(drvs, x) + + def compute_occurrence_probability(self, lambda_value): + r"""Compute the occurrence probability :math:`O(x)` for the discrete random variable using a Poisson process model. + + We assume i.i.d. random variables. + + In this case we have: + + .. math:: F_X(x) = \\frac{1}{\\lambda} \\log(1 - O(x)) + 1, + + where :math:`F_X(x)` is the CDF of the random variable. + + Parameters + ---------- + lambda_value : float + The rate parameter of the Poisson process (number of occurrences per time unit). + + Returns + ------- + occurrence_prob : np.ndarray + An array representing the occurrence probabilities O(s) for the values of the random variable. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> lambda_value = 0.5 # Example rate parameter for the Poisson process + >>> drv.compute_occurrence_probability(lambda_value) + array([0.36237185, 0.25918178, 0.13929202, 0.04877058, 0. ]) + + """ + fs = self.compute_cdf() + occurrence_prob = 1 - np.exp(-lambda_value * (1 - fs)) + return occurrence_prob + + @staticmethod + def compute_occurrence_probability_vectorized(drvs, lambda_value, x): + """Compute the occurrence probabilities at `x` for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + lambda_value : float + The rate parameter of the Poisson process (number of occurrences per time unit). + x : float + Value at which to evaluate the occurrence probability function. + + Returns + ------- + np.ndarray + An array of floats representing the occurrence probabilities of the discrete random variables evaluated at `x`. + + Notes + ----- + This method utilizes np.vectorize to apply the occurrence probability calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> lambda_value = 0.5 # Example rate parameter for the Poisson process + >>> DiscreteRandomVariable.compute_occurrence_probability_vectorized(drvs, lambda_value, 0.3) + array([0.25918178, 0.39346934]) + + """ + compute_occurrence = np.vectorize( + lambda drv, lambda_value, x: 1 + - np.exp( + -lambda_value + * (1 - np.sum(drv.probabilities[np.where(drv.values <= x)[0]])) + ) + ) + return compute_occurrence(drvs, lambda_value, x) + + def compute_var(self, percentile=95): + r"""Compute the Value at Risk :math:`V^{p}_{X}` for a discrete random variable :math:`X`. + + The Value at Risk (:math:`V^{p}_{X}`) of a discrete random variable :math:`X` at the level + :math:`p \in (0, 1)` is the p-quantile of :math:`X` defined by the condition that the cumulative + distribution function :math:`F_{X}(x)` is greater than or equal to :math:`p`. Formally, + :math:`V^{p}_{X}` is given by: + + .. math:: V^{p}_{X} := \inf\{x \in \mathbb{R} : P(X \leq x) \geq p\}. + + Parameters + ---------- + percentile : float, optional + The confidence level (:math:`p`) for VaR expressed as a percentile (0-100). Default is 95. + + Returns + ------- + var_value : float + The computed VaR at the given percentile (confidence level). + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.compute_var() + 0.9 + + """ + if not 0 < percentile < 100: + raise ValueError("Percentile must be between 0 and 100.") + + # Compute the cumulative distribution function (CDF) + cdf = self.compute_cdf() + + # Find the index of the first occurrence where the CDF exceeds the target percentile + # np.isclose is used to avoid comparison numerical errors # TODO: Think of better ways to do this. + target_index = np.where( + np.isclose(cdf, percentile / 100.0) + (cdf > percentile / 100.0) + )[0][0] + var_value = self.values[target_index] + + return var_value + + @staticmethod + def compute_var_vectorized(drvs, percentile=95): + """Compute VaRs for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + percentile : float, optional + The confidence level (:math:`p`) for VaR expressed as a percentile (0-100). Default is 95. + + Returns + ------- + np.ndarray + An array of floats representing the VaRs of the discrete random variables. + + Notes + ----- + This method utilizes np.vectorize to apply the VaR calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> DiscreteRandomVariable.compute_var_vectorized(drvs) + array([ 0.9, 10. ]) + + """ + compute_var_percentile = np.vectorize( + lambda drv: drv.compute_var(percentile=percentile) + ) + return compute_var_percentile(drvs) + + def compute_es(self, percentile=95): + r"""Compute the Expected Shortfall :math:`\\mathrm{ES}^{p}_{X}` for a discrete random variable :math:`X`. + + The Expected Shortfall at level :math:`p` for a discrete random variable :math:`X`, is defined formally as: + + .. math:: \\text{ES}^{p}_X = \\frac{1}{1-p} \int_{p}^{1} V^{q}_X \, dq + + Where :math:`V^{p}_X` is the Value at Risk at level :math:`p`. + + + Parameters + ---------- + percentile : float, optional + The confidence level (:math:`p`) for ES, expressed as a percentile (0-100). Default is 95. + + Returns + ------- + es_value : float + The computed ES at the given percentile (confidence level). + + Raises + ------ + ValueError + If `percentile` is not within the range (0, 100). + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drv.compute_es() + 0.899999999999998 + + """ + # Check that percentile is between 0 and 100 + if not 0 < percentile < 100: + raise ValueError("Percentile must be between 0 and 100.") + + p = percentile / 100.0 + cdf = self.compute_cdf() + + target_indices = np.where(cdf >= p)[0] + + es = ( + np.sum((self.values * self.probabilities)[target_indices][1:]) + + self.values[target_indices[0]] * (cdf[target_indices[0]] - p) + ) / (1 - p) + + return es + + @staticmethod + def compute_es_vectorized(drvs, percentile=95): + """Compute the Expected Shortfall (ES) for an array of DiscreteRandomVariable instances using a vectorized approach. + + Parameters + ---------- + drvs : np.ndarray + An array of DiscreteRandomVariable instances. + percentile : float, optional + The confidence level (:math:`p`) for ES expressed as a percentile (0-100). Default is 95. + + Returns + ------- + np.ndarray + An array of floats representing the ESs of the discrete random variables. + + Notes + ----- + This method utilizes np.vectorize to apply the ES calculation to each instance in the array. It is primarily + for convenience and does not offer performance benefits over a traditional loop. + + Examples + -------- + >>> values = [0.1, 0.3, 0.5, 0.7, 0.9] + >>> probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + >>> drv = DiscreteRandomVariable(values=values, probabilities=probabilities) + >>> drvs = np.array([drv, 1 / drv]) + >>> DiscreteRandomVariable.compute_es_vectorized(drvs) + array([ 0.9, 10. ]) + + """ + compute_es_percentile = np.vectorize( + lambda drv: drv.compute_es(percentile=percentile) + ) + return compute_es_percentile(drvs) diff --git a/src/osc_physrisk_financial/skeleton.py b/src/osc_physrisk_financial/skeleton.py deleted file mode 100644 index 46239d8..0000000 --- a/src/osc_physrisk_financial/skeleton.py +++ /dev/null @@ -1,149 +0,0 @@ -""" -This is a skeleton file that can serve as a starting point for a Python -console script. To run this script uncomment the following lines in the -``[options.entry_points]`` section in ``setup.cfg``:: - - console_scripts = - fibonacci = osc_physrisk_financial.skeleton:run - -Then run ``pip install .`` (or ``pip install -e .`` for editable mode) -which will install the command ``fibonacci`` inside your current environment. - -Besides console scripts, the header (i.e. until ``_logger``...) of this file can -also be used as template for Python modules. - -Note: - This file can be renamed depending on your needs or safely removed if not needed. - -References: - - https://setuptools.pypa.io/en/latest/userguide/entry_point.html - - https://pip.pypa.io/en/stable/reference/pip_install -""" - -import argparse -import logging -import sys - -from osc_physrisk_financial import __version__ - -__author__ = "github-actions[bot]" -__copyright__ = "github-actions[bot]" -__license__ = "Apache-2.0" - -_logger = logging.getLogger(__name__) - - -# ---- Python API ---- -# The functions defined in this section can be imported by users in their -# Python scripts/interactive interpreter, e.g. via -# `from osc_physrisk_financial.skeleton import fib`, -# when using this Python module as a library. - - -def fib(n): - """Fibonacci example function - - Args: - n (int): integer - - Returns: - int: n-th Fibonacci number - """ - assert n > 0 - a, b = 1, 1 - for _i in range(n - 1): - a, b = b, a + b - return a - - -# ---- CLI ---- -# The functions defined in this section are wrappers around the main Python -# API allowing them to be called directly from the terminal as a CLI -# executable/script. - - -def parse_args(args): - """Parse command line parameters - - Args: - args (List[str]): command line parameters as list of strings - (for example ``["--help"]``). - - Returns: - :obj:`argparse.Namespace`: command line parameters namespace - """ - parser = argparse.ArgumentParser(description="Just a Fibonacci demonstration") - parser.add_argument( - "--version", - action="version", - version=f"osc-physrisk-financial {__version__}", - ) - parser.add_argument(dest="n", help="n-th Fibonacci number", type=int, metavar="INT") - parser.add_argument( - "-v", - "--verbose", - dest="loglevel", - help="set loglevel to INFO", - action="store_const", - const=logging.INFO, - ) - parser.add_argument( - "-vv", - "--very-verbose", - dest="loglevel", - help="set loglevel to DEBUG", - action="store_const", - const=logging.DEBUG, - ) - return parser.parse_args(args) - - -def setup_logging(loglevel): - """Setup basic logging - - Args: - loglevel (int): minimum loglevel for emitting messages - """ - logformat = "[%(asctime)s] %(levelname)s:%(name)s:%(message)s" - logging.basicConfig( - level=loglevel, stream=sys.stdout, format=logformat, datefmt="%Y-%m-%d %H:%M:%S" - ) - - -def main(args): - """Wrapper allowing :func:`fib` to be called with string arguments in a CLI fashion - - Instead of returning the value from :func:`fib`, it prints the result to the - ``stdout`` in a nicely formatted message. - - Args: - args (List[str]): command line parameters as list of strings - (for example ``["--verbose", "42"]``). - """ - args = parse_args(args) - setup_logging(args.loglevel) - _logger.debug("Starting crazy calculations...") - print(f"The {args.n}-th Fibonacci number is {fib(args.n)}") - _logger.info("Script ends here") - - -def run(): - """Calls :func:`main` passing the CLI arguments extracted from :obj:`sys.argv` - - This function can be used as entry point to create console scripts with setuptools. - """ - main(sys.argv[1:]) - - -if __name__ == "__main__": - # ^ This is a guard statement that will prevent the following code from - # being executed in the case someone imports this file instead of - # executing it as a script. - # https://docs.python.org/3/library/__main__.html - - # After installing your project with pip, users can also run your Python - # modules as scripts via the ``-m`` flag, as defined in PEP 338:: - # - # python -m osc_physrisk_financial.skeleton 42 - # - run() diff --git a/tests/conftest.py b/tests/conftest.py index 067af0d..d9ef314 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,5 +1,4 @@ -""" -Dummy conftest.py for osc_physrisk_financial. +"""Dummy conftest.py for osc_physrisk_financial. If you don't know what this is for, just leave it empty. Read more about conftest.py under: @@ -7,4 +6,7 @@ - https://docs.pytest.org/en/stable/writing_plugins.html """ -# import pytest +import os +import sys + +sys.path.append(os.path.normpath(os.path.join(os.path.dirname(__file__), "../src"))) diff --git a/tests/test_dynamics.py b/tests/test_dynamics.py new file mode 100644 index 0000000..099443d --- /dev/null +++ b/tests/test_dynamics.py @@ -0,0 +1,25 @@ +import numpy as np +import pandas as pd + +from osc_physrisk_financial.dynamics import ConstantGrowth + +value0 = 1000 +growth_rate = 0.05 +dates = pd.date_range(start="2020-01-01", periods=5, freq="YE") + + +def test_init(): + assert ( + ConstantGrowth(growth_rate=growth_rate, value0=value0, name="Test Growth") + is not None + ) + + +def test_compute_value(): + const_growth = ConstantGrowth( + growth_rate=growth_rate, value0=value0, name="Test Growth" + ) + expected_values = value0 * (1 + growth_rate) ** np.arange(0, 5) + expected_values = np.array(expected_values) + assert const_growth.compute_value(dates).all() == expected_values.all() + assert const_growth.compute_value(dates.tolist()).all() == expected_values.all() diff --git a/tests/test_functions.py b/tests/test_functions.py new file mode 100644 index 0000000..cc62d71 --- /dev/null +++ b/tests/test_functions.py @@ -0,0 +1,205 @@ +import pytest +import pandas as pd +import numpy as np +from scipy import optimize +from osc_physrisk_financial.functions import ( + find_root, + check_all_nonnumeric, + dates_formatting, + contains_word, +) + + +# Test for find_root +def func_quad(x): + return x**2 - 2 + + +def fprime_quad(x): + return 2 * x + + +def func_cos(x): + return np.cos(x) - x + + +def func_non_quadratic(x): + return np.tan(x) + + +def func_cubic(x): + return x**3 - x - 2 + + +def func_no_real_root(x): + return x**2 + 1 + + +def func_large_error(x): + return (x - 1) ** 2 - 0.01 # This will trigger the numerical error too large + + +interval_quad = [0, 2] +interval_cubic = [1, 2] +interval_no_real_root = [-1, 1] +interval_large_error = [0, 2] + + +def test_newton_secant_method(): + root = find_root(func_quad, x0=1.0, interval=interval_quad) + assert np.isclose(root, np.sqrt(2), atol=1e-8) + + +def test_newton_raphson_method(): + root = find_root(func_quad, x0=1.0, interval=interval_quad, fprime=fprime_quad) + assert np.isclose(root, np.sqrt(2), atol=1e-8) + + +def test_fixed_point_method(): + root = find_root(func_non_quadratic, x0=0.5, interval=[0, 1]) + expected_root = optimize.fixed_point(lambda x: x - func_non_quadratic(x), 0.5) + assert np.isclose(root, expected_root, atol=1e-8) + + +def test_bisection_method(): + root = find_root(func_cubic, x0=1.5, interval=interval_cubic) + expected_root = optimize.bisect(func_cubic, interval_cubic[0], interval_cubic[1]) + assert np.isclose(root, expected_root, atol=1e-8) + + +def test_brentq_method(): + root = find_root(func_cubic, x0=1.5, interval=interval_cubic) + expected_root = optimize.brentq(func_cubic, interval_cubic[0], interval_cubic[1]) + assert np.isclose(root, expected_root, atol=10**-300) + + +def test_ridder_method(): + root = find_root(func_cubic, x0=1.5, interval=interval_cubic) + expected_root = optimize.ridder(func_cubic, interval_cubic[0], interval_cubic[1]) + assert np.isclose(root, expected_root, atol=1e-8) + + +def test_all_methods_fail(): + with pytest.raises(Exception, match="All methods failed"): + find_root(func_no_real_root, x0=0, interval=interval_no_real_root) + + +def test_numerical_error_too_large(): + with pytest.raises(Exception, match="The numerical error is too large."): + find_root( + func_large_error, x0=1.0, interval=interval_large_error, tolerance=10**-300 + ) + + +# test for check_all_nonnumeric + + +def test_all_nonnumeric_empty(): + arr = np.array([]) + assert check_all_nonnumeric(arr) + arr = [] + assert check_all_nonnumeric(arr) + + +def test_all_nonnumeric_numeric_elements(): + arr = np.array([1, 2, 3]) + assert not check_all_nonnumeric(arr) + arr = [1, 2, 3] + assert not check_all_nonnumeric(arr) + + +def test_all_nonnumeric_float_and_nan(): + arr = np.array([np.nan, 1.0, 3.5]) + assert not check_all_nonnumeric(arr) + arr = [np.nan, 1.0, 3.5] + assert not check_all_nonnumeric(arr) + + +def test_all_nonnumeric_strings(): + arr = np.array(["abc", "def", "ghi"]) + assert check_all_nonnumeric(arr) + arr = ["abc", "def", "ghi"] + assert check_all_nonnumeric(arr) + + +def test_all_nonnumeric_mixed(): + arr = np.array([1, "abc", np.nan]) + assert not check_all_nonnumeric(arr) + arr = [1, "abc", np.nan] + assert not check_all_nonnumeric(arr) + + +def test_all_nonnumeric_non_iterable(): + with pytest.raises(TypeError): + check_all_nonnumeric(123) + + +def test_all_nonnumeric_numeric_with_non_nan(): + arr = np.array([1.0, 2.0, 3.0]) + assert not check_all_nonnumeric(arr) + arr = [1.0, 2.0, 3.0] + assert not check_all_nonnumeric(arr) + + +def test_all_nonnumeric_integers_and_non_nan(): + arr = np.array([1, 2, 3]) + assert not check_all_nonnumeric(arr) + arr = [1, 2, 3] + assert not check_all_nonnumeric(arr) + + +def test_all_nonnumeric_float_and_integer(): + arr = np.array([1.0, 2, 3.5]) + assert not check_all_nonnumeric(arr) + arr = [1.0, 2, 3.5] + assert not check_all_nonnumeric(arr) + + +# test for dates_formatting + + +def test_single_date_string(): + result = dates_formatting("2022-01-01") + expected = pd.DatetimeIndex(["2022-01-01"]) + assert result.equals(expected) + + +def test_list_of_dates_strings(): + result = dates_formatting(["2022-01-03", "2022-01-01", "2022-01-02"]) + expected = pd.DatetimeIndex(["2022-01-01", "2022-01-02", "2022-01-03"]) + assert result.equals(expected) + + +def test_list_of_dates_strings_with_single_date(): + result = dates_formatting(["2022-01-03", "2022-01-01", "2022-01-02"], "2022-01-01") + expected1 = pd.DatetimeIndex(["2022-01-01", "2022-01-02", "2022-01-03"]) + expected2 = pd.DatetimeIndex(["2022-01-01"]) + assert result[0].equals(expected1) + assert result[1].equals(expected2) + + +def test_pandas_datetime_index(): + dates = pd.to_datetime(["2022-01-03", "2022-01-01", "2022-01-02"]) + result = dates_formatting(dates) + expected = pd.DatetimeIndex(["2022-01-01", "2022-01-02", "2022-01-03"]) + assert result.equals(expected) + + +def test_mixed_input_formats(): + dates = ["2022-01-03", "2022-01-01", "2022-01-02"] + mixed_dates = [pd.to_datetime(dates), "2022-01-01"] + result = dates_formatting(*mixed_dates) + expected1 = pd.DatetimeIndex(["2022-01-01", "2022-01-02", "2022-01-03"]) + expected2 = pd.DatetimeIndex(["2022-01-01"]) + assert result[0].equals(expected1) + assert result[1].equals(expected2) + + +# test for contains_word (delete if function is deleted from functions.py) + + +def test_contains_word_single_match(): + string_list = ["word1_word2", "word3_word4", "word2_word5"] + word = "word2" + expected_output = ["word1_word2", "word2_word5"] + assert contains_word(string_list, word) == expected_output diff --git a/tests/test_powerPlant.py b/tests/test_powerPlant.py new file mode 100644 index 0000000..f8e9ee2 --- /dev/null +++ b/tests/test_powerPlant.py @@ -0,0 +1,68 @@ +import numpy as np +import pytest + +from osc_physrisk_financial.assets import PowerPlants +from osc_physrisk_financial.random_variables import DiscreteRandomVariable + + +def test_power_plants(): + # Check Random variables + values = [0.1, 0.3, 0.5, 0.7, 0.9] + probabilities = (0.1, 0.2, 0.3, 0.1, 0.3) # This should sum up to 1 + _ = DiscreteRandomVariable(probabilities, values) + + intervals = [0, 0.2, 0.4, 0.6, 0.8, 1] + probabilities = (0.1, 0.2, 0.3, 0.1, 0.3) # This should sum up to 1 + drv_intervals = DiscreteRandomVariable(probabilities, intervals=intervals) + + prod = 7892 * (10**9) # Wh generated in 2019 + elec_price = 48.87 / (10**6) # euros/Wh + name = "Central Nuclear Trillo" + + with pytest.raises( + ValueError, + match="Must provide either 'production' or both 'capacity' and 'av_rate'.", + ): + PowerPlants() + + pp = PowerPlants(production=prod, name=name) + + n_years = 2050 - 2019 + r_cst = [0.02] + r_var = n_years * r_cst + + disc_cst = pp.discount(r=r_cst, n=n_years) + disc_var = pp.discount(r_var) + + with pytest.raises( + ValueError, match="Discounting cash flows in negative number of year" + ): + pp.discount(r=r_cst, n=0.1) + + with pytest.raises(ValueError, match="Discounting cash flows has a wrong format"): + pp.discount(r=[0.01, 0.02], n=1.1) + + assert np.isclose(disc_cst, disc_var), "Discount is not calculated properly" + + damage = drv_intervals + + loss_cst = pp.financial_losses( + damages=damage, energy_price=elec_price, r=r_cst, n=n_years + ) + + loss_var = pp.financial_losses(damages=damage, energy_price=elec_price, r=r_var) + + assert np.isclose( + loss_cst.mean(), loss_var.mean() + ), "Losses are not calculated properly" + + # Now the same pp in two different ways + + pp2 = PowerPlants(capacity=900913242.0091324, av_rate=1, name=name) + loss_var2 = pp2.financial_losses(damages=damage, energy_price=elec_price, r=r_var) + + assert np.isclose( + loss_var.mean(), loss_var2.mean() + ), "Losses are not calculated properly" + + print("FINISHED DCV TEST SUCCESSFULLY!!!") diff --git a/tests/test_random_variables.py b/tests/test_random_variables.py new file mode 100644 index 0000000..e696cd4 --- /dev/null +++ b/tests/test_random_variables.py @@ -0,0 +1,295 @@ +import numpy as np +import pytest + +from osc_physrisk_financial.random_variables import DiscreteRandomVariable + +values = [0.1, 0.3, 0.5, 0.7, 0.9] +intervals = [0, 0.2, 0.4, 0.6, 0.8, 1.0] +probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] +percentiles = [10, 20, 30, 40, 50, 60, 70, 80, 90] + +drv = DiscreteRandomVariable(values=values, probabilities=probabilities) +drv2 = DiscreteRandomVariable(intervals=intervals, probabilities=probabilities) +drvs = np.array([drv, 1 / drv]) + + +def test_init(): + assert drv == drv2 + assert drv == DiscreteRandomVariable(values=values, probabilities=probabilities) + assert ( + np.array_equal( + np.array([0.1, 0.3, 0.3, 0.2, 0.05]), + DiscreteRandomVariable( + intervals=intervals, + values=None, + probabilities=[0.1, 0.3, 0.3, 0.2, 0.05], + convert_to_osc_format=True, + ).probabilities, + ) + is False + ) + assert ( + DiscreteRandomVariable( + intervals=intervals, + values=None, + probabilities=probabilities, + convert_to_osc_format=True, + ) + is not None + ) + + +def test_init_value_errors(): + with pytest.raises( + ValueError, match="Either intervals or values must be provided." + ): + DiscreteRandomVariable(intervals=None, values=None, probabilities=probabilities) + + with pytest.raises( + ValueError, + match="Only one of intervals or values should be provided, not both.", + ): + DiscreteRandomVariable( + intervals=intervals, values=values, probabilities=probabilities + ) + + with pytest.raises(ValueError, match="The intervals must be sorted increasingly."): + DiscreteRandomVariable( + intervals=[0.4, 0.3, 0.2, 0.1], values=None, probabilities=probabilities + ) + + with pytest.raises( + ValueError, + match="The number of intervals must be one more than the number of probabilities.", + ): + DiscreteRandomVariable( + intervals=[0, 0.2, 0.4, 0.6, 0.8], values=None, probabilities=probabilities + ) + + with pytest.raises( + ValueError, match="The number of values must match the number of probabilities." + ): + DiscreteRandomVariable( + intervals=None, values=[0.1, 0.3, 0.5], probabilities=[0.1, 0.2, 0.3, 0.7] + ) + + with pytest.raises(ValueError, match="The probabilities must sum up to 1."): + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=[0.1, 0.3, 0.3, 0.2, 0.7] + ) + + with pytest.raises(ValueError, match="All probabilities must be between 0 and 1."): + DiscreteRandomVariable( + intervals=intervals, + values=None, + probabilities=[-0.1, 0.3, 0.3, 0.2, 0.3], + convert_to_osc_format=True, + ) + + with pytest.raises( + ValueError, match="Impact bins must be sorted in non-decreasing order." + ): + DiscreteRandomVariable( + intervals=[1.0, 0.8, 0.6, 0.4, 0.2, 0], + values=None, + probabilities=probabilities, + convert_to_osc_format=True, + ) + + +def test_not_implemented(): + assert NotImplemented == DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).__mul__(other="a") + assert NotImplemented == DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).__add__(other="a") + + +def test_rtruediv(): + with pytest.raises(TypeError, match="Numerator must be a real number"): + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).__rtruediv__(other="a") + + with pytest.raises( + ValueError, + match="Division by zero encountered in DiscreteRandomVariable values", + ): + DiscreteRandomVariable( + intervals=None, + values=[0.0, 0.3, 0.5, 0.7, 0.9], + probabilities=probabilities, + ).__rtruediv__(other=1.0) + + +def test_eq(): + assert ( + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).__eq__(1.0) + is False + ) + + +def test_check_values(): + assert ( + 0.0 + <= DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).check_values(0.0, 1.0) + <= 1.0 + ) + + +def test_sample(): + assert 4 == len( + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).sample(4) + ) + + +def test_compute_var(): + with pytest.raises(ValueError, match="Percentile must be between 0 and 100."): + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).compute_var(percentile=101) + + +def test_compute_es(): + with pytest.raises(ValueError, match="Percentile must be between 0 and 100."): + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).compute_es(percentile=101) + + +def test_plot(): + DiscreteRandomVariable( + intervals=intervals, values=None, probabilities=probabilities + ).plot_pmf() + + +def test_magic(): + # Negative + assert -drv == DiscreteRandomVariable( + values=[-x for x in values], probabilities=probabilities + ) + + # Multiplication + assert -6 * drv == DiscreteRandomVariable( + values=[-6 * x for x in values], probabilities=probabilities + ) + assert -6 * drv == drv * (-6) + + # Addition + assert 6 + drv == DiscreteRandomVariable( + values=[6 + x for x in values], probabilities=probabilities + ) + assert 6 + drv == drv + 6 + + # Subtraction + assert drv - 6 == DiscreteRandomVariable( + values=[x - 6 for x in values], probabilities=probabilities + ) + assert 6 - drv == DiscreteRandomVariable( + values=[6 - x for x in values], probabilities=probabilities + ) + + # Division + assert -6 / drv == DiscreteRandomVariable( + values=[-6 / x for x in values], probabilities=probabilities + ) + + +def test_metrics(): + # Mean + assert np.isclose(drv.mean(), 0.48) + + # Variance + assert np.isclose(drv.var(), 0.0516) + + # Exceedance Probability + assert np.allclose( + drv.compute_exceedance_probability(), + np.array([0.9, 0.6, 0.3, 0.1, 0.0]), + ) + + # Occurrence Probability + assert np.allclose( + drv.compute_occurrence_probability(1), + 1 - np.exp(np.array([0.1, 0.4, 0.7, 0.9, 1]) - 1), + ) + + # VaR + assert np.allclose( + [drv.compute_var(p) for p in percentiles], + [0.1, 0.3, 0.3, 0.3, 0.5, 0.5, 0.5, 0.7, 0.7], + ) + + # Expected Shortfall + assert np.allclose( + [drv.compute_es(p) for p in percentiles], + (48 - np.array([1, 4, 7, 10, 15, 20, 25, 32, 39])) + / 100 + / (1 - np.array(percentiles) / 100), + ) + + +def test_metrics_vectorized(): + # Mean + assert np.allclose(DiscreteRandomVariable.means_vectorized(drvs), [0.48, 2.9968]) + + # Variance + assert np.allclose( + DiscreteRandomVariable.vars_vectorized(drvs), + [0.0516, 6.08399], + ) + + # Exceedance Probability + assert np.allclose( + DiscreteRandomVariable.compute_exceedance_probability_vectorized(drvs, 0.8), + [0.1, 1], + ) + assert np.allclose( + DiscreteRandomVariable.compute_exceedance_probability_vectorized(drvs, 3), + [0, 0.4], + ) + + # Occurrence Probability + assert np.allclose( + DiscreteRandomVariable.compute_occurrence_probability_vectorized(drvs, 1, 0.8), + [1 - np.exp(-0.1), 1 - np.exp(-1)], + ) + assert np.allclose( + DiscreteRandomVariable.compute_occurrence_probability_vectorized(drvs, 1, 3), + [0, 1 - np.exp(-0.4)], + ) + + # VaR + assert np.allclose( + [DiscreteRandomVariable.compute_var_vectorized(drvs, p) for p in percentiles], + np.vstack( + ( + [0.1, 0.3, 0.3, 0.3, 0.5, 0.5, 0.5, 0.7, 0.7], + [10 / 9, 10 / 7, 10 / 7, 2, 2, 2, 10 / 3, 10 / 3, 10 / 3], + ) + ).transpose(), + ) + + # Expected Shortfall + assert np.allclose( + np.array( + [DiscreteRandomVariable.compute_es_vectorized(drvs, p) for p in percentiles] + ), + ( + np.vstack( + ( + (48 - np.array([1, 4, 7, 10, 15, 20, 25, 32, 39])) / 100, + (1888 - np.array([70, 160, 250, 376, 502, 628, 838, 1048, 1258])) + / 630, + ) + ) + / (1 - np.array(percentiles) / 100) + ).transpose(), + ) diff --git a/tests/test_realstate.py b/tests/test_realstate.py new file mode 100644 index 0000000..bffd048 --- /dev/null +++ b/tests/test_realstate.py @@ -0,0 +1,321 @@ +from osc_physrisk_financial.assets import RealAsset +from osc_physrisk_financial.dynamics import ConstantGrowth +from osc_physrisk_financial.random_variables import DiscreteRandomVariable + +import pytest +import numpy as np + + +def test_real_asset(): + # TODO: This script should be transformed in a proper test. + + # Check dynamics + constant_g = ConstantGrowth(growth_rate=0.02, name="RealAsset", value0=100) + valuet = constant_g.compute_value( + dates=["2024-02-09", "2025-12-25", "2023-07-01", "2022-07-01"] + ) + expected_values = [ + 100.0, + 102, + 104.04, + 106.1208, + ] # Expected values from simple calculation + assert np.allclose(valuet, expected_values), f"Value_t = {valuet}" + + # Check Random variables + values = [0.1, 0.3, 0.5, 0.7, 0.9] + probabilities = (0.1, 0.2, 0.3, 0.1, 0.3) # This should sum up to 1 + discrete_rand_var_values = DiscreteRandomVariable(probabilities, values) + + intervals = [0, 0.2, 0.4, 0.6, 0.8, 1] + probabilities = (0.1, 0.2, 0.3, 0.1, 0.3) # This should sum up to 1 + discrete_rand_var_intervals = DiscreteRandomVariable( + probabilities, intervals=intervals + ) + assert discrete_rand_var_values == discrete_rand_var_intervals + discrete_rand_var = discrete_rand_var_values + print(discrete_rand_var.mean()) + discrete_rand_var_1 = 1.3 + discrete_rand_var + discrete_rand_var_2 = discrete_rand_var + 1.3 + assert discrete_rand_var_1 == discrete_rand_var_2 + + five_discrete_rand_var = 5 * discrete_rand_var + rfive_discrete_rand_var = discrete_rand_var * 5 + assert five_discrete_rand_var == rfive_discrete_rand_var + + divided_rv = 1 / discrete_rand_var + + # Create a numpy array of these random variables + rv_array = np.array( + [discrete_rand_var, five_discrete_rand_var, rfive_discrete_rand_var], + dtype=object, + ) + # Check __eq__ and np.array stuff + rv_array_div = 1 / rv_array + + assert divided_rv == rv_array_div[0] # Dummy test for __rtruediv__ and __eq__ + + assert (1 + discrete_rand_var) == (1 + rv_array)[0] # Dummy test for __sum__ + + +def test_asset(): + constant_g = ConstantGrowth(growth_rate=0.02, name="RealAsset", value0=100) + probabilities = (0.1, 0.2, 0.3, 0.1, 0.3) # This should sum up to 1 + values = [0.1, 0.3, 0.5, 0.7, 0.9] + discrete_rand_var_values = DiscreteRandomVariable(probabilities, values) + discrete_rand_var = discrete_rand_var_values + # Check assets + real_asset = RealAsset(value_0=100, dynamics=constant_g, name="RealState") + real_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var) + + error_asset = RealAsset(value_0=100, dynamics=None, name="RealState") + with pytest.raises(ValueError, match="Dynamics must be provided."): + error_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var) + + # real_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var)[0].plot_pmf() + losses = real_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var) + mean_loss = losses[0].mean() + expected_mean_loss = 56.0 + variance_loss = losses[0].var() + expected_variance_loss = 724.0 + print( + f'Mean Financial Losses: {real_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var)[0].mean()}' + ) + print( + f'Variance Financial Losses: {real_asset.financial_losses(["2030-02-09"], damage=discrete_rand_var)[0].var()}' + ) + assert np.allclose(mean_loss, expected_mean_loss), "Mean is not calculated properly" + assert np.allclose( + variance_loss, expected_variance_loss + ), "Variance is not calculated properly" + + intervals_osc = np.array( + [ + 0.00012346, + 0.00021273, + 0.000302, + 0.0003516, + 0.00040436, + 0.00043349, + 0.00048287, + 0.000516, + 0.0005943, + ] + ) + probabilities_osc = np.array( + [ + 0.00166667, + 0.00083333, + 0.0005, + 0.00033333, + 0.0002381, + 0.00017857, + 0.00013889, + 0.00011111, + ] + ) + discrete_rand_var_osc = DiscreteRandomVariable( + probabilities=probabilities_osc, + intervals=intervals_osc, + convert_to_osc_format=True, + ) + + expected_intervals = np.array( + [ + 0.0, + 0.00012346, + 0.00021273, + 0.000302, + 0.0003516, + 0.00040436, + 0.00043349, + 0.00048287, + 0.000516, + 0.0005943, + ] + ) + expected_probabilities = np.array( + [ + [ + 9.96000000e-01, + 1.66666667e-03, + 8.33333333e-04, + 5.00000000e-04, + 3.33333333e-04, + 2.38095238e-04, + 1.78571429e-04, + 1.38888889e-04, + 1.11111111e-04, + ] + ] + ) + + assert np.allclose( + discrete_rand_var_osc.intervals, expected_intervals + ), "Intervals are not calculated properly" + assert np.allclose( + discrete_rand_var_osc.probabilities, expected_probabilities + ), "Probabilities are not calculated properly" + + # zero included + + intervals_osc_zero = np.array( + [ + 0, + 0.00012346, + 0.00021273, + 0.000302, + 0.0003516, + 0.00040436, + 0.00043349, + 0.00048287, + 0.000516, + ] + ) + probabilities_osc_zero = np.array( + [ + 0.00166667, + 0.00083333, + 0.0005, + 0.00033333, + 0.0002381, + 0.00017857, + 0.00013889, + 0.00011111, + ] + ) + + discrete_rand_var_osc_zero = DiscreteRandomVariable( + probabilities=probabilities_osc_zero, + intervals=intervals_osc_zero, + convert_to_osc_format=True, + ) + + a = np.array(intervals_osc_zero[:-1] + intervals_osc_zero[1:]) / 2 + b = discrete_rand_var_osc_zero.values + + assert np.allclose(a[1:], b[1:]), "Intervals are not calculated properly" + assert np.isclose(b[0], 0), "Values are not calculated properly" + + # zero not included + discrete_rand_var_osc_zero = DiscreteRandomVariable( + probabilities=probabilities_osc, + intervals=intervals_osc, + convert_to_osc_format=True, + ) + + a = np.array(intervals_osc[:-1] + intervals_osc[1:]) / 2 + b = discrete_rand_var_osc_zero.values + + assert np.all(np.isclose(a, b[1:])), "Intervals are not calculated properly" + assert np.isclose(b[0], 0), "Values are not calculated properly" + # LTV + damage_1 = 1 / 100 * discrete_rand_var + damage_2 = 2 / 100 * discrete_rand_var + damage_3 = 0.01 + 1 / 100 * discrete_rand_var + loan_amounts = [1, 3, 5] + damages = [damage_1, damage_2, damage_3] + ltv = real_asset.ltv( + dates=["2030-02-09", "2031-02-09"], damages=damages, loan_amounts=loan_amounts + ) + + with pytest.raises(ValueError, match="Dynamics must be provided."): + error_asset.ltv( + dates=["2030-02-09", "2031-02-09"], + damages=damages, + loan_amounts=loan_amounts, + ) + + with pytest.raises( + ValueError, match="One or more damages have values outside the 0 to 1 range." + ): + damage_4 = damage_1 + 1 + ltv = real_asset.ltv( + dates=["2030-02-09", "2031-02-09"], + damages=[damage_4, damage_2, damage_3], + loan_amounts=loan_amounts, + ) + + with pytest.raises( + ValueError, + match="The lengths of 'damage' and 'loan_amount' \\(number of assets\\) must match\\.", + ): + ltv = real_asset.ltv( + dates=["2030-02-09", "2031-02-09"], + damages=[damage_1, damage_2], + loan_amounts=loan_amounts, + ) + + print(f" LTV mean value (first date, fist asset): {ltv[0,0].mean()}") + means = DiscreteRandomVariable.means_vectorized(ltv) + print(f" LTV mean values: {means}") + + expected_means = np.array( + [[0.01005639, 0.0303407, 0.05079274], [0.0098592, 0.02974579, 0.0497968]] + ) + + assert np.allclose(means, expected_means), "LTV mean values calculation failed" + + # Variances + print(f" LTV variance (first date, fist asset): {ltv[0,0].var()}") + vars = DiscreteRandomVariable.vars_vectorized(ltv) + print(f" LTV variances: {vars}") + + expected_vars = np.array( + [ + [7.40214348e-10, 2.72496428e-08, 1.92687839e-08], + [7.11470923e-10, 2.61915059e-08, 1.85205535e-08], + ] + ) + + assert np.allclose(vars, expected_vars), "LTV variance calculation failed" + + # VaR + values = np.array([-100, -20, 0, 50]) + probabilities = np.array([0.1, 0.3, 0.4, 0.2]) + drv_var = DiscreteRandomVariable(values=values, probabilities=probabilities) + percentile = 95 + # drv_var.plot_pmf() + var = drv_var.compute_var(percentile=percentile) + print(f"The Value at Risk (VaR) at the {percentile}% confidence level is: {var}") + + vars = DiscreteRandomVariable.compute_var_vectorized(ltv) + print(f" LTV VaRs: {vars}") + print(f"Works as expected? {vars[0][0] == ltv[0][0].compute_var()}") # Dummy test + expected_var = 50 + expected_es = 50 + # VaR & ES + es = drv_var.compute_es(percentile=percentile) + print(f"Percentile = {percentile}, VaR: {var}, ES: {es}") + assert np.allclose(var, expected_var), "VaR calculation failed" + assert np.allclose(es, expected_es), "ES calculation failed" + + # CDF & EP + + values = [0.1, 0.3, 0.5, 0.7, 0.9] + probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + discrete_rand_var = DiscreteRandomVariable( + values=values, probabilities=probabilities + ) + + _ = discrete_rand_var.compute_cdf() + + check_values = np.linspace(min(values), max(values), 20) + results = [] + for _ in check_values: + exceedance_probability = discrete_rand_var.compute_exceedance_probability() + cdf = discrete_rand_var.compute_cdf() + sum_check = exceedance_probability + cdf + results.append(sum_check) + + print(f"Check EP & CDF: {np.allclose(results, 1)}") + + # O(s) + + values = [0.1, 0.3, 0.5, 0.7, 0.9] + probabilities = [0.1, 0.3, 0.3, 0.2, 0.1] + lambda_value = 0.5 # Example rate parameter for the Poisson process + discrete_rand_var = DiscreteRandomVariable( + values=values, probabilities=probabilities + ) + _ = discrete_rand_var.compute_occurrence_probability(lambda_value) diff --git a/tests/test_skeleton.py b/tests/test_skeleton.py deleted file mode 100644 index 06da5fc..0000000 --- a/tests/test_skeleton.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from osc_physrisk_financial.skeleton import fib, main - -__author__ = "github-actions[bot]" -__copyright__ = "github-actions[bot]" -__license__ = "Apache-2.0" - - -def test_fib(): - """API Tests""" - assert fib(1) == 1 - assert fib(2) == 1 - assert fib(7) == 13 - with pytest.raises(AssertionError): - fib(-10) - - -def test_main(capsys): - """CLI Tests""" - # capsys is a pytest fixture that allows asserts against stdout/stderr - # https://docs.pytest.org/en/stable/capture.html - main(["7"]) - captured = capsys.readouterr() - assert "The 7-th Fibonacci number is 13" in captured.out diff --git a/tox.ini b/tox.ini deleted file mode 100644 index 69f8159..0000000 --- a/tox.ini +++ /dev/null @@ -1,93 +0,0 @@ -# Tox configuration file -# Read more under https://tox.wiki/ -# THIS SCRIPT IS SUPPOSED TO BE AN EXAMPLE. MODIFY IT ACCORDING TO YOUR NEEDS! - -[tox] -minversion = 3.24 -envlist = default -isolated_build = True - - -[testenv] -description = Invoke pytest to run automated tests -setenv = - TOXINIDIR = {toxinidir} -passenv = - HOME - SETUPTOOLS_* -extras = - testing -commands = - pytest {posargs} - - -# # To run `tox -e lint` you need to make sure you have a -# # `.pre-commit-config.yaml` file. See https://pre-commit.com -# [testenv:lint] -# description = Perform static analysis and style checks -# skip_install = True -# deps = pre-commit -# passenv = -# HOMEPATH -# PROGRAMDATA -# SETUPTOOLS_* -# commands = -# pre-commit run --all-files {posargs:--show-diff-on-failure} - - -[testenv:{build,clean}] -description = - build: Build the package in isolation according to PEP517, see https://github.com/pypa/build - clean: Remove old distribution files and temporary build artifacts (./build and ./dist) -# https://setuptools.pypa.io/en/stable/build_meta.html#how-to-use-it -skip_install = True -changedir = {toxinidir} -deps = - build: build[virtualenv] -passenv = - SETUPTOOLS_* -commands = - clean: python -c 'import shutil; [shutil.rmtree(p, True) for p in ("build", "dist", "docs/_build")]' - clean: python -c 'import pathlib, shutil; [shutil.rmtree(p, True) for p in pathlib.Path("src").glob("*.egg-info")]' - build: python -m build {posargs} -# By default, both `sdist` and `wheel` are built. If your sdist is too big or you don't want -# to make it available, consider running: `tox -e build -- --wheel` - - -[testenv:{docs,doctests,linkcheck}] -description = - docs: Invoke sphinx-build to build the docs - doctests: Invoke sphinx-build to run doctests - linkcheck: Check for broken links in the documentation -passenv = - SETUPTOOLS_* -setenv = - DOCSDIR = {toxinidir}/docs - BUILDDIR = {toxinidir}/docs/_build - docs: BUILD = html - doctests: BUILD = doctest - linkcheck: BUILD = linkcheck -deps = - -r {toxinidir}/docs/requirements.txt - # ^ requirements.txt shared with Read The Docs -commands = - sphinx-build --color -b {env:BUILD} -d "{env:BUILDDIR}/doctrees" "{env:DOCSDIR}" "{env:BUILDDIR}/{env:BUILD}" {posargs} - - -[testenv:publish] -description = - Publish the package you have been developing to a package index server. - By default, it uses testpypi. If you really want to publish your package - to be publicly accessible in PyPI, use the `-- --repository pypi` option. -skip_install = True -changedir = {toxinidir} -passenv = - # See: https://twine.readthedocs.io/en/latest/ - TWINE_USERNAME - TWINE_PASSWORD - TWINE_REPOSITORY - TWINE_REPOSITORY_URL -deps = twine -commands = - python -m twine check dist/* - python -m twine upload {posargs:--repository {env:TWINE_REPOSITORY:testpypi}} dist/*