From e39ecda9f567d05d048ea0129824a95afabfabbf Mon Sep 17 00:00:00 2001 From: Stefan Webb Date: Sat, 30 Oct 2021 17:16:09 -0700 Subject: [PATCH] Initial commit --- .github/ISSUE_TEMPLATE.md | 15 + .github/PULL_REQUEST_TEMPLATE.md | 23 + .github/workflows/deploy-on-release.yml | 72 + .github/workflows/documentation.yml | 67 + .github/workflows/python-package.yml | 61 + .gitignore | 14 + .readthedocs.yml | 24 + CHANGELOG.md | 6 + CODE_OF_CONDUCT.md | 80 + CONTRIBUTING.md | 33 + LICENSE.txt | 21 + MANIFEST.in | 7 + README.md | 34 + examples/learn_bivariate_normal.py | 123 + flowtorch/__init__.py | 6 + flowtorch/bijectors/__init__.py | 97 + flowtorch/bijectors/affine.py | 31 + flowtorch/bijectors/affine_autoregressive.py | 31 + flowtorch/bijectors/affine_fixed.py | 54 + flowtorch/bijectors/autoregressive.py | 70 + flowtorch/bijectors/base.py | 143 + flowtorch/bijectors/compose.py | 88 + flowtorch/bijectors/elementwise.py | 26 + flowtorch/bijectors/elu.py | 41 + flowtorch/bijectors/exp.py | 38 + flowtorch/bijectors/fixed.py | 32 + flowtorch/bijectors/leaky_relu.py | 38 + flowtorch/bijectors/ops/__init__.py | 2 + flowtorch/bijectors/ops/affine.py | 86 + flowtorch/bijectors/ops/spline.py | 116 + flowtorch/bijectors/permute.py | 60 + flowtorch/bijectors/power.py | 52 + flowtorch/bijectors/sigmoid.py | 39 + flowtorch/bijectors/softplus.py | 40 + flowtorch/bijectors/spline.py | 30 + flowtorch/bijectors/spline_autoregressive.py | 31 + flowtorch/bijectors/tanh.py | 40 + flowtorch/bijectors/volume_preserving.py | 24 + flowtorch/distributions/__init__.py | 13 + flowtorch/distributions/flow.py | 127 + flowtorch/distributions/neals_funnel.py | 53 + flowtorch/docs.py | 159 + flowtorch/lazy.py | 107 + flowtorch/nn/__init__.py | 7 + flowtorch/nn/made.py | 121 + flowtorch/ops/__init__.py | 303 + flowtorch/parameters/__init__.py | 14 + flowtorch/parameters/base.py | 41 + flowtorch/parameters/dense_autoregressive.py | 185 + flowtorch/parameters/tensor.py | 28 + flowtorch/utils.py | 104 + pyproject.toml | 2 + scripts/copyright_headers.py | 166 + scripts/generate_api_docs.py | 80 + scripts/generate_imports.py | 199 + setup.cfg | 9 + setup.py | 87 + tests/conftest.py | 16 + tests/test_bijector.py | 130 + tests/test_compose.py | 41 + tests/test_distribution.py | 117 + tests/test_imports.py | 66 + tests/test_interface.py | 31 + website/.gitignore | 22 + website/README.md | 41 + website/babel.config.js | 3 + website/docs/dev/about.mdx | 40 + website/docs/dev/bibliography.mdx | 89 + website/docs/dev/bijector.mdx | 69 + website/docs/dev/contributing.md | 28 + website/docs/dev/docs.md | 15 + website/docs/dev/ops.md | 91 + website/docs/dev/overview.md | 40 + website/docs/dev/params.mdx | 45 + website/docs/dev/releases.md | 15 + website/docs/dev/tests.md | 9 + website/docs/users/bijectors.md | 11 + website/docs/users/caching.md | 13 + website/docs/users/composing.md | 17 + website/docs/users/conditional.mdx | 29 + website/docs/users/conditioning.md | 11 + website/docs/users/constraints.md | 11 + website/docs/users/gpu_support.md | 9 + website/docs/users/initialization.md | 11 + website/docs/users/installation.md | 35 + website/docs/users/intro.mdx | 27 + website/docs/users/methods.md | 11 + website/docs/users/multivariate.mdx | 133 + website/docs/users/parameters.md | 11 + website/docs/users/serialization.md | 9 + website/docs/users/shapes.mdx | 106 + website/docs/users/start.mdx | 136 + website/docs/users/structure.md | 56 + website/docs/users/torchscript.md | 9 + .../docs/users/transformed_distributions.md | 11 + website/docs/users/univariate.mdx | 254 + website/docusaurus.config.js | 169 + website/flowtorch-ai.png | Bin 0 -> 168097 bytes website/package.json | 45 + website/sidebars.js | 15 + website/src/css/custom.css | 528 + website/src/pages/index.js | 35 + website/src/pages/styles.module.css | 42 + website/src/theme/CodeSnippet/index.js | 57 + .../src/theme/CodeSnippet/styles.module.css | 3 + website/src/theme/Examples/index.js | 63 + website/src/theme/Examples/snippets.js | 43 + website/src/theme/Examples/styles.module.css | 105 + website/src/theme/Features/index.js | 85 + website/src/theme/Features/styles.module.css | 40 + website/src/theme/Headline/index.js | 33 + website/src/theme/Headline/styles.module.css | 28 + website/src/theme/Hero/index.js | 62 + website/src/theme/Hero/styles.module.css | 27 + website/static/.gitignore | 1 + website/static/.nojekyll | 0 website/static/CNAME | 1 + website/static/assets/normalizing-flows.bib | 154 + .../static/img/bivariate-normal-frame-0.svg | 1641 +++ .../static/img/bivariate-normal-frame-1.svg | 1644 +++ .../static/img/bivariate-normal-frame-2.svg | 1649 +++ .../static/img/bivariate-normal-frame-3.svg | 1645 +++ .../static/img/bivariate-normal-frame-4.svg | 1642 +++ .../static/img/bivariate-normal-frame-5.svg | 1641 +++ website/static/img/book-solid.svg | 1 + website/static/img/claude_shannon.png | Bin 0 -> 962815 bytes website/static/img/code-branch-solid.svg | 1 + website/static/img/code-solid.svg | 1 + website/static/img/comments-solid.svg | 1 + website/static/img/download-solid.svg | 1 + website/static/img/favicon.png | Bin 0 -> 14857 bytes website/static/img/github-brands.svg | 1 + website/static/img/hand-paper-solid.svg | 1 + website/static/img/hand-sparkles-solid.svg | 1 + website/static/img/handshake-solid.svg | 1 + website/static/img/hat-wizard-solid.svg | 1 + website/static/img/logo-alt.svg | 21 + website/static/img/logo.svg | 17 + .../static/img/long-arrow-alt-right-solid.svg | 1 + website/static/img/normalizing-flow.svg | 1 + .../static/img/normalizing_flows_i_11_0.png | Bin 0 -> 15934 bytes .../static/img/normalizing_flows_i_14_0.png | Bin 0 -> 102698 bytes .../static/img/normalizing_flows_i_14_1.png | Bin 0 -> 29655 bytes .../static/img/normalizing_flows_i_20_0.png | Bin 0 -> 121893 bytes .../static/img/normalizing_flows_i_20_1.png | Bin 0 -> 38075 bytes .../static/img/normalizing_flows_i_27_0.png | Bin 0 -> 118223 bytes .../static/img/normalizing_flows_i_27_1.png | Bin 0 -> 41128 bytes .../static/img/normalizing_flows_i_38_0.png | Bin 0 -> 108365 bytes .../static/img/normalizing_flows_i_38_1.png | Bin 0 -> 40687 bytes .../static/img/normalizing_flows_i_9_0.png | Bin 0 -> 14714 bytes website/static/img/play-circle-regular.svg | 1 + website/static/img/play-circle-solid.svg | 1 + website/static/img/scroll-solid.svg | 1 + website/static/img/smile-regular.svg | 1 + website/static/img/smile-solid.svg | 1 + .../static/img/standard_normal_samples.png | Bin 0 -> 162413 bytes website/static/img/tasks-solid.svg | 1 + website/static/img/terminal-solid.svg | 1 + website/static/img/tools-solid.svg | 1 + .../static/img/undraw_docusaurus_mountain.svg | 170 + .../static/img/undraw_docusaurus_react.svg | 169 + website/static/img/undraw_docusaurus_tree.svg | 1 + website/static/img/user-astronaut-solid.svg | 1 + website/static/img/wrench-solid.svg | 1 + website/yarn.lock | 11179 ++++++++++++++++ 165 files changed, 28395 insertions(+) create mode 100644 .github/ISSUE_TEMPLATE.md create mode 100644 .github/PULL_REQUEST_TEMPLATE.md create mode 100644 .github/workflows/deploy-on-release.yml create mode 100644 .github/workflows/documentation.yml create mode 100644 .github/workflows/python-package.yml create mode 100644 .gitignore create mode 100644 .readthedocs.yml create mode 100644 CHANGELOG.md create mode 100644 CODE_OF_CONDUCT.md create mode 100644 CONTRIBUTING.md create mode 100644 LICENSE.txt create mode 100644 MANIFEST.in create mode 100644 README.md create mode 100644 examples/learn_bivariate_normal.py create mode 100644 flowtorch/__init__.py create mode 100644 flowtorch/bijectors/__init__.py create mode 100644 flowtorch/bijectors/affine.py create mode 100644 flowtorch/bijectors/affine_autoregressive.py create mode 100644 flowtorch/bijectors/affine_fixed.py create mode 100644 flowtorch/bijectors/autoregressive.py create mode 100644 flowtorch/bijectors/base.py create mode 100644 flowtorch/bijectors/compose.py create mode 100644 flowtorch/bijectors/elementwise.py create mode 100644 flowtorch/bijectors/elu.py create mode 100644 flowtorch/bijectors/exp.py create mode 100644 flowtorch/bijectors/fixed.py create mode 100644 flowtorch/bijectors/leaky_relu.py create mode 100644 flowtorch/bijectors/ops/__init__.py create mode 100644 flowtorch/bijectors/ops/affine.py create mode 100644 flowtorch/bijectors/ops/spline.py create mode 100644 flowtorch/bijectors/permute.py create mode 100644 flowtorch/bijectors/power.py create mode 100644 flowtorch/bijectors/sigmoid.py create mode 100644 flowtorch/bijectors/softplus.py create mode 100644 flowtorch/bijectors/spline.py create mode 100644 flowtorch/bijectors/spline_autoregressive.py create mode 100644 flowtorch/bijectors/tanh.py create mode 100644 flowtorch/bijectors/volume_preserving.py create mode 100644 flowtorch/distributions/__init__.py create mode 100644 flowtorch/distributions/flow.py create mode 100644 flowtorch/distributions/neals_funnel.py create mode 100644 flowtorch/docs.py create mode 100644 flowtorch/lazy.py create mode 100644 flowtorch/nn/__init__.py create mode 100644 flowtorch/nn/made.py create mode 100644 flowtorch/ops/__init__.py create mode 100644 flowtorch/parameters/__init__.py create mode 100644 flowtorch/parameters/base.py create mode 100644 flowtorch/parameters/dense_autoregressive.py create mode 100644 flowtorch/parameters/tensor.py create mode 100644 flowtorch/utils.py create mode 100644 pyproject.toml create mode 100644 scripts/copyright_headers.py create mode 100644 scripts/generate_api_docs.py create mode 100644 scripts/generate_imports.py create mode 100644 setup.cfg create mode 100644 setup.py create mode 100644 tests/conftest.py create mode 100644 tests/test_bijector.py create mode 100644 tests/test_compose.py create mode 100644 tests/test_distribution.py create mode 100644 tests/test_imports.py create mode 100644 tests/test_interface.py create mode 100644 website/.gitignore create mode 100644 website/README.md create mode 100644 website/babel.config.js create mode 100644 website/docs/dev/about.mdx create mode 100644 website/docs/dev/bibliography.mdx create mode 100644 website/docs/dev/bijector.mdx create mode 100644 website/docs/dev/contributing.md create mode 100644 website/docs/dev/docs.md create mode 100644 website/docs/dev/ops.md create mode 100644 website/docs/dev/overview.md create mode 100644 website/docs/dev/params.mdx create mode 100644 website/docs/dev/releases.md create mode 100644 website/docs/dev/tests.md create mode 100644 website/docs/users/bijectors.md create mode 100644 website/docs/users/caching.md create mode 100644 website/docs/users/composing.md create mode 100644 website/docs/users/conditional.mdx create mode 100644 website/docs/users/conditioning.md create mode 100644 website/docs/users/constraints.md create mode 100644 website/docs/users/gpu_support.md create mode 100644 website/docs/users/initialization.md create mode 100644 website/docs/users/installation.md create mode 100644 website/docs/users/intro.mdx create mode 100644 website/docs/users/methods.md create mode 100644 website/docs/users/multivariate.mdx create mode 100644 website/docs/users/parameters.md create mode 100644 website/docs/users/serialization.md create mode 100644 website/docs/users/shapes.mdx create mode 100644 website/docs/users/start.mdx create mode 100644 website/docs/users/structure.md create mode 100644 website/docs/users/torchscript.md create mode 100644 website/docs/users/transformed_distributions.md create mode 100644 website/docs/users/univariate.mdx create mode 100644 website/docusaurus.config.js create mode 100644 website/flowtorch-ai.png create mode 100644 website/package.json create mode 100644 website/sidebars.js create mode 100644 website/src/css/custom.css create mode 100644 website/src/pages/index.js create mode 100644 website/src/pages/styles.module.css create mode 100644 website/src/theme/CodeSnippet/index.js create mode 100644 website/src/theme/CodeSnippet/styles.module.css create mode 100644 website/src/theme/Examples/index.js create mode 100644 website/src/theme/Examples/snippets.js create mode 100644 website/src/theme/Examples/styles.module.css create mode 100644 website/src/theme/Features/index.js create mode 100644 website/src/theme/Features/styles.module.css create mode 100644 website/src/theme/Headline/index.js create mode 100644 website/src/theme/Headline/styles.module.css create mode 100644 website/src/theme/Hero/index.js create mode 100644 website/src/theme/Hero/styles.module.css create mode 100644 website/static/.gitignore create mode 100644 website/static/.nojekyll create mode 100644 website/static/CNAME create mode 100644 website/static/assets/normalizing-flows.bib create mode 100644 website/static/img/bivariate-normal-frame-0.svg create mode 100644 website/static/img/bivariate-normal-frame-1.svg create mode 100644 website/static/img/bivariate-normal-frame-2.svg create mode 100644 website/static/img/bivariate-normal-frame-3.svg create mode 100644 website/static/img/bivariate-normal-frame-4.svg create mode 100644 website/static/img/bivariate-normal-frame-5.svg create mode 100644 website/static/img/book-solid.svg create mode 100644 website/static/img/claude_shannon.png create mode 100644 website/static/img/code-branch-solid.svg create mode 100644 website/static/img/code-solid.svg create mode 100644 website/static/img/comments-solid.svg create mode 100644 website/static/img/download-solid.svg create mode 100644 website/static/img/favicon.png create mode 100644 website/static/img/github-brands.svg create mode 100644 website/static/img/hand-paper-solid.svg create mode 100644 website/static/img/hand-sparkles-solid.svg create mode 100644 website/static/img/handshake-solid.svg create mode 100644 website/static/img/hat-wizard-solid.svg create mode 100644 website/static/img/logo-alt.svg create mode 100644 website/static/img/logo.svg create mode 100644 website/static/img/long-arrow-alt-right-solid.svg create mode 100644 website/static/img/normalizing-flow.svg create mode 100644 website/static/img/normalizing_flows_i_11_0.png create mode 100644 website/static/img/normalizing_flows_i_14_0.png create mode 100644 website/static/img/normalizing_flows_i_14_1.png create mode 100644 website/static/img/normalizing_flows_i_20_0.png create mode 100644 website/static/img/normalizing_flows_i_20_1.png create mode 100644 website/static/img/normalizing_flows_i_27_0.png create mode 100644 website/static/img/normalizing_flows_i_27_1.png create mode 100644 website/static/img/normalizing_flows_i_38_0.png create mode 100644 website/static/img/normalizing_flows_i_38_1.png create mode 100644 website/static/img/normalizing_flows_i_9_0.png create mode 100644 website/static/img/play-circle-regular.svg create mode 100644 website/static/img/play-circle-solid.svg create mode 100644 website/static/img/scroll-solid.svg create mode 100644 website/static/img/smile-regular.svg create mode 100644 website/static/img/smile-solid.svg create mode 100644 website/static/img/standard_normal_samples.png create mode 100644 website/static/img/tasks-solid.svg create mode 100644 website/static/img/terminal-solid.svg create mode 100644 website/static/img/tools-solid.svg create mode 100644 website/static/img/undraw_docusaurus_mountain.svg create mode 100644 website/static/img/undraw_docusaurus_react.svg create mode 100644 website/static/img/undraw_docusaurus_tree.svg create mode 100644 website/static/img/user-astronaut-solid.svg create mode 100644 website/static/img/wrench-solid.svg create mode 100644 website/yarn.lock diff --git a/.github/ISSUE_TEMPLATE.md b/.github/ISSUE_TEMPLATE.md new file mode 100644 index 00000000..9bcfa51f --- /dev/null +++ b/.github/ISSUE_TEMPLATE.md @@ -0,0 +1,15 @@ +### Issue Description +A clear and concise description of the issue. If it's a feature request, please add [Feature Request] to the title. + +### Steps to Reproduce +Please provide steps to reproduce the issue attaching any error messages and stack traces. + +### Expected Behavior +What did you expect to happen? + +### System Info +Please provide information about your setup +- PyTorch Version (run `print(torch.__version__)` +- Python version + +### Additional Context diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md new file mode 100644 index 00000000..5e893cca --- /dev/null +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -0,0 +1,23 @@ +### Motivation +Please describe your motivation for the changes. Provide link to any related issues. + +### Changes proposed +Outline the proposed changes and alternatives considered. + +### Test Plan +Please provide clear instructions on how the changes were verified. Attach screenshots if applicable. + +### Types of changes +- [ ] Docs change / refactoring / dependency upgrade +- [ ] Bug fix (non-breaking change which fixes an issue) +- [ ] New feature (non-breaking change which adds functionality) +- [ ] Breaking change (fix or feature that would cause existing functionality to change) + +### Checklist +- [ ] My code follows the code style of this project. +- [ ] My change requires a change to the documentation. +- [ ] I have updated the documentation accordingly. +- [ ] I have read the **[CONTRIBUTING](https://github.com/facebookincubator/flowtorch/blob/main/CONTRIBUTING.md)** document. +- [ ] I have added tests to cover my changes. +- [ ] All new and existing tests passed. +- [ ] The title of my pull request is a short description of the requested changes. diff --git a/.github/workflows/deploy-on-release.yml b/.github/workflows/deploy-on-release.yml new file mode 100644 index 00000000..4f8a998f --- /dev/null +++ b/.github/workflows/deploy-on-release.yml @@ -0,0 +1,72 @@ +name: Deploy On Release + +on: + release: + types: [created] + +jobs: + tests-and-coverage-pip: + name: Tests and coverage (pip, Python ${{ matrix.python-version }}, ${{ matrix.os }}) + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: ["ubuntu-latest", "macos-latest", "windows-latest"] + python-version: ['3.7', '3.8', '3.9'] + steps: + - uses: actions/checkout@v2 + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v2 + with: + python-version: ${{ matrix.python-version }} + - name: Install (auto-install dependencies) + run: | + python -m pip install --upgrade pip + pip install -e .[test] + - name: Test with pytest + run: | + pytest --cov=tests --cov-report=xml -W ignore::DeprecationWarning tests/ + - name: Upload coverage to Codecov + if: ${{ runner.os == 'Linux' && matrix.python-version == 3.7 }} + uses: codecov/codecov-action@v1 + with: + token: 9667eb01-c300-4166-b8ba-605deb2682e4 + files: coverage.xml + directory: ./ + flags: unittests + env_vars: OS,PYTHON + name: codecov-umbrella + fail_ci_if_error: true + path_to_write_report: ./codecov_report.txt + verbose: true + + release-pypi: + name: Release to pypi.org + runs-on: ubuntu-latest + needs: tests-and-coverage-pip + strategy: + fail-fast: true + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + fetch-depth: 0 + - name: Set up Python 3.7 + uses: actions/setup-python@v2 + with: + python-version: 3.7 + - name: Install packaging dependencies + run: | + python -m pip install --upgrade pip + pip install --upgrade setuptools wheel + - name: Build source distribution + run: python setup.py sdist bdist_wheel + - name: Upload to PyPI + uses: pypa/gh-action-pypi-publish@release/v1 + with: + verbose: true + user: __token__ + password: ${{ secrets.PYPI_PASSWORD }} + # swap for test PyPI + # password: ${{ secrets.TEST_PYPI_API_TOKEN }} + # repository_url: https://test.pypi.org/legacy/ diff --git a/.github/workflows/documentation.yml b/.github/workflows/documentation.yml new file mode 100644 index 00000000..a8d9fb1c --- /dev/null +++ b/.github/workflows/documentation.yml @@ -0,0 +1,67 @@ +name: documentation + +on: + pull_request: + branches: [main] + push: + branches: [main] + +jobs: + checks: + if: github.event_name != 'push' + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: actions/setup-python@v2 + with: + python-version: 3.8 + - uses: actions/setup-node@v2 + with: + node-version: '12.x' + - name: Test Build + working-directory: ./website + run: | + python -m pip install --upgrade pip + pip install -e .. + python ../scripts/generate_api_docs.py + if [ -e yarn.lock ]; then + yarn install --frozen-lockfile + elif [ -e package-lock.json ]; then + npm ci + else + npm i + fi + npm run build + gh-release: + if: github.event_name != 'pull_request' + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: actions/setup-python@v2 + with: + python-version: 3.8 + - uses: actions/setup-node@v2 + with: + node-version: '12.x' + - uses: webfactory/ssh-agent@v0.5.0 + with: + ssh-private-key: ${{ secrets.GH_PAGES_DEPLOY }} + - name: Release to GitHub Pages + env: + USE_SSH: true + GIT_USER: git + working-directory: ./website + run: | + git config --global user.email "feynmanl@fb.com" + git config --global user.name "Feynman Liang" + python -m pip install --upgrade pip + pip install -e .. + python ../scripts/generate_api_docs.py + if [ -e yarn.lock ]; then + yarn install --frozen-lockfile + elif [ -e package-lock.json ]; then + npm ci + else + npm i + fi + npm run deploy diff --git a/.github/workflows/python-package.yml b/.github/workflows/python-package.yml new file mode 100644 index 00000000..53b1a60a --- /dev/null +++ b/.github/workflows/python-package.yml @@ -0,0 +1,61 @@ +# This workflow will install Python dependencies, run tests and lint with a variety of Python versions +# For more information see: https://help.github.com/actions/language-and-framework-guides/using-python-with-github-actions + +name: Python package + +on: + push: + branches: [ main ] + pull_request: + branches: [ main ] + +jobs: + tests-and-coverage-pip: + name: Tests and coverage (pip, Python ${{ matrix.python-version }}, ${{ matrix.os }}) + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: ["ubuntu-latest", "macos-latest", "windows-latest"] + python-version: ['3.7', '3.8', '3.9'] + steps: + - uses: actions/checkout@v2 + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v2 + with: + python-version: ${{ matrix.python-version }} + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install -e .[dev] + - name: Check copyright headers + run: | + python scripts/copyright_headers.py --check flowtorch tests scripts examples + - name: Check formatting with black + run: | + black --check flowtorch tests scripts examples + - name: Check imports with usort + run: | + usort check flowtorch tests scripts examples + - name: Lint with flake8 + run: | + flake8 . --count --show-source --statistics + - name: Check types with mypy + run: | + mypy --disallow-untyped-defs flowtorch + - name: Test with pytest + run: | + pytest --cov=tests --cov-report=xml -W ignore::DeprecationWarning tests/ + - name: Upload coverage to Codecov + if: ${{ runner.os == 'Linux' && matrix.python-version == 3.7 }} + uses: codecov/codecov-action@v1 + with: + token: 9667eb01-c300-4166-b8ba-605deb2682e4 + files: coverage.xml + directory: ./ + flags: unittests + env_vars: OS,PYTHON + name: codecov-umbrella + fail_ci_if_error: true + path_to_write_report: ./codecov_report.txt + verbose: true diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..c88eed5e --- /dev/null +++ b/.gitignore @@ -0,0 +1,14 @@ +flowtorch.egg-info +flowtorch/version.py +*.pyc +.ipynb_checkpoints/ +docs/_build/ +debug/.vscode +debug/*.svg +build/ +dist/ +.eggs/ +coverage.xml +.mypy_cache/ +.vscode/ +/.coverage diff --git a/.readthedocs.yml b/.readthedocs.yml new file mode 100644 index 00000000..ec839c54 --- /dev/null +++ b/.readthedocs.yml @@ -0,0 +1,24 @@ +# .readthedocs.yml +# Read the Docs configuration file +# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details + +# Required +version: 2 + +# 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 + +# Optionally set the version of Python and requirements required to build your docs +python: + version: 3.7 + install: + - requirements: docs/requirements.txt diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 00000000..7d31e52a --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,6 @@ +0.3 (September 15, 2021) + +* Deferred initialization of `Bijector`s and `Parameters` is expressed using the `flowtorch.LazyMeta` metaclass +* AffineAutoregressive can operate on inputs with arbitrary `event_shape`s +* A few cosmetic changes like changing `flowtorch.params.*` to `flowtorch.parameters.*` +* Temporarily removed conditional bijectors/transformed distributions and inverting bijectors diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 00000000..83f431e8 --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,80 @@ +# Code of Conduct + +## Our Pledge + +In the interest of fostering an open and welcoming environment, we as +contributors and maintainers pledge to make participation in our project and +our community a harassment-free experience for everyone, regardless of age, body +size, disability, ethnicity, sex characteristics, gender identity and expression, +level of experience, education, socio-economic status, nationality, personal +appearance, race, religion, or sexual identity and orientation. + +## Our Standards + +Examples of behavior that contributes to creating a positive environment +include: + +* Using welcoming and inclusive language +* Being respectful of differing viewpoints and experiences +* Gracefully accepting constructive criticism +* Focusing on what is best for the community +* Showing empathy towards other community members + +Examples of unacceptable behavior by participants include: + +* The use of sexualized language or imagery and unwelcome sexual attention or +advances +* Trolling, insulting/derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or electronic +address, without explicit permission +* Other conduct which could reasonably be considered inappropriate in a +professional setting + +## Our Responsibilities + +Project maintainers are responsible for clarifying the standards of acceptable +behavior and are expected to take appropriate and fair corrective action in +response to any instances of unacceptable behavior. + +Project maintainers have the right and responsibility to remove, edit, or +reject comments, commits, code, wiki edits, issues, and other contributions +that are not aligned to this Code of Conduct, or to ban temporarily or +permanently any contributor for other behaviors that they deem inappropriate, +threatening, offensive, or harmful. + +## Scope + +This Code of Conduct applies within all project spaces, and it also applies when +an individual is representing the project or its community in public spaces. +Examples of representing a project or community include using an official +project e-mail address, posting via an official social media account, or acting +as an appointed representative at an online or offline event. Representation of +a project may be further defined and clarified by project maintainers. + +This Code of Conduct also applies outside the project spaces when there is a +reasonable belief that an individual's behavior may have a negative impact on +the project or its community. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported by contacting the project team at . All +complaints will be reviewed and investigated and will result in a response that +is deemed necessary and appropriate to the circumstances. The project team is +obligated to maintain confidentiality with regard to the reporter of an incident. +Further details of specific enforcement policies may be posted separately. + +Project maintainers who do not follow or enforce the Code of Conduct in good +faith may face temporary or permanent repercussions as determined by other +members of the project's leadership. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, +available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html + +[homepage]: https://www.contributor-covenant.org + +For answers to common questions about this code of conduct, see +https://www.contributor-covenant.org/faq diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 00000000..eefe96ad --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,33 @@ +# Contributing to FlowTorch +We want to make contributing to this project as easy and transparent as +possible. + +## Pull Requests +We actively welcome your pull requests. + +1. Fork the repo and create your branch from `main`. +2. If you've added code that should be tested, add tests. +3. If you've changed APIs, update the documentation. +4. Ensure the test suite passes. +5. Make sure your code lints. +6. If you haven't already, complete the Contributor License Agreement ("CLA"). + +For more details see the [developers guide](https://flowtorch.ai/dev). + +## Contributor License Agreement ("CLA") +In order to accept your pull request, we need you to submit a CLA. You only need +to do this once to work on any of Facebook's open source projects. + +Complete your CLA here: + +## Issues +We use GitHub issues to track public bugs. Please ensure your description is +clear and has sufficient instructions to be able to reproduce the issue. + +Facebook has a [bounty program](https://www.facebook.com/whitehat/) for the safe +disclosure of security bugs. In those cases, please go through the process +outlined on that page and do not file a public issue. + +## License +By contributing to FlowTorch, you agree that your contributions will be licensed +under the LICENSE file in the root directory of this source tree. diff --git a/LICENSE.txt b/LICENSE.txt new file mode 100644 index 00000000..f2082510 --- /dev/null +++ b/LICENSE.txt @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) FlowTorch Development Team. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/MANIFEST.in b/MANIFEST.in new file mode 100644 index 00000000..b44fbdab --- /dev/null +++ b/MANIFEST.in @@ -0,0 +1,7 @@ +recursive-include flowtorch +global-exclude *.bat *.yml .gitignore +prune debug +prune docs +prune tests +prune website +prune .github \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 00000000..b1241d3c --- /dev/null +++ b/README.md @@ -0,0 +1,34 @@ +

+ +[![](https://github.com/facebookincubator/flowtorch/workflows/Python%20package/badge.svg)](https://github.com/facebookincubator/flowtorch/actions?query=workflow%3A%22Python+package%22) + +Copyright (c) FlowTorch Development Team. + +This source code is licensed under the MIT license found in the +[LICENSE.txt](https://github.com/facebookincubator/flowtorch/blob/main/LICENSE.txt) file in the root directory of this source tree. + +> :boom: **FlowTorch is currently in pre-release and many of its planned features and documentation are incomplete!** You may wish to wait until the first release planned for **8/03/2021**. + +# Overview + +FlowTorch is a PyTorch library for learning and sampling from complex probability distributions using a class of methods called [Normalizing Flows](https://arxiv.org/abs/1908.09257). + +# Installing + +An easy way to get started is to install from source: + + git clone https://github.com/facebookincubator/flowtorch.git + cd flowtorch + pip install -e . + +# Further Information + +We refer you to the [FlowTorch website](https://flowtorch.ai) for more information about installation, using the library, and becoming a contributor. Here is a handy guide: + +* [What are normalizing flows?](https://flowtorch.ai/users) +* [How do I install FlowTorch?](https://flowtorch.ai/users/installation) +* [How do I construct and train a distribution?](https://flowtorch.ai/users/start) +* [How do I contribute new normalizing flow methods?](https://flowtorch.ai/dev) +* [Where can I report bugs?](https://github.com/facebookincubator/flowtorch/issues) +* [Where can I ask general questions and make feature requests?](https://github.com/facebookincubator/flowtorch/discussions) +* [What features are planned for the near future?](https://github.com/facebookincubator/flowtorch/projects) diff --git a/examples/learn_bivariate_normal.py b/examples/learn_bivariate_normal.py new file mode 100644 index 00000000..6f832661 --- /dev/null +++ b/examples/learn_bivariate_normal.py @@ -0,0 +1,123 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import os + +import flowtorch.bijectors as bij +import flowtorch.distributions as dist +import flowtorch.parameters as params +import matplotlib.pyplot as plt +import torch + +os.environ["KMP_DUPLICATE_LIB_OK"] = "True" + +""" +This is a simple example to demonstrate training of normalizing flows. +Standard bivariate normal noise is sampled from the base distribution +and we learnt to transform it to a bivariate normal distribution with +independent but not identical components (see the produced figures). + +""" + + +def learn_bivariate_normal() -> None: + # Lazily instantiated flow plus base and target distributions + bijectors = bij.AffineAutoregressive( + params=params.DenseAutoregressive(hidden_dims=(32,)) + ) + base_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), 1 + ) + target_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2) + 5, torch.ones(2) * 0.5), 1 + ) + + # Instantiate transformed distribution and parameters + flow = dist.Flow(base_dist, bijectors) + + # Fixed samples for plotting + y_initial = flow.sample( + torch.Size( + [ + 300, + ] + ) + ) + y_target = target_dist.sample( + torch.Size( + [ + 300, + ] + ) + ) + + # Training loop + opt = torch.optim.Adam(flow.parameters(), lr=5e-3) + frame = 0 + for idx in range(3001): + opt.zero_grad() + + # Minimize KL(p || q) + y = target_dist.sample((1000,)) + loss = -flow.log_prob(y).mean() + + if idx % 500 == 0: + print("epoch", idx, "loss", loss) + + # Save SVG + y_learnt = ( + flow.sample( + torch.Size( + [ + 300, + ] + ) + ) + .detach() + .numpy() + ) + + plt.figure(figsize=(5, 5), dpi=100) + plt.plot( + y_target[:, 0], + y_target[:, 1], + "o", + color="blue", + alpha=0.95, + label="target", + ) + plt.plot( + y_initial[:, 0], + y_initial[:, 1], + "o", + color="grey", + alpha=0.95, + label="initial", + ) + plt.plot( + y_learnt[:, 0], + y_learnt[:, 1], + "o", + color="red", + alpha=0.95, + label="learnt", + ) + plt.xlim((-4, 8)) + plt.ylim((-4, 8)) + plt.xlabel("$x_1$") + plt.ylabel("$x_2$") + plt.legend(loc="lower right", facecolor=(1, 1, 1, 1.0)) + plt.savefig( + f"bivariate-normal-frame-{frame}.svg", + bbox_inches="tight", + transparent=True, + ) + + frame += 1 + + loss.backward() + opt.step() + + +if __name__ == "__main__": + learn_bivariate_normal() diff --git a/flowtorch/__init__.py b/flowtorch/__init__.py new file mode 100644 index 00000000..ab9c4091 --- /dev/null +++ b/flowtorch/__init__.py @@ -0,0 +1,6 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from flowtorch.lazy import Lazy, LazyMeta + +__all__ = ["Lazy", "LazyMeta"] diff --git a/flowtorch/bijectors/__init__.py b/flowtorch/bijectors/__init__.py new file mode 100644 index 00000000..1b768233 --- /dev/null +++ b/flowtorch/bijectors/__init__.py @@ -0,0 +1,97 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +""" +Warning: This file was generated by flowtorch/scripts/generate_imports.py +Do not modify or delete! + +""" + +import inspect +from typing import cast, List, Tuple + +import torch +from flowtorch.bijectors.affine import Affine +from flowtorch.bijectors.affine_autoregressive import AffineAutoregressive +from flowtorch.bijectors.affine_fixed import AffineFixed +from flowtorch.bijectors.autoregressive import Autoregressive +from flowtorch.bijectors.base import Bijector +from flowtorch.bijectors.compose import Compose +from flowtorch.bijectors.elementwise import Elementwise +from flowtorch.bijectors.elu import ELU +from flowtorch.bijectors.exp import Exp +from flowtorch.bijectors.fixed import Fixed +from flowtorch.bijectors.leaky_relu import LeakyReLU +from flowtorch.bijectors.permute import Permute +from flowtorch.bijectors.power import Power +from flowtorch.bijectors.sigmoid import Sigmoid +from flowtorch.bijectors.softplus import Softplus +from flowtorch.bijectors.spline import Spline +from flowtorch.bijectors.spline_autoregressive import SplineAutoregressive +from flowtorch.bijectors.tanh import Tanh +from flowtorch.bijectors.volume_preserving import VolumePreserving + +standard_bijectors = [ + ("Affine", Affine), + ("AffineAutoregressive", AffineAutoregressive), + ("AffineFixed", AffineFixed), + ("ELU", ELU), + ("Exp", Exp), + ("LeakyReLU", LeakyReLU), + ("Permute", Permute), + ("Power", Power), + ("Sigmoid", Sigmoid), + ("Softplus", Softplus), + ("Spline", Spline), + ("SplineAutoregressive", SplineAutoregressive), + ("Tanh", Tanh), +] + +meta_bijectors = [ + ("Elementwise", Elementwise), + ("Autoregressive", Autoregressive), + ("Fixed", Fixed), + ("Bijector", Bijector), + ("Compose", Compose), + ("VolumePreserving", VolumePreserving), +] + + +def isbijector(cls: type) -> bool: + # A class must inherit from flowtorch.Bijector to be considered a valid bijector + return issubclass(cls, Bijector) + + +def standard_bijector(cls: type) -> bool: + # "Standard bijectors" are the ones we can perform standard automated tests upon + return ( + inspect.isclass(cls) + and isbijector(cls) + and cls.__name__ not in [clx for clx, _ in meta_bijectors] + ) + + +# Determine invertible bijectors +invertible_bijectors = [] +for bij_name, cls in standard_bijectors: + # TODO: Use factored out version of the following + # Define plan for flow + event_dim = max(cls.domain.event_dim, 1) # type: ignore + event_shape = event_dim * [4] + # base_dist = dist.Normal(torch.zeros(event_shape), torch.ones(event_shape)) + bij = cls(shape=torch.Size(event_shape)) + + try: + y = torch.randn(*bij.forward_shape(event_shape)) + bij.inverse(y) + except NotImplementedError: + pass + else: + invertible_bijectors.append((bij_name, cls)) + + +__all__ = ["standard_bijectors", "meta_bijectors", "invertible_bijectors"] + [ + cls + for cls, _ in cast(List[Tuple[str, Bijector]], meta_bijectors) + + cast(List[Tuple[str, Bijector]], standard_bijectors) +] diff --git a/flowtorch/bijectors/affine.py b/flowtorch/bijectors/affine.py new file mode 100644 index 00000000..3e9b1f4f --- /dev/null +++ b/flowtorch/bijectors/affine.py @@ -0,0 +1,31 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import torch +from flowtorch.bijectors.elementwise import Elementwise +from flowtorch.bijectors.ops.affine import Affine as AffineOp + + +class Affine(AffineOp, Elementwise): + r""" + Elementwise bijector via the affine mapping :math:`\mathbf{y} = \mu + + \sigma \otimes \mathbf{x}` where $\mu$ and $\sigma$ are learnable parameters. + """ + + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + log_scale_min_clip: float = -5.0, + log_scale_max_clip: float = 3.0, + sigmoid_bias: float = 2.0, + ) -> None: + super().__init__(params, shape=shape, context_shape=context_shape) + self.log_scale_min_clip = log_scale_min_clip + self.log_scale_max_clip = log_scale_max_clip + self.sigmoid_bias = sigmoid_bias diff --git a/flowtorch/bijectors/affine_autoregressive.py b/flowtorch/bijectors/affine_autoregressive.py new file mode 100644 index 00000000..21c224de --- /dev/null +++ b/flowtorch/bijectors/affine_autoregressive.py @@ -0,0 +1,31 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import flowtorch.parameters +import torch +from flowtorch.bijectors.autoregressive import Autoregressive +from flowtorch.bijectors.ops.affine import Affine as AffineOp + + +class AffineAutoregressive(AffineOp, Autoregressive): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + log_scale_min_clip: float = -5.0, + log_scale_max_clip: float = 3.0, + sigmoid_bias: float = 2.0, + ) -> None: + super().__init__( + params, + shape=shape, + context_shape=context_shape, + ) + self.log_scale_min_clip = log_scale_min_clip + self.log_scale_max_clip = log_scale_max_clip + self.sigmoid_bias = sigmoid_bias diff --git a/flowtorch/bijectors/affine_fixed.py b/flowtorch/bijectors/affine_fixed.py new file mode 100644 index 00000000..9c39eab3 --- /dev/null +++ b/flowtorch/bijectors/affine_fixed.py @@ -0,0 +1,54 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import math +from typing import Optional + +import flowtorch +import torch +from flowtorch.bijectors.fixed import Fixed + + +class AffineFixed(Fixed): + r""" + Elementwise bijector via the affine mapping :math:`\mathbf{y} = \mu + + \sigma \otimes \mathbf{x}` where $\mu$ and $\sigma$ are fixed rather than + learnable. + """ + + # TODO: Handle non-scalar loc and scale with correct broadcasting semantics + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + loc: float = 0.0, + scale: float = 1.0 + ) -> None: + super().__init__(params, shape=shape, context_shape=context_shape) + self.loc = loc + self.scale = scale + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return self.loc + self.scale * x + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return (y - self.loc) / self.scale + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.full_like(x, math.log(abs(self.scale))) diff --git a/flowtorch/bijectors/autoregressive.py b/flowtorch/bijectors/autoregressive.py new file mode 100644 index 00000000..8593f11e --- /dev/null +++ b/flowtorch/bijectors/autoregressive.py @@ -0,0 +1,70 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Any, cast, Optional + +import flowtorch +import flowtorch.parameters +import torch +import torch.distributions.constraints as constraints +from flowtorch.bijectors.base import Bijector +from flowtorch.parameters.dense_autoregressive import DenseAutoregressive + + +class Autoregressive(Bijector): + # "Default" event shape is to operate on vectors + domain = constraints.real_vector + codomain = constraints.real_vector + + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + **kwargs: Any + ) -> None: + # Event shape is determined by `shape` argument + self.domain = constraints.independent(constraints.real, len(shape)) + self.codomain = constraints.independent(constraints.real, len(shape)) + + # currently only DenseAutoregressive has a `permutation` buffer + if not params: + params = DenseAutoregressive() # type: ignore + + # TODO: Replace P.DenseAutoregressive with P.Autoregressive + # In the future there will be other autoregressive parameter classes + assert params is not None and issubclass(params.cls, DenseAutoregressive) + + super().__init__(params, shape=shape, context_shape=context_shape) + + def inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + # TODO: Allow that context can have a batch shape + assert context is None # or context.shape == (self._context_size,) + params = self.params + assert params is not None + + x_new = torch.zeros_like(y) + # NOTE: Inversion is an expensive operation that scales in the + # dimension of the input + permutation = ( + params.permutation + ) # TODO: type-safe named buffer (e.g. "permutation") access + # TODO: Make permutation, inverse work for other event shapes + for idx in cast(torch.LongTensor, permutation): + x_new[..., idx] = self._inverse(y, x_new.clone(), context)[..., idx] + + return x_new + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + raise NotImplementedError diff --git a/flowtorch/bijectors/base.py b/flowtorch/bijectors/base.py new file mode 100644 index 00000000..0e6b7b40 --- /dev/null +++ b/flowtorch/bijectors/base.py @@ -0,0 +1,143 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Optional, Sequence, Union + +import flowtorch +import flowtorch.distributions +import flowtorch.parameters +import torch +import torch.distributions +from flowtorch.parameters import Parameters +from torch.distributions import constraints + + +class Bijector(metaclass=flowtorch.LazyMeta): + codomain: constraints.Constraint = constraints.real + domain: constraints.Constraint = constraints.real + _shape: torch.Size + _context_shape: Optional[torch.Size] + _params: Optional[Union[Parameters, torch.nn.ModuleList]] = None + + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + ) -> None: + # Prevent "meta bijectors" from being initialized + # NOTE: We define a "standard bijector" as one that inherits from a + # subclass of Bijector, hence why we need to test the length of the MRO + if ( + self.__class__.__module__ == "flowtorch.bijectors.base" + or len(self.__class__.__mro__) <= 3 + ): + raise TypeError("Only standard bijectors can be initialized.") + + self._shape = shape + self._context_shape = context_shape + + # Instantiate parameters (tensor, hypernets, etc.) + if params is not None: + param_shapes = self.param_shapes(shape) + self._params = params( # type: ignore + param_shapes, self._shape, self._context_shape + ) + + @property + def params(self) -> Optional[Union[Parameters, torch.nn.ModuleList]]: + return self._params + + @params.setter + def params(self, value: Optional[Union[Parameters, torch.nn.ModuleList]]) -> None: + self._params = value + + def forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + # TODO: Allow that context can have a batch shape + assert context is None # or context.shape == (self._context_size,) + return self._forward(x, context) + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + """ + Abstract method to compute forward transformation. + """ + raise NotImplementedError + + def inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + # TODO: Allow that context can have a batch shape + assert context is None # or context.shape == (self._context_size,) + return self._inverse(y, x, context) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + """ + Abstract method to compute inverse transformation. + """ + raise NotImplementedError + + def log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + """ + Computes the log det jacobian `log |dy/dx|` given input and output. + By default, assumes a volume preserving bijection. + """ + return self._log_abs_det_jacobian(x, y, context) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + """ + Computes the log det jacobian `log |dy/dx|` given input and output. + By default, assumes a volume preserving bijection. + """ + + # TODO: Sum out self.event_dim right-most dimensions + # self.event_dim may be > 0 for derived classes! + return torch.zeros_like(x) + + def param_shapes(self, shape: torch.Size) -> Sequence[torch.Size]: + """ + Abstract method to return shapes of parameters + """ + raise NotImplementedError + + def __repr__(self) -> str: + return self.__class__.__name__ + "()" + + def forward_shape(self, shape: torch.Size) -> torch.Size: + """ + Infers the shape of the forward computation, given the input shape. + Defaults to preserving shape. + """ + return shape + + def inverse_shape(self, shape: torch.Size) -> torch.Size: + """ + Infers the shapes of the inverse computation, given the output shape. + Defaults to preserving shape. + """ + return shape diff --git a/flowtorch/bijectors/compose.py b/flowtorch/bijectors/compose.py new file mode 100644 index 00000000..20f1606f --- /dev/null +++ b/flowtorch/bijectors/compose.py @@ -0,0 +1,88 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional, Sequence + +import flowtorch +import flowtorch.parameters +import torch +import torch.distributions +from flowtorch.bijectors.base import Bijector +from torch.distributions.utils import _sum_rightmost + + +class Compose(Bijector): + def __init__( + self, + bijectors: Sequence[flowtorch.Lazy], + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + ): + assert len(bijectors) > 0 + + # Instantiate all bijectors, propagating shape information + self.bijectors = [] + for bijector in bijectors: + assert issubclass(bijector.cls, Bijector) + + self.bijectors.append(bijector(shape=shape)) + shape = self.bijectors[-1].forward_shape(shape) # type: ignore + + self.domain = self.bijectors[0].domain # type: ignore + self.codomain = self.bijectors[-1].codomain # type: ignore + + # Make parameters accessible to dist.Flow + self._params = torch.nn.ModuleList( + [ + b._params # type: ignore + for b in self.bijectors + if isinstance(b._params, torch.nn.Module) # type: ignore + ] + ) + + self._context_shape = context_shape + + # NOTE: We overwrite forward rather than _forward so that the composed + # bijectors can handle the caching separately! + def forward(self, x: torch.Tensor, context: torch.Tensor = None) -> torch.Tensor: + for bijector in self.bijectors: + x = bijector.forward(x, context) # type: ignore + + return x + + def inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + for bijector in reversed(self.bijectors): + y = bijector.inverse(y, x, context) # type: ignore + + return y + + def log_abs_det_jacobian( + self, x: torch.Tensor, y: torch.Tensor, context: torch.Tensor = None + ) -> torch.Tensor: + """ + Computes the log det jacobian `log |dy/dx|` given input and output. + By default, assumes a volume preserving bijection. + """ + ldj = _sum_rightmost( + torch.zeros_like(y), + self.domain.event_dim, + ) + for bijector in reversed(self.bijectors): + y_inv = bijector.inverse(y, context) # type: ignore + ldj += bijector.log_abs_det_jacobian(y_inv, y, context) # type: ignore + y = y_inv + return ldj + + def param_shapes(self, shape: torch.Size) -> Sequence[torch.Size]: + """ + Given a base distribution, calculate the parameters for the transformation + of that distribution under this bijector. By default, no parameters are + set. + """ + return [] diff --git a/flowtorch/bijectors/elementwise.py b/flowtorch/bijectors/elementwise.py new file mode 100644 index 00000000..e4805cc5 --- /dev/null +++ b/flowtorch/bijectors/elementwise.py @@ -0,0 +1,26 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Any, Optional + +import flowtorch +import torch +import torch.distributions +from flowtorch.bijectors.base import Bijector +from flowtorch.parameters.tensor import Tensor + + +class Elementwise(Bijector): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + **kwargs: Any + ) -> None: + if not params: + params = Tensor() # type: ignore + + assert params is None or issubclass(params.cls, Tensor) + + super().__init__(params, shape=shape, context_shape=context_shape) diff --git a/flowtorch/bijectors/elu.py b/flowtorch/bijectors/elu.py new file mode 100644 index 00000000..63332f99 --- /dev/null +++ b/flowtorch/bijectors/elu.py @@ -0,0 +1,41 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import torch +import torch.distributions.constraints as constraints +import torch.nn.functional as F +from flowtorch.bijectors.fixed import Fixed +from flowtorch.ops import eps + + +class ELU(Fixed): + codomain = constraints.greater_than(-1.0) + + # TODO: Setting the alpha value of ELU as __init__ argument + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return F.elu(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.max(y, torch.zeros_like(y)) + torch.min( + torch.log1p(y + eps), torch.zeros_like(y) + ) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return -F.relu(-x) diff --git a/flowtorch/bijectors/exp.py b/flowtorch/bijectors/exp.py new file mode 100644 index 00000000..2b9d73e6 --- /dev/null +++ b/flowtorch/bijectors/exp.py @@ -0,0 +1,38 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import torch +import torch.distributions.constraints as constraints +from flowtorch.bijectors.fixed import Fixed + + +class Exp(Fixed): + r""" + Elementwise bijector via the mapping :math:`y = \exp(x)`. + """ + codomain = constraints.positive + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.exp(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return y.log() + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return x diff --git a/flowtorch/bijectors/fixed.py b/flowtorch/bijectors/fixed.py new file mode 100644 index 00000000..ad8deb1b --- /dev/null +++ b/flowtorch/bijectors/fixed.py @@ -0,0 +1,32 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Optional, Sequence + +import flowtorch +import torch +import torch.distributions +from flowtorch.bijectors.base import Bijector + + +class Fixed(Bijector): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + ) -> None: + # TODO: In the future, make Fixed actually mean that there is no autograd + # through params + super().__init__(params, shape=shape, context_shape=context_shape) + assert params is None + + def param_shapes(self, shape: torch.Size) -> Sequence[torch.Size]: + """ + Given a base distribution, calculate the parameters for the transformation + of that distribution under this bijector. By default, no parameters are + set. + """ + # TODO: In the future, make Fixed actually mean that there is no autograd + # through params + return [] diff --git a/flowtorch/bijectors/leaky_relu.py b/flowtorch/bijectors/leaky_relu.py new file mode 100644 index 00000000..b8136575 --- /dev/null +++ b/flowtorch/bijectors/leaky_relu.py @@ -0,0 +1,38 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import math +from typing import Optional + +import torch +import torch.nn.functional as F +from flowtorch.bijectors.fixed import Fixed + + +class LeakyReLU(Fixed): + # TODO: Setting the slope of Leaky ReLU as __init__ argument + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return F.leaky_relu(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return F.leaky_relu(y, negative_slope=100.0) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.where( + x >= 0.0, torch.zeros_like(x), torch.ones_like(x) * math.log(0.01) + ) diff --git a/flowtorch/bijectors/ops/__init__.py b/flowtorch/bijectors/ops/__init__.py new file mode 100644 index 00000000..bb884103 --- /dev/null +++ b/flowtorch/bijectors/ops/__init__.py @@ -0,0 +1,2 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT diff --git a/flowtorch/bijectors/ops/affine.py b/flowtorch/bijectors/ops/affine.py new file mode 100644 index 00000000..b799ea7e --- /dev/null +++ b/flowtorch/bijectors/ops/affine.py @@ -0,0 +1,86 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional, Tuple + +import flowtorch +import torch +from flowtorch.bijectors.base import Bijector +from flowtorch.ops import clamp_preserve_gradients +from torch.distributions.utils import _sum_rightmost + + +class Affine(Bijector): + r""" + Affine mapping :math:`\mathbf{y} = \mu + \sigma \otimes \mathbf{x}` where + $\mu$ and $\sigma$ are learnable parameters. + + """ + + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + log_scale_min_clip: float = -5.0, + log_scale_max_clip: float = 3.0, + sigmoid_bias: float = 2.0, + ) -> None: + super().__init__(params, shape=shape, context_shape=context_shape) + self.log_scale_min_clip = log_scale_min_clip + self.log_scale_max_clip = log_scale_max_clip + self.sigmoid_bias = sigmoid_bias + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + params = self.params + assert params is not None + + mean, log_scale = params(x, context=context) + log_scale = clamp_preserve_gradients( + log_scale, self.log_scale_min_clip, self.log_scale_max_clip + ) + scale = torch.exp(log_scale) + y = scale * x + mean + return y + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + params = self.params + assert params is not None + + mean, log_scale = params(x, context=context) + log_scale = clamp_preserve_gradients( + log_scale, self.log_scale_min_clip, self.log_scale_max_clip + ) + inverse_scale = torch.exp(-log_scale) + x_new = (y - mean) * inverse_scale + return x_new + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + params = self.params + assert params is not None + + # Note: params will take care of caching "mean, log_scale = params(x)" + _, log_scale = params(x, context=context) + log_scale = clamp_preserve_gradients( + log_scale, self.log_scale_min_clip, self.log_scale_max_clip + ) + return _sum_rightmost(log_scale, self.domain.event_dim) + + def param_shapes(self, shape: torch.Size) -> Tuple[torch.Size, torch.Size]: + # A mean and log variance for every dimension of the event shape + return shape, shape diff --git a/flowtorch/bijectors/ops/spline.py b/flowtorch/bijectors/ops/spline.py new file mode 100644 index 00000000..ba2f9e8b --- /dev/null +++ b/flowtorch/bijectors/ops/spline.py @@ -0,0 +1,116 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +# This implementation is adapted in part from: +# * https://github.com/tonyduan/normalizing-flows/blob/master/nf/flows.py; +# * https://github.com/hmdolatabadi/LRS_NF/blob/master/nde/transforms/ +# nonlinearities.py; and, +# * https://github.com/bayesiains/nsf/blob/master/nde/transforms/splines/ +# rational_quadratic.py +# under the MIT license. + +from typing import Any, Optional, Sequence, Tuple + +import flowtorch +import torch +import torch.nn.functional as F +from flowtorch.bijectors.base import Bijector +from flowtorch.ops import monotonic_rational_spline +from torch.distributions.utils import _sum_rightmost + + +class Spline(Bijector): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + count_bins: int = 8, + bound: float = 3.0, + order: str = "linear" + ) -> None: + if order not in ["linear", "quadratic"]: + raise ValueError( + "Keyword argument 'order' must be one of ['linear', \ +'quadratic'], but '{}' was found!".format( + order + ) + ) + + self.count_bins = count_bins + self.bound = bound + self.order = order + + super().__init__(params, shape=shape, context_shape=context_shape) + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + y, _ = self._op(x, x, context) + return y + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + x_new, _ = self._op(y, x, context=context, inverse=True) + return x_new + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + _, log_detJ = self._op(x, x, context) + return _sum_rightmost(log_detJ, self.domain.event_dim) + + def _op( + self, + input: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + inverse: bool = False, + **kwargs: Any + ) -> Tuple[torch.Tensor, torch.Tensor]: + params = self.params + assert params is not None + + if self.order == "linear": + widths, heights, derivatives, lambdas = params(x, context=context) + lambdas = torch.sigmoid(lambdas) + else: + widths, heights, derivatives = params(x, context=context) + lambdas = None + + # Constrain parameters + # TODO: Move to flowtorch.ops function? + widths = F.softmax(widths, dim=-1) + heights = F.softmax(heights, dim=-1) + derivatives = F.softplus(derivatives) + + y, log_detJ = monotonic_rational_spline( + input, + widths, + heights, + derivatives, + lambdas, + bound=self.bound, + inverse=inverse, + **kwargs + ) + return y, log_detJ + + def param_shapes(self, shape: torch.Size) -> Sequence[torch.Size]: + s1 = torch.Size(shape + (self.count_bins,)) + s2 = torch.Size(shape + (self.count_bins - 1,)) + + if self.order == "linear": + return s1, s1, s2, s1 + else: + return s1, s1, s2 diff --git a/flowtorch/bijectors/permute.py b/flowtorch/bijectors/permute.py new file mode 100644 index 00000000..565bf64c --- /dev/null +++ b/flowtorch/bijectors/permute.py @@ -0,0 +1,60 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import torch +import torch.distributions.constraints as constraints +from flowtorch.bijectors.fixed import Fixed +from flowtorch.bijectors.volume_preserving import VolumePreserving +from torch.distributions.utils import lazy_property + + +class Permute(Fixed, VolumePreserving): + domain = constraints.real_vector + codomain = constraints.real_vector + + # TODO: A new abstraction so can defer construction of permutation + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + permutation: Optional[torch.Tensor] = None + ) -> None: + super().__init__(params, shape=shape, context_shape=context_shape) + self.permutation = permutation + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + if self.permutation is None: + self.permutation = torch.randperm(x.shape[-1]) + + return torch.index_select(x, -1, self.permutation) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + if self.permutation is None: + self.permutation = torch.randperm(y.shape[-1]) + + return torch.index_select(y, -1, self.inv_permutation) + + @lazy_property + def inv_permutation(self) -> Optional[torch.Tensor]: + if self.permutation is None: + return None + + result = torch.empty_like(self.permutation, dtype=torch.long) + result[self.permutation] = torch.arange( + self.permutation.size(0), dtype=torch.long, device=self.permutation.device + ) + return result diff --git a/flowtorch/bijectors/power.py b/flowtorch/bijectors/power.py new file mode 100644 index 00000000..a26521d1 --- /dev/null +++ b/flowtorch/bijectors/power.py @@ -0,0 +1,52 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import torch +import torch.distributions.constraints as constraints +from flowtorch.bijectors.fixed import Fixed + + +class Power(Fixed): + r""" + Elementwise bijector via the mapping :math:`y = x^{\text{exponent}}`. + """ + domain = constraints.positive + codomain = constraints.positive + + # TODO: Tensor valued exponents and corresponding determination of event_dim + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + exponent: float = 2.0, + ) -> None: + super().__init__(params, shape=shape, context_shape=context_shape) + self.exponent = exponent + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return x.pow(self.exponent) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return y.pow(1 / self.exponent) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.abs(self.exponent * y / x).log() diff --git a/flowtorch/bijectors/sigmoid.py b/flowtorch/bijectors/sigmoid.py new file mode 100644 index 00000000..a28069e4 --- /dev/null +++ b/flowtorch/bijectors/sigmoid.py @@ -0,0 +1,39 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import torch +import torch.distributions.constraints as constraints +import torch.nn.functional as F +from flowtorch.bijectors.fixed import Fixed +from flowtorch.ops import clipped_sigmoid + + +class Sigmoid(Fixed): + codomain = constraints.unit_interval + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return clipped_sigmoid(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + finfo = torch.finfo(y.dtype) + y = y.clamp(min=finfo.tiny, max=1.0 - finfo.eps) + return y.log() - torch.log1p(-y) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return -F.softplus(-x) - F.softplus(x) diff --git a/flowtorch/bijectors/softplus.py b/flowtorch/bijectors/softplus.py new file mode 100644 index 00000000..5459fd50 --- /dev/null +++ b/flowtorch/bijectors/softplus.py @@ -0,0 +1,40 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch.ops +import torch +import torch.distributions.constraints as constraints +import torch.nn.functional as F +from flowtorch.bijectors.fixed import Fixed + + +class Softplus(Fixed): + r""" + Elementwise bijector via the mapping :math:`\text{Softplus}(x) = \log(1 + \exp(x))`. + """ + codomain = constraints.positive + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return F.softplus(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return flowtorch.ops.softplus_inv(y) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return -F.softplus(-x) diff --git a/flowtorch/bijectors/spline.py b/flowtorch/bijectors/spline.py new file mode 100644 index 00000000..479a7d5e --- /dev/null +++ b/flowtorch/bijectors/spline.py @@ -0,0 +1,30 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import torch +from flowtorch.bijectors.elementwise import Elementwise +from flowtorch.bijectors.ops.spline import Spline as SplineOp + + +class Spline(SplineOp, Elementwise): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + count_bins: int = 8, + bound: float = 3.0, + order: str = "linear" + ) -> None: + super().__init__( + params, + shape=shape, + context_shape=context_shape, + count_bins=count_bins, + bound=bound, + order=order, + ) diff --git a/flowtorch/bijectors/spline_autoregressive.py b/flowtorch/bijectors/spline_autoregressive.py new file mode 100644 index 00000000..ae29c787 --- /dev/null +++ b/flowtorch/bijectors/spline_autoregressive.py @@ -0,0 +1,31 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional + +import flowtorch +import flowtorch.parameters +import torch +from flowtorch.bijectors.autoregressive import Autoregressive +from flowtorch.bijectors.ops.spline import Spline as SplineOp + + +class SplineAutoregressive(SplineOp, Autoregressive): + def __init__( + self, + params: Optional[flowtorch.Lazy] = None, + *, + shape: torch.Size, + context_shape: Optional[torch.Size] = None, + count_bins: int = 8, + bound: float = 3.0, + order: str = "linear" + ) -> None: + super().__init__( + params, + shape=shape, + context_shape=context_shape, + count_bins=count_bins, + bound=bound, + order=order, + ) diff --git a/flowtorch/bijectors/tanh.py b/flowtorch/bijectors/tanh.py new file mode 100644 index 00000000..82d6500f --- /dev/null +++ b/flowtorch/bijectors/tanh.py @@ -0,0 +1,40 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import math +from typing import Optional + +import torch +import torch.distributions.constraints as constraints +import torch.nn.functional as F +from flowtorch.bijectors.fixed import Fixed + + +class Tanh(Fixed): + r""" + Transform via the mapping :math:`y = \tanh(x)`. + """ + codomain = constraints.interval(-1.0, 1.0) + + def _forward( + self, + x: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.tanh(x) + + def _inverse( + self, + y: torch.Tensor, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return torch.atanh(y) + + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + return 2.0 * (math.log(2.0) - x - F.softplus(-2.0 * x)) diff --git a/flowtorch/bijectors/volume_preserving.py b/flowtorch/bijectors/volume_preserving.py new file mode 100644 index 00000000..3eb9cc7f --- /dev/null +++ b/flowtorch/bijectors/volume_preserving.py @@ -0,0 +1,24 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Optional + +import torch +import torch.distributions +from flowtorch.bijectors.base import Bijector + + +class VolumePreserving(Bijector): + def _log_abs_det_jacobian( + self, + x: torch.Tensor, + y: torch.Tensor, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + # TODO: Confirm that this should involve `x`/`self.domain` and not + # `y`/`self.codomain` + return torch.zeros( + x.size()[: -self.domain.event_dim], + dtype=x.dtype, + layout=x.layout, # pyre-ignore[16] + device=x.device, + ) diff --git a/flowtorch/distributions/__init__.py b/flowtorch/distributions/__init__.py new file mode 100644 index 00000000..5e63c266 --- /dev/null +++ b/flowtorch/distributions/__init__.py @@ -0,0 +1,13 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +""" +Warning: This file was generated by flowtorch/scripts/generate_imports.py +Do not modify or delete! + +""" + +from flowtorch.distributions.flow import Flow +from flowtorch.distributions.neals_funnel import NealsFunnel + +__all__ = ["Flow", "NealsFunnel"] diff --git a/flowtorch/distributions/flow.py b/flowtorch/distributions/flow.py new file mode 100644 index 00000000..11b09654 --- /dev/null +++ b/flowtorch/distributions/flow.py @@ -0,0 +1,127 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Any, Dict, Optional, Union + +import flowtorch +import torch +import torch.distributions as dist +from torch import Tensor +from torch.distributions.utils import _sum_rightmost + + +class Flow(torch.nn.Module, dist.Distribution, metaclass=flowtorch.LazyMeta): + _default_sample_shape = torch.Size() + arg_constraints: Dict[str, dist.constraints.Constraint] = {} + + def __init__( + self, + base_dist: dist.Distribution, + bijector: flowtorch.Lazy, + validate_args: Any = None, + ) -> None: + torch.nn.Module.__init__(self) + + self.base_dist = base_dist + self._context: Optional[torch.Tensor] = None + self.bijector = bijector(shape=base_dist.event_shape) + + # Required so that parameters are registered with nn.Module + self.params = self.bijector._params # type: ignore + + # TODO: Confirm that the following logic works. Shouldn't it use + # .domain and .codomain?? Infer shape from constructed self.bijector + shape = ( + self.base_dist.batch_shape + self.base_dist.event_shape # pyre-ignore[16] + ) + event_dim = self.bijector.domain.event_dim # type: ignore + event_dim = max(event_dim, len(self.base_dist.event_shape)) + batch_shape = shape[: len(shape) - event_dim] + event_shape = shape[len(shape) - event_dim :] + + dist.Distribution.__init__( + self, batch_shape, event_shape, validate_args=validate_args + ) + + def condition(self, context: torch.Tensor) -> "Flow": + self._context = context + return self + + def sample( + self, + sample_shape: Union[Tensor, torch.Size] = _default_sample_shape, + context: Optional[torch.Tensor] = None, + ) -> Tensor: + """ + Generates a sample_shape shaped sample or sample_shape shaped batch of + samples if the distribution parameters are batched. Samples first from + base distribution and applies `transform()` for every transform in the + list. + """ + if context is None: + context = self._context + with torch.no_grad(): + x = self.base_dist.sample(sample_shape) + x = self.bijector.forward(x, context) # type: ignore + return x + + def rsample( + self, + sample_shape: Union[Tensor, torch.Size] = _default_sample_shape, + context: Optional[torch.Tensor] = None, + ) -> Tensor: + """ + Generates a sample_shape shaped reparameterized sample or sample_shape + shaped batch of reparameterized samples if the distribution parameters + are batched. Samples first from base distribution and applies + `transform()` for every transform in the list. + """ + if context is None: + context = self._context + x = self.base_dist.rsample(sample_shape) + x = self.bijector.forward(x, context) # type: ignore + return x + + def rnormalize( + self, value: torch.Tensor, context: Optional[torch.Tensor] = None + ) -> Tensor: + """ + Push a tensor through the normalizing direction of the flow where + we can take autodiff gradients on the bijector. + """ + if context is None: + context = self._context + + return self.bijector.inverse(value, context) # type: ignore + + def normalize( + self, value: torch.Tensor, context: Optional[torch.Tensor] = None + ) -> Tensor: + """ + Push a tensor through the normalizing direction of the flow and + block autodiff gradients on the bijector. + """ + with torch.no_grad(): + return self.rnormalize(value, context) + + def log_prob( + self, value: torch.Tensor, context: Optional[torch.Tensor] = None + ) -> torch.Tensor: + """ + Scores the sample by inverting the transform(s) and computing the score + using the score of the base distribution and the log abs det jacobian. + """ + if context is None: + context = self._context + event_dim = len(self.event_shape) # pyre-ignore[16] + + x = self.bijector.inverse(value, context) # type: ignore + log_prob = -_sum_rightmost( + self.bijector.log_abs_det_jacobian(x, value, context), # type: ignore + event_dim - self.bijector.domain.event_dim, # type: ignore + ) + log_prob = log_prob + _sum_rightmost( + self.base_dist.log_prob(x), + event_dim - len(self.base_dist.event_shape), # pyre-ignore[16] + ) + + return log_prob diff --git a/flowtorch/distributions/neals_funnel.py b/flowtorch/distributions/neals_funnel.py new file mode 100644 index 00000000..56a1e8ca --- /dev/null +++ b/flowtorch/distributions/neals_funnel.py @@ -0,0 +1,53 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Any, Dict, Optional, Union + +import torch +import torch.distributions as dist +from torch.distributions import constraints +from torch.distributions.utils import _standard_normal + + +class NealsFunnel(dist.Distribution): + """ + Neal's funnel. + p(x,y) = N(y|0,3) N(x|0,exp(y/2)) + """ + + support = constraints.real + arg_constraints: Dict[str, dist.constraints.Constraint] = {} + + def __init__(self, validate_args: Any = None) -> None: + d = 2 + batch_shape, event_shape = torch.Size([]), (d,) + super(NealsFunnel, self).__init__( + batch_shape, event_shape, validate_args=validate_args + ) + + def rsample( + self, + sample_shape: Union[torch.Tensor, torch.Size] = None, + context: Optional[torch.Tensor] = None, + ) -> torch.Tensor: + if not sample_shape: + sample_shape = torch.Size() + eps = _standard_normal( + (sample_shape[0], 2), dtype=torch.float, device=torch.device("cpu") + ) + z = torch.zeros(eps.shape) + z[..., 1] = torch.tensor(3.0) * eps[..., 1] + z[..., 0] = torch.exp(z[..., 1] / 2.0) * eps[..., 0] + return z + + def log_prob( + self, value: torch.Tensor, context: Optional[torch.Tensor] = None + ) -> torch.Tensor: + if self._validate_args: + self._validate_sample(value) + x = value[..., 0] + y = value[..., 1] + + log_prob = dist.Normal(0, 3).log_prob(y) + log_prob += dist.Normal(0, torch.exp(y / 2)).log_prob(x) + + return log_prob diff --git a/flowtorch/docs.py b/flowtorch/docs.py new file mode 100644 index 00000000..b1a1749b --- /dev/null +++ b/flowtorch/docs.py @@ -0,0 +1,159 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import importlib +import types +from collections import OrderedDict +from functools import lru_cache +from inspect import isclass, isfunction, ismodule +from typing import Any, Dict, Sequence, Mapping, Tuple + +# We don't want to include, e.g. both flowtorch.bijectors.Affine and +# flowtorch.bijectors.affine.Affine. Hence, we specify a list of modules +# to explicitly include in the API docs (and don't recurse on them). +# TODO: Include flowtorch.ops and flowtorch.numerical + +include_modules = [ + "flowtorch", + "flowtorch.bijectors", + "flowtorch.distributions", + # "flowtorch.experimental.parameters", + "flowtorch.nn", + "flowtorch.ops", + "flowtorch.parameters", + "flowtorch.utils", +] + + +def ispublic(name: str) -> bool: + return not name.startswith("_") + + +@lru_cache(maxsize=1) +def _documentable_modules() -> Mapping[types.ModuleType, Sequence[Tuple[str, Any]]]: + """ + Returns a list of (module, [(name, entity), ...]) pairs for modules + that are documentable + """ + + # TODO: Self document flowtorch.docs module + results = {} + + def dfs(dict: Mapping[str, Any]) -> None: + for key, val in dict.items(): + module = importlib.import_module(key) + entities = [ + (n, getattr(module, n)) + for n in sorted( + [ + n + for n in dir(module) + if ispublic(n) + and ( + isclass(getattr(module, n)) + or isfunction(getattr(module, n)) + ) + ] + ) + ] + results[module] = entities + + dfs(val) + + # Depth first search over module hierarchy, loading modules and extracting entities + dfs(_module_hierarchy()) + return results + + +@lru_cache(maxsize=1) +def _documentable_entities() -> Tuple[Sequence[str], Dict[str, Any]]: + """ + Returns a list of (str, entity) pairs for entities that are documentable + """ + + name_entity_mapping = {} + documentable_modules = _documentable_modules() + for module, entities in documentable_modules.items(): + if len(entities) > 0: + name_entity_mapping[module.__name__] = module + + for name, entity in entities: + qualified_name = f"{module.__name__}.{name}" + name_entity_mapping[qualified_name] = entity + + sorted_entity_names = sorted(name_entity_mapping.keys()) + return sorted_entity_names, name_entity_mapping + + +@lru_cache(maxsize=1) +def _module_hierarchy() -> Mapping[str, Any]: + # Make list of modules to search and their hierarchy + results: Dict[str, Any] = OrderedDict() + for module in sorted(include_modules): + submodules = module.split(".") + this_dict = results.setdefault(submodules[0], {}) + + for idx in range(1, len(submodules)): + submodule = ".".join(submodules[0 : (idx + 1)]) + this_dict.setdefault(submodule, {}) + this_dict = this_dict[submodule] + + return results + + +def generate_markdown(name: str, entity: Any) -> Tuple[str, str]: + """ + TODO: Method that inputs an object, extracts signature/docstring, + and formats as markdown + TODO: Method that build index markdown for overview files + The overview for the entire API is a special case + """ + + if name == "": + header = """--- +id: overview +sidebar_label: "Overview" +slug: "/api" +--- + +:::info + +These API stubs are generated from Python via a custom script and will filled +out in the future. + +::: + +""" + filename = "../website/docs/api/overview.mdx" + return filename, header + + # Regular modules/functions + item = { + "id": name, + "sidebar_label": "Overview" if ismodule(entity) else name.split(".")[-1], + "slug": f"/api/{name}", + "ref": entity, + "filename": f"../website/docs/api/{name}.mdx", + } + + header = f"""--- +id: {item['id']} +sidebar_label: {item['sidebar_label']} +slug: {item['slug']} +---""" + + markdown = header + return item["filename"], markdown + + +module_hierarchy = _module_hierarchy() +documentable_modules = _documentable_modules() +sorted_entity_names, name_entity_mapping = _documentable_entities() + +__all__ = [ + "documentable_modules", + "generate_markdown", + "module_hierarchy", + "name_entity_mapping", + "sorted_entity_names", +] diff --git a/flowtorch/lazy.py b/flowtorch/lazy.py new file mode 100644 index 00000000..d1572104 --- /dev/null +++ b/flowtorch/lazy.py @@ -0,0 +1,107 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import inspect +from collections import OrderedDict +from typing import Tuple, Mapping, Any + + +# TODO: Move functions to flowtorch.utils? +def partial_signature( + sig: inspect.Signature, *args: Any, **kwargs: Any +) -> Tuple[inspect.Signature, Mapping[str, Any]]: + """ + Given an inspect.Signature object and a dictionary of (name, val) pairs, + bind the names to the signature and return a new modified signature + """ + bindings = dict(sig.bind_partial(*args, **kwargs).arguments) + + old_parameters = sig.parameters + new_parameters = OrderedDict() + + for param_name in old_parameters: + if param_name not in bindings: + new_parameters[param_name] = old_parameters[param_name] + + bound_sig = sig.replace(parameters=list(new_parameters.values())) + + return bound_sig, bindings + + +def count_unbound(sig: inspect.Signature) -> int: + return len( + [p for p, v in sig.parameters.items() if v.default is inspect.Parameter.empty] + ) + + +class LazyMeta(type): + def __call__(cls: Any, *args: Any, **kwargs: Any) -> Any: + """ + Intercept instance creation + """ + # Special behaviour for Lazy class + if cls.__qualname__ == "Lazy": + lazy_cls = args[0] + args = args[1:] + else: + lazy_cls = cls + + # Remove first argument (i.e., self) from signature of class' initializer + sig = inspect.signature(lazy_cls.__init__) + new_parameters = OrderedDict( + [(k, v) for idx, (k, v) in enumerate(sig.parameters.items()) if idx != 0] + ) + sig = sig.replace(parameters=list(new_parameters.values())) + + # Attempt binding arguments to initializer + bound_sig, bindings = partial_signature(sig, *args, **kwargs) + + # If there are no unbound arguments then instantiate class + if not count_unbound(bound_sig): + return type.__call__(lazy_cls, *args, **kwargs) + + # Otherwise, return Lazy instance + else: + return type.__call__(Lazy, lazy_cls, bindings, sig, bound_sig) + + +class Lazy(metaclass=LazyMeta): + """ + Represents delayed instantiation of a class. + """ + + def __init__( + self, + cls: Any, + bindings: Mapping[str, Any], + sig: inspect.Signature, + bound_sig: inspect.Signature, + ): + self.cls = cls + self.bindings = bindings + self.sig = sig + self.bound_sig = bound_sig + + def __repr__(self) -> str: + return f"Lazy(cls={self.cls.__name__}, bindings={self.bindings})" + + def __call__(self, *args: Any, **kwargs: Any) -> "Lazy": + """ + Apply additional bindings + """ + new_bindings = dict(self.bound_sig.bind_partial(*args, **kwargs).arguments) + new_bindings.update(self.bindings) + + # Update args and kwargs + new_args = [] + new_kwargs = {} + for n, p in self.sig.parameters.items(): + if n in new_bindings: + if p.kind == inspect.Parameter.POSITIONAL_ONLY: + new_args.append(new_bindings[n]) + + else: + new_kwargs[n] = new_bindings[n] + + # Attempt object creation + return Lazy(self.cls, *new_args, **new_kwargs) diff --git a/flowtorch/nn/__init__.py b/flowtorch/nn/__init__.py new file mode 100644 index 00000000..0c9263f7 --- /dev/null +++ b/flowtorch/nn/__init__.py @@ -0,0 +1,7 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + + +from flowtorch.nn.made import MaskedLinear, create_mask + +__all__ = ["create_mask", "MaskedLinear"] diff --git a/flowtorch/nn/made.py b/flowtorch/nn/made.py new file mode 100644 index 00000000..3556801e --- /dev/null +++ b/flowtorch/nn/made.py @@ -0,0 +1,121 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Sequence, Tuple + +import torch +import torch.nn as nn +from torch.nn import functional as F + + +def sample_mask_indices( + input_dim: int, hidden_dim: int, simple: bool = True +) -> torch.Tensor: + """ + Samples the indices assigned to hidden units during the construction of MADE masks + :param input_dim: the dimensionality of the input variable + :param hidden_dim: the dimensionality of the hidden layer + :param simple: True to space fractional indices by rounding to nearest + int, false round randomly + """ + indices = torch.linspace(1, input_dim, steps=hidden_dim, device="cpu").to( + torch.Tensor().device + ) + if simple: + # Simple procedure tries to space fractional indices evenly by rounding + # to nearest int + return torch.round(indices) + else: + # "Non-simple" procedure creates fractional indices evenly then rounds + # at random + ints = indices.floor() + ints += torch.bernoulli(indices - ints) + return ints + + +def create_mask( + input_dim: int, + context_dim: int, + hidden_dims: Sequence[int], + permutation: torch.LongTensor, + output_multiplier: int, +) -> Tuple[Sequence[torch.Tensor], torch.Tensor]: + """ + Creates MADE masks for a conditional distribution + :param input_dim: the dimensionality of the input variable + :param context_dim: the dimensionality of the variable that is + conditioned on (for conditional densities) + :param hidden_dims: the dimensionality of the hidden layers(s) + :param permutation: the order of the input variables + :param output_multipliers: tiles the output (e.g. for when a separate + mean and scale parameter are desired) + """ + # Create mask indices for input, hidden layers, and final layer + # We use 0 to refer to the elements of the variable being conditioned on, + # and range(1:(D_latent+1)) for the input variable + var_index = torch.empty(permutation.shape, dtype=torch.get_default_dtype()) + var_index[permutation] = torch.arange(input_dim, dtype=torch.get_default_dtype()) + + # Create the indices that are assigned to the neurons + input_indices = torch.cat((torch.zeros(context_dim), 1 + var_index)) + + # For conditional MADE, introduce a 0 index that all the conditioned + # variables are connected to as per Paige and Wood (2016) (see below) + if context_dim > 0: + hidden_indices = [sample_mask_indices(input_dim, h) - 1 for h in hidden_dims] + else: + hidden_indices = [sample_mask_indices(input_dim - 1, h) for h in hidden_dims] + + # *** TODO: Fix this line *** + output_indices = ( + (var_index + 1).unsqueeze(-1).repeat(1, output_multiplier).reshape(-1) + ) + + # Create mask from input to output for the skips connections + mask_skip = (output_indices.unsqueeze(-1) > input_indices.unsqueeze(0)).type_as( + var_index + ) + + # Create mask from input to first hidden layer, and between subsequent + # hidden layers + masks = [ + (hidden_indices[0].unsqueeze(-1) >= input_indices.unsqueeze(0)).type_as( + var_index + ) + ] + for i in range(1, len(hidden_dims)): + masks.append( + ( + hidden_indices[i].unsqueeze(-1) >= hidden_indices[i - 1].unsqueeze(0) + ).type_as(var_index) + ) + + # Create mask from last hidden layer to output layer + masks.append( + (output_indices.unsqueeze(-1) > hidden_indices[-1].unsqueeze(0)).type_as( + var_index + ) + ) + + return masks, mask_skip + + +class MaskedLinear(nn.Linear): + """ + A linear mapping with a given mask on the weights (arbitrary bias) + :param in_features: the number of input features + :param out_features: the number of output features + :param mask: the mask to apply to the in_features x out_features weight matrix + :param bias: whether or not `MaskedLinear` should include a bias term. + defaults to `True` + """ + + def __init__( + self, in_features: int, out_features: int, mask: torch.Tensor, bias: bool = True + ) -> None: + super().__init__(in_features, out_features, bias) + self.register_buffer("mask", mask.data) + + def forward(self, _input: torch.Tensor) -> torch.Tensor: + masked_weight = self.weight * self.mask + return F.linear(_input, masked_weight, self.bias) diff --git a/flowtorch/ops/__init__.py b/flowtorch/ops/__init__.py new file mode 100644 index 00000000..876a4124 --- /dev/null +++ b/flowtorch/ops/__init__.py @@ -0,0 +1,303 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional, Tuple + +import torch +import torch.nn.functional as F + +eps = 1e-8 + + +def clamp_preserve_gradients(x: torch.Tensor, min: float, max: float) -> torch.Tensor: + """ + This helper function clamps gradients but still passes through the + gradient in clamped regions + """ + return x + (x.clamp(min, max) - x).detach() + + +def clipped_sigmoid(x: torch.Tensor) -> torch.Tensor: + finfo = torch.finfo(x.dtype) + return torch.clamp(torch.sigmoid(x), min=finfo.tiny, max=1.0 - finfo.eps) + + +def softplus_inv(y: torch.Tensor) -> torch.Tensor: + return y + y.neg().expm1().neg().log() + + +def _searchsorted(sorted_sequence: torch.Tensor, values: torch.Tensor) -> torch.Tensor: + """ + Searches for which bin an input belongs to (in a way that is parallelizable and + amenable to autodiff) + TODO: Replace with torch.searchsorted once it is released + """ + return torch.sum(values[..., None] >= sorted_sequence, dim=-1) - 1 + + +def _select_bins(x: torch.Tensor, idx: torch.Tensor) -> torch.Tensor: + """ + Performs gather to select the bin in the correct way on batched inputs + """ + idx = idx.clamp(min=0, max=x.size(-1) - 1) + + """ + Broadcast dimensions of idx over x + idx ~ (batch_dims, input_dim, 1) + x ~ (context_batch_dims, input_dim, count_bins) + Note that by convention, the context variable batch dimensions must broadcast + over the input batch dimensions. + """ + if len(idx.shape) >= len(x.shape): + x = x.reshape((1,) * (len(idx.shape) - len(x.shape)) + x.shape) + x = x.expand(idx.shape[:-2] + (-1,) * 2) + + return x.gather(-1, idx).squeeze(-1) + + +def _calculate_knots( + lengths: torch.Tensor, lower: float, upper: float +) -> Tuple[torch.Tensor, torch.Tensor]: + """ + Given a tensor of unscaled bin lengths that sum to 1, plus the lower and upper + limits, returns the shifted and scaled lengths plus knot positions + """ + + # Cumulative widths gives x (y for inverse) position of knots + knots = torch.cumsum(lengths, dim=-1) + + # Pad left of last dimension with 1 zero to compensate for dim lost to cumsum + knots = F.pad(knots, pad=(1, 0), mode="constant", value=0.0) + + # Translate [0,1] knot points to [-B, B] + knots = (upper - lower) * knots + lower + + # Convert the knot points back to lengths + # NOTE: Are following two lines a necessary fix for accumulation (round-off) error? + knots[..., 0] = lower + knots[..., -1] = upper + lengths = knots[..., 1:] - knots[..., :-1] + + return lengths, knots + + +def monotonic_rational_spline( + inputs: torch.Tensor, + widths: torch.Tensor, + heights: torch.Tensor, + derivatives: torch.Tensor, + lambdas: Optional[torch.Tensor] = None, + inverse: bool = False, + bound: float = 3.0, + min_bin_width: float = 1e-3, + min_bin_height: float = 1e-3, + min_derivative: float = 1e-3, + min_lambda: float = 0.025, + eps: float = 1e-6, +) -> Tuple[torch.Tensor, torch.Tensor]: + """ + Calculating a monotonic rational spline (linear or quadratic) or its inverse, + plus the log(abs(detJ)) required for normalizing flows. + NOTE: I omit the docstring with parameter descriptions for this method since it + is not considered "public" yet! + """ + + # Ensure bound is positive + # NOTE: For simplicity, we apply the identity function outside [-B, B] X [-B, B] + # rather than allowing arbitrary corners to the bounding box. If you want a + # different bounding box you can apply an affine transform before and after the + # input. + assert bound > 0.0 + + num_bins = widths.shape[-1] + if min_bin_width * num_bins > 1.0: + raise ValueError("Minimal bin width too large for the number of bins") + if min_bin_height * num_bins > 1.0: + raise ValueError("Minimal bin height too large for the number of bins") + + # inputs, inside_interval_mask, outside_interval_mask ~ (batch_dim, input_dim) + left, right = -bound, bound + bottom, top = -bound, bound + inside_interval_mask = (inputs >= left) & (inputs <= right) + outside_interval_mask = ~inside_interval_mask + + # outputs, logabsdet ~ (batch_dim, input_dim) + outputs = torch.zeros_like(inputs) + logabsdet = torch.zeros_like(inputs) + + # For numerical stability, put lower/upper limits on parameters. E.g. give + # every bin min_bin_width, then add width fraction of remaining length. + # NOTE: Do this here rather than higher up because we want everything to + # ensure numerical stability within this function. + widths = min_bin_width + (1.0 - min_bin_width * num_bins) * widths + heights = min_bin_height + (1.0 - min_bin_height * num_bins) * heights + derivatives = min_derivative + derivatives + + # Cumulative widths are x (y for inverse) position of knots + # Similarly, cumulative heights are y (x for inverse) position of knots + widths, cumwidths = _calculate_knots(widths, left, right) + heights, cumheights = _calculate_knots(heights, bottom, top) + + # Pad left and right derivatives with fixed values at first and last knots + # These are 1 since the function is the identity outside the bounding box + # and the derivative is continuous. + # NOTE: Not sure why this is 1.0 - min_derivative rather than 1.0. I've + # copied this from original implementation + derivatives = F.pad( + derivatives, pad=(1, 1), mode="constant", value=1.0 - min_derivative + ) + + # Get the index of the bin that each input is in + # bin_idx ~ (batch_dim, input_dim, 1) + bin_idx = _searchsorted( + cumheights + eps if inverse else cumwidths + eps, inputs + ).unsqueeze(-1) + + # Select the value for the relevant bin for the variables + # used in the main calculation + input_widths = _select_bins(widths, bin_idx) + input_cumwidths = _select_bins(cumwidths, bin_idx) + input_cumheights = _select_bins(cumheights, bin_idx) + input_delta = _select_bins(heights / widths, bin_idx) + input_derivatives = _select_bins(derivatives, bin_idx) + input_derivatives_plus_one = _select_bins(derivatives[..., 1:], bin_idx) + input_heights = _select_bins(heights, bin_idx) + + # Calculate monotonic *linear* rational spline + if lambdas is not None: + lambdas = (1 - 2 * min_lambda) * lambdas + min_lambda + input_lambdas = _select_bins(lambdas, bin_idx) + + # The weight, w_a, at the left-hand-side of each bin + # We are free to choose w_a, so set it to 1 + wa = 1.0 + + # The weight, w_b, at the right-hand-side of each bin + # This turns out to be a multiple of the w_a + # TODO: Should this be done in log space for numerical stability? + wb = torch.sqrt(input_derivatives / input_derivatives_plus_one) * wa + + # The weight, w_c, at the division point of each bin + # Recall that each bin is divided into two parts so we have enough + # d.o.f. to fit spline + wc = ( + input_lambdas * wa * input_derivatives + + (1 - input_lambdas) * wb * input_derivatives_plus_one + ) / input_delta + + # Calculate y coords of bins + ya = input_cumheights + yb = input_heights + input_cumheights + yc = ((1.0 - input_lambdas) * wa * ya + input_lambdas * wb * yb) / ( + (1.0 - input_lambdas) * wa + input_lambdas * wb + ) + + if inverse: + numerator = (input_lambdas * wa * (ya - inputs)) * ( + inputs <= yc + ).float() + ( + (wc - input_lambdas * wb) * inputs + input_lambdas * wb * yb - wc * yc + ) * ( + inputs > yc + ).float() + + denominator = ((wc - wa) * inputs + wa * ya - wc * yc) * ( + inputs <= yc + ).float() + ((wc - wb) * inputs + wb * yb - wc * yc) * (inputs > yc).float() + + theta = numerator / denominator + + outputs = theta * input_widths + input_cumwidths + + derivative_numerator = ( + wa * wc * input_lambdas * (yc - ya) * (inputs <= yc).float() + + wb * wc * (1 - input_lambdas) * (yb - yc) * (inputs > yc).float() + ) * input_widths + + logabsdet = torch.log(derivative_numerator) - 2 * torch.log( + torch.abs(denominator) + ) + + else: + theta = (inputs - input_cumwidths) / input_widths + + numerator = (wa * ya * (input_lambdas - theta) + wc * yc * theta) * ( + theta <= input_lambdas + ).float() + (wc * yc * (1 - theta) + wb * yb * (theta - input_lambdas)) * ( + theta > input_lambdas + ).float() + + denominator = (wa * (input_lambdas - theta) + wc * theta) * ( + theta <= input_lambdas + ).float() + (wc * (1 - theta) + wb * (theta - input_lambdas)) * ( + theta > input_lambdas + ).float() + + outputs = numerator / denominator + + derivative_numerator = ( + wa * wc * input_lambdas * (yc - ya) * (theta <= input_lambdas).float() + + wb + * wc + * (1 - input_lambdas) + * (yb - yc) + * (theta > input_lambdas).float() + ) / input_widths + + logabsdet = torch.log(derivative_numerator) - 2 * torch.log( + torch.abs(denominator) + ) + + # Calculate monotonic *quadratic* rational spline + else: + if inverse: + a = (inputs - input_cumheights) * ( + input_derivatives + input_derivatives_plus_one - 2 * input_delta + ) + input_heights * (input_delta - input_derivatives) + b = input_heights * input_derivatives - (inputs - input_cumheights) * ( + input_derivatives + input_derivatives_plus_one - 2 * input_delta + ) + c = -input_delta * (inputs - input_cumheights) + + discriminant = b.pow(2) - 4 * a * c + assert (discriminant >= 0).all() + + root = (2 * c) / (-b - torch.sqrt(discriminant)) + outputs = root * input_widths + input_cumwidths + + theta_one_minus_theta = root * (1 - root) + denominator = input_delta + ( + (input_derivatives + input_derivatives_plus_one - 2 * input_delta) + * theta_one_minus_theta + ) + derivative_numerator = input_delta.pow(2) * ( + input_derivatives_plus_one * root.pow(2) + + 2 * input_delta * theta_one_minus_theta + + input_derivatives * (1 - root).pow(2) + ) + logabsdet = -(torch.log(derivative_numerator) - 2 * torch.log(denominator)) + + else: + theta = (inputs - input_cumwidths) / input_widths + theta_one_minus_theta = theta * (1 - theta) + + numerator = input_heights * ( + input_delta * theta.pow(2) + input_derivatives * theta_one_minus_theta + ) + denominator = input_delta + ( + (input_derivatives + input_derivatives_plus_one - 2 * input_delta) + * theta_one_minus_theta + ) + outputs = input_cumheights + numerator / denominator + + derivative_numerator = input_delta.pow(2) * ( + input_derivatives_plus_one * theta.pow(2) + + 2 * input_delta * theta_one_minus_theta + + input_derivatives * (1 - theta).pow(2) + ) + logabsdet = torch.log(derivative_numerator) - 2 * torch.log(denominator) + + # Apply the identity function outside the bounding box + outputs[outside_interval_mask] = inputs[outside_interval_mask] + logabsdet[outside_interval_mask] = 0.0 + return outputs, logabsdet diff --git a/flowtorch/parameters/__init__.py b/flowtorch/parameters/__init__.py new file mode 100644 index 00000000..eb42fca2 --- /dev/null +++ b/flowtorch/parameters/__init__.py @@ -0,0 +1,14 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +""" +Warning: This file was generated by flowtorch/scripts/generate_imports.py +Do not modify or delete! + +""" + +from flowtorch.parameters.base import Parameters +from flowtorch.parameters.dense_autoregressive import DenseAutoregressive +from flowtorch.parameters.tensor import Tensor + +__all__ = ["Parameters", "DenseAutoregressive", "Tensor"] diff --git a/flowtorch/parameters/base.py b/flowtorch/parameters/base.py new file mode 100644 index 00000000..592bc0c5 --- /dev/null +++ b/flowtorch/parameters/base.py @@ -0,0 +1,41 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +from typing import Optional, Sequence + +import torch +from flowtorch import LazyMeta + + +class Parameters(torch.nn.Module, metaclass=LazyMeta): + """ + Deferred initialization of parameters. + """ + + def __init__( + self, + param_shapes: Sequence[torch.Size], + input_shape: torch.Size, + context_shape: Optional[torch.Size], + ) -> None: + super().__init__() + self.input_shape = input_shape + self.param_shapes = param_shapes + self.context_shape = context_shape + + def forward( + self, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> Sequence[torch.Tensor]: + # TODO: Caching etc. + return self._forward(x, context) + + def _forward( + self, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> Sequence[torch.Tensor]: + # I raise an exception rather than using @abstractmethod and + # metaclass=ABC so that we can reserve the metaclass for lazy + # evaluation. + raise NotImplementedError() diff --git a/flowtorch/parameters/dense_autoregressive.py b/flowtorch/parameters/dense_autoregressive.py new file mode 100644 index 00000000..dc02de63 --- /dev/null +++ b/flowtorch/parameters/dense_autoregressive.py @@ -0,0 +1,185 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import warnings +from typing import Callable, Optional, Sequence + +import torch +import torch.nn as nn +from flowtorch.nn.made import MaskedLinear, create_mask +from flowtorch.parameters.base import Parameters + + +class DenseAutoregressive(Parameters): + autoregressive = True + + def __init__( + self, + param_shapes: Sequence[torch.Size], + input_shape: torch.Size, + context_shape: Optional[torch.Size], + *, + hidden_dims: Sequence[int] = (256, 256), + nonlinearity: Callable[[], nn.Module] = nn.ReLU, + permutation: Optional[torch.LongTensor] = None, + skip_connections: bool = False, + ) -> None: + super().__init__(param_shapes, input_shape, context_shape) + + # Check consistency of input_shape with param_shapes + # We need each param_shapes to match input_shape in + # its leftmost dimensions + for s in param_shapes: + assert len(s) >= len(input_shape) and s[: len(input_shape)] == input_shape + + self.hidden_dims = hidden_dims + self.nonlinearity = nonlinearity + self.skip_connections = skip_connections + self._build(input_shape, param_shapes, context_shape, permutation) + + def _build( + self, + input_shape: torch.Size, + param_shapes: Sequence[torch.Size], + context_shape: Optional[torch.Size], + permutation: Optional[torch.LongTensor], + ) -> None: + # Work out flattened input and output shapes + param_shapes_ = list(param_shapes) + input_dims = int(torch.sum(torch.tensor(input_shape)).int().item()) + if input_dims == 0: + input_dims = 1 # scalars represented by torch.Size([]) + if permutation is None: + # By default set a random permutation of variables, which is + # important for performance with multiple steps + permutation = torch.LongTensor( + torch.randperm(input_dims, device="cpu").to( + torch.LongTensor((1,)).device + ) + ) + else: + # The permutation is chosen by the user + permutation = torch.LongTensor(permutation) + + self.param_dims = [ + int(max(torch.prod(torch.tensor(s[len(input_shape) :])).item(), 1)) + for s in param_shapes_ + ] + + self.output_multiplier = sum(self.param_dims) + + if input_dims == 1: + warnings.warn( + "DenseAutoregressive input_dim = 1. " + "Consider using an affine transformation instead." + ) + + # Calculate the indices on the output corresponding to each parameter + # TODO: Is this logic correct??? + # ends = torch.cumsum( + # torch.tensor( + # [max(torch.prod(torch.tensor(s)).item(), 1) for s in param_shapes_] + # ), + # dim=0, + # ) + # starts = torch.cat((torch.zeros(1).type_as(ends), ends[:-1])) + # self.param_slices = [slice(s.item(), e.item()) for s, e in zip(starts, ends)] + + # Hidden dimension must be not less than the input otherwise it isn't + # possible to connect to the outputs correctly + for h in self.hidden_dims: + if h < input_dims: + raise ValueError( + "Hidden dimension must not be less than input dimension." + ) + + # TODO: Check that the permutation is valid for the input dimension! + # Implement ispermutation() that sorts permutation and checks whether it + # has all integers from 0, 1, ..., self.input_dims - 1 + self.register_buffer("permutation", permutation) + + # Create masks + hidden_dims = self.hidden_dims + masks, mask_skip = create_mask( + input_dim=input_dims, + context_dim=0, # context_dims, + hidden_dims=hidden_dims, + permutation=permutation, + output_multiplier=self.output_multiplier, + ) + + # Create masked layers + layers = [ + MaskedLinear( + input_dims, # + context_dims, + hidden_dims[0], + masks[0], + ), + self.nonlinearity(), + ] + for i in range(1, len(hidden_dims)): + layers.extend( + [ + MaskedLinear(hidden_dims[i - 1], hidden_dims[i], masks[i]), + self.nonlinearity(), + ] + ) + layers.append( + MaskedLinear( + hidden_dims[-1], + input_dims * self.output_multiplier, + masks[-1], + ) + ) + + if self.skip_connections: + layers.append( + MaskedLinear( + input_dims, # + context_dims, + input_dims * self.output_multiplier, + mask_skip, + bias=False, + ) + ) + + self.layers = nn.ModuleList(layers) + + def _forward( + self, + x: Optional[torch.Tensor] = None, + context: Optional[torch.Tensor] = None, + ) -> Sequence[torch.Tensor]: + assert x is not None + + # Flatten x + batch_shape = x.shape[: len(x.shape) - len(self.input_shape)] + if len(batch_shape) > 0: + x = x.reshape(batch_shape + (-1,)) + + if context is not None: + # TODO: Fix the following! + h = torch.cat([context.expand((x.shape[0], -1)), x], dim=-1) + else: + h = x + + for idx in range(len(self.layers) // 2): + h = self.layers[2 * idx + 1](self.layers[2 * idx](h)) + h = self.layers[-1](h) + + # TODO: Get skip_layers working again! + # if self.skip_layer is not None: + # h = h + self.skip_layer(x) + + # Shape the output + # h ~ (batch_dims * input_dims, total_params_per_dim) + h = h.reshape(-1, self.output_multiplier) + + # result ~ (batch_dims * input_dims, params_per_dim[0]), ... + result = h.split(list(self.param_dims), dim=-1) + + # results ~ (batch_shape, param_shapes[0]), ... + result = tuple( + h_slice.view(batch_shape + p_shape) + for h_slice, p_shape in zip(result, list(self.param_shapes)) + ) + return result diff --git a/flowtorch/parameters/tensor.py b/flowtorch/parameters/tensor.py new file mode 100644 index 00000000..4d5ea3f2 --- /dev/null +++ b/flowtorch/parameters/tensor.py @@ -0,0 +1,28 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +from typing import Optional, Sequence + +import torch +import torch.nn as nn +from flowtorch.parameters.base import Parameters + + +class Tensor(Parameters): + def __init__( + self, + param_shapes: Sequence[torch.Size], + input_shape: torch.Size, + context_shape: Optional[torch.Size] = None, + ) -> None: + super().__init__(param_shapes, input_shape, context_shape) + + # TODO: Initialization strategies and constraints! + self.params = nn.ParameterList( + [nn.Parameter(torch.randn(shape) * 0.001) for shape in param_shapes] + ) + + def _forward( + self, x: Optional[torch.Tensor] = None, context: Optional[torch.Tensor] = None + ) -> Sequence[torch.Tensor]: + return list(self.params) diff --git a/flowtorch/utils.py b/flowtorch/utils.py new file mode 100644 index 00000000..9d85a302 --- /dev/null +++ b/flowtorch/utils.py @@ -0,0 +1,104 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import importlib +import inspect +import os +import pkgutil +from functools import partial +from typing import Sequence, Tuple, Callable, Optional, Any + +import flowtorch +from flowtorch.bijectors.base import Bijector +from flowtorch.parameters.base import Parameters +from torch.distributions import Distribution + + +copyright_header = """Copyright (c) Facebook, Inc. and its affiliates. \ +All Rights Reserved +SPDX-License-Identifier: MIT""" + + +def classname(cls: type) -> str: + return ".".join([cls.__module__, cls.__name__]) + + +def issubclass_byname(cls: type, test_cls: type) -> bool: + """ + Test whether a class is a subclass of another by class names, in contrast + to the built-in issubclass that does it by instance. + """ + return classname(test_cls) in [classname(c) for c in cls.__mro__] + + +def isderivedclass(cls: type, base_cls: type) -> bool: + # NOTE issubclass won't always do what we want here if base_cls is imported + # inside the module of cls. I.e. issubclass returns False if cls inherits + # from a base_cls with a different instance. + return inspect.isclass(cls) and issubclass_byname(cls, base_cls) + + +def list_bijectors() -> Sequence[Tuple[str, Bijector]]: + ans = _walk_packages("bijectors", partial(isderivedclass, base_cls=Bijector)) + ans = [a for a in ans if ".ops." not in a[1].__module__] + return list({classname(cls[1]): cls for cls in ans}.values()) + + +def list_parameters() -> Sequence[Tuple[str, Parameters]]: + ans = _walk_packages("parameters", partial(isderivedclass, base_cls=Parameters)) + return list({classname(cls[1]): cls for cls in ans}.values()) + + +def list_distributions() -> Sequence[Tuple[str, Parameters]]: + ans = _walk_packages( + "distributions", partial(isderivedclass, base_cls=Distribution) + ) + return list({classname(cls[1]): cls for cls in ans}.values()) + + +def _walk_packages( + modname: str, filter: Optional[Callable[[Any], bool]] +) -> Sequence[Tuple[str, Any]]: + classes = [] + + # NOTE: I use path of flowtorch rather than e.g. flowtorch.bijectors + # to avoid circular imports + path = [os.path.join(flowtorch.__path__[0], modname)] # type: ignore + + # The followings line uncovered a bug that hasn't been fixed in mypy: + # https://github.com/python/mypy/issues/1422 + for importer, this_modname, _ in pkgutil.walk_packages( + path=path, # type: ignore # mypy issue #1422 + prefix=f"{flowtorch.__name__}.{modname}.", + onerror=lambda x: None, + ): + # Conditions required for mypy + if importer is not None: + if isinstance(importer, importlib.abc.MetaPathFinder): + finder = importer.find_module(this_modname, None) + elif isinstance(importer, importlib.abc.PathEntryFinder): + finder = importer.find_module(this_modname) + else: + finder = None + + if finder is not None: + module = finder.load_module(this_modname) + + else: + raise Exception("Finder is none") + + if module is not None: + this_classes = inspect.getmembers(module, filter) + classes.extend(this_classes) + + del module + del finder + + else: + raise Exception("Module is none") + + return classes + + +class InterfaceError(Exception): + pass diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 00000000..39c467c0 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,2 @@ +[tool.usort] +first_party_detection=false diff --git a/scripts/copyright_headers.py b/scripts/copyright_headers.py new file mode 100644 index 00000000..1a7964a8 --- /dev/null +++ b/scripts/copyright_headers.py @@ -0,0 +1,166 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import argparse +import os +import sys +from enum import Enum + +from flowtorch.utils import copyright_header + +lines_header = ["# " + ln + "\n" for ln in copyright_header.splitlines()] + + +class ReadState(Enum): + EMPTY = 0 + COMMENT = 1 + TRIPLE_QUOTES = 2 + + +def get_header(filename): + state = ReadState.EMPTY + header = [] + with open(filename, "r") as f: + for line_idx, line in enumerate(f.readlines()): + line = line.strip() + # Finite state machine to read "header" of Python source + # TODO: Can I write this much more compactly with regular expressions? + if state is ReadState.EMPTY: + if len(line) and line[0] == "#": + state = ReadState.COMMENT + header.append(line[1:].strip()) + continue + elif len(line) >= 3 and line[:3] == '"""': + state = ReadState.TRIPLE_QUOTES + header.append(line[3:].strip()) + continue + else: + # If the file doesn't begin with a comment we consider the + # header to be empty + return "\n".join(header).strip(), line_idx, state + + elif state is ReadState.COMMENT: + if len(line) and line[0] == "#": + header.append(line[1:].strip()) + continue + else: + return "\n".join(header).strip(), line_idx, state + + elif state is ReadState.TRIPLE_QUOTES: + if len(line) >= 3 and '"""' in line: + char_idx = line.find('"""') + header.append(line[:char_idx].strip()) + return "\n".join(header).strip(), line_idx, state + else: + header.append(line.strip()) + continue + + else: + raise RuntimeError("Invalid read state!") + + # Return error if triple quotes don't terminate + if state is ReadState.TRIPLE_QUOTES: + raise RuntimeError(f"Unterminated multi-line string in {f}") + + # If we get to here then the file is all header + return "\n".join(header).strip(), line_idx + 1, state + + +def walk_source(paths): + # Find all Python source files that are not Git ignored + source_files = set() + for path in paths: + for root, _, files in os.walk(path): + for name in files: + full_name = os.path.join(root, name) + if name.endswith(".py") and os.system( + f"git check-ignore -q {full_name}" + ): + source_files.add(full_name) + + return sorted(source_files) + + +def print_results(count_changed, args): + # Print results + if count_changed == 0 and args.check: + print(f"{count_changed} files would be left unchanged.") + elif count_changed == len(source_files) and args.check: + print(f"{count_changed} files would be changed.") + elif args.check: + print( + f"""{count_changed} files would be changed and {len(source_files) / + - count_changed} files would be unchanged.""" + ) + elif count_changed: + print(f"{count_changed} files fixed.") + + +if __name__ == "__main__": + # Parse command line arguments + # Example usage: python scripts/copyright_headers.py --check flowtorch tests scripts + parser = argparse.ArgumentParser( + description="Checks and adds the Facebook Incubator copyright header" + ) + parser.add_argument( + "-c", + "--check", + action="store_true", + help="just checks files and does not change any", + ) + parser.add_argument( + "-v", "--verbose", action="store_true", help="prints extra information on files" + ) + parser.add_argument( + "paths", nargs="+", help="paths to search for Python source files" + ) + args = parser.parse_args() + + source_files = walk_source(args.paths) + + # Loop over source files and get the "header" + count_changed = 0 + for name in source_files: + header, line_idx, state = get_header(name) + + # Replace if it's not equal, starts with empty space, or is not a comment + if ( + header != copyright_header + or line_idx != 2 + or not state == ReadState.COMMENT + ): + count_changed += 1 + if args.verbose: + print(name) + + if not args.check: + # Read the file + with open(name, "r") as f: + lines = f.readlines() + + # Replace the header + # TODO: Debug the following! + if state == ReadState.TRIPLE_QUOTES: + after_quotes = lines[line_idx][ + (lines[line_idx].find('"""') + 3) : + ].lstrip() + if after_quotes == "": + lines = lines[line_idx + 1 :] + elif after_quotes.startswith(";"): + lines = [after_quotes[1:].lstrip()] + lines[line_idx + 1 :] + else: + raise RuntimeError( + "Statements must be separated by newlines or semicolons" + ) + else: + lines = lines[line_idx:] + + lines = lines_header + lines + filestring = "".join(lines) + + # Save back to disk + with open(name, "w") as f: + f.write(filestring) + + print_results(count_changed, args) + sys.exit(count_changed) diff --git a/scripts/generate_api_docs.py b/scripts/generate_api_docs.py new file mode 100644 index 00000000..e43fe033 --- /dev/null +++ b/scripts/generate_api_docs.py @@ -0,0 +1,80 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +""" +Generates MDX (Markdown + JSX, see https://mdxjs.com/) files and sidebar +information for the Docusaurus v2 website from the library components' +docstrings. + +We have chosen to take this approach to integrate our API documentation +with Docusaurus because there is no pre-existing robust solution to use +Sphinx output with Docusaurus. + +This script will be run by the "documentation" GitHub workflow on pushes +and pull requests to the main branch. It will function corrrectly from +any working directory. + +""" + +import errno +import os + +import flowtorch +from flowtorch.docs import ( + documentable_modules, + generate_markdown, + module_hierarchy, + name_entity_mapping, +) + +if __name__ == "__main__": + # Create website/docs/api if doesn't exist + try: + os.makedirs(os.path.join(flowtorch.__path__[0], "../website/docs/api")) + except OSError as e: + if e.errno != errno.EEXIST: + raise + + # Build sidebar JSON based on module hierarchy and save to 'website/api.sidebar.js' + all_sidebar_items = [] + + documentable_module_names = {m.__name__: v for m, v in documentable_modules.items()} + + def module_sidebar(mod_name, items): + return f"{{\n type: 'category',\n label: '{mod_name}',\n \ +collapsed: {'false' if mod_name in module_hierarchy.keys() else 'true'},\ + items: [{', '.join(items)}],\n}}" + + def dfs(dict): + sidebar_items = [] + for key, val in dict.items(): + items = ( + [f'"api/{key}"'] + + [f'"api/{key}.{item[0]}"' for item in documentable_module_names[key]] + if len(documentable_module_names[key]) > 0 + else [] + ) + + if val != {}: + items.extend(dfs(val)) + + sidebar_items.append(module_sidebar(key, items)) + + return sidebar_items + + # Convert class hierarchy into API sidebar + with open( + os.path.join(flowtorch.__path__[0], "../website/api.sidebar.js"), "w" + ) as file: + print("module.exports = [\n'api/overview',", file=file) + print(",".join(dfs(module_hierarchy)), file=file) + print("];", file=file) + + # Generate markdown files for documentable entities + name_entity_mapping = name_entity_mapping.copy() + name_entity_mapping[""] = None + for name, entity in name_entity_mapping.items(): + filename, markdown = generate_markdown(name, entity) + + with open(os.path.join(flowtorch.__path__[0], filename), "w") as file: + print(markdown, file=file) diff --git a/scripts/generate_imports.py b/scripts/generate_imports.py new file mode 100644 index 00000000..d160f0a9 --- /dev/null +++ b/scripts/generate_imports.py @@ -0,0 +1,199 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +""" +Generates imports files for bijectors, distributions, and parameters. +This script assumes that you have used `setup.py develop`. + +""" + +import errno +import io +import os + +import black +import flowtorch +import torch +from flowtorch.utils import ( + classname, + copyright_header, + list_distributions, + list_bijectors, + list_parameters, +) + +copyright_header = "".join(["# " + ln + "\n" for ln in copyright_header.splitlines()]) + +autogen_msg = """\"\"\" +Warning: This file was generated by flowtorch/scripts/generate_imports.py +Do not modify or delete! + +\"\"\" +""" + +bijectors_imports = """import inspect +from typing import cast, List, Tuple + +import torch""" + +bijectors_code = """def isbijector(cls: type) -> bool: + # A class must inherit from flowtorch.Bijector to be considered a valid bijector + return issubclass(cls, Bijector) + + +def standard_bijector(cls: type) -> bool: + # "Standard bijectors" are the ones we can perform standard automated tests upon + return ( + inspect.isclass(cls) + and isbijector(cls) + and cls.__name__ not in [clx for clx, _ in meta_bijectors] + ) + +# Determine invertible bijectors +invertible_bijectors = [] +for bij_name, cls in standard_bijectors: + # TODO: Use factored out version of the following + # Define plan for flow + event_dim = max(cls.domain.event_dim, 1) # type: ignore + event_shape = event_dim * [4] + # base_dist = dist.Normal(torch.zeros(event_shape), torch.ones(event_shape)) + bij = cls(shape=torch.Size(event_shape)) + + try: + y = torch.randn(*bij.forward_shape(event_shape)) + bij.inverse(y) + except NotImplementedError: + pass + else: + invertible_bijectors.append((bij_name, cls)) + + +__all__ = ["standard_bijectors", "meta_bijectors", "invertible_bijectors"] + [ + cls + for cls, _ in cast(List[Tuple[str, Bijector]], meta_bijectors) + + cast(List[Tuple[str, Bijector]], standard_bijectors) +]""" + +mode = black.FileMode() +fast = False + + +def generate_imports_plain(filename, classes): + with io.StringIO() as file: + # Sort classes by qualified name + classes = sorted(classes, key=lambda tup: classname(tup[1])) + + print(copyright_header, file=file) + print(autogen_msg, file=file) + for s, cls in classes: + print(f"from {cls.__module__} import {s}", file=file) + + print("", file=file) + + print("__all__ = [", file=file) + all_list = ",\n\t".join([f'"{s}"' for s, _ in classes]) + print("\t", end="", file=file) + print(all_list, file=file) + print("]", end="", file=file) + + contents = file.getvalue() + + with open(filename, "w") as real_file: + print( + black.format_file_contents(contents, fast=fast, mode=mode), + file=real_file, + end="", + ) + + +def generate_imports_bijectors(filename): + bij = list_bijectors() + meta_bijectors = [] + standard_bijectors = [] + + # Standard bijectors can be initialized with a shape, whereas + # meta bijectors will throw a TypeError or require additional + # keyword arguments (e.g., bij.Compose) + # TODO: Refactor this into flowtorch.utils.ismetabijector + for b in bij: + try: + cls = b[1] + x = cls(shape=torch.Size([2] * cls.domain.event_dim)) + except TypeError: + meta_bijectors.append(b) + else: + if isinstance(x, flowtorch.Lazy): + meta_bijectors.append(b) + else: + standard_bijectors.append(b) + + with io.StringIO() as file: + # Sort classes by qualified name + classes = standard_bijectors + meta_bijectors + classes = sorted(classes, key=lambda tup: classname(tup[1])) + + # Copyright header and warning message + print(copyright_header, file=file) + print(autogen_msg, file=file) + + # Non-FlowTorch imports + print(bijectors_imports, file=file) + + # FlowTorch imports + for s, cls in classes: + print(f"from {cls.__module__} import {s}", file=file) + print("", file=file) + + # Create lists of bijectors for each type + meta_str = ",\n ".join([f'("{b[0]}", {b[0]})' for b in meta_bijectors]) + standard_str = ",\n ".join( + [f'("{b[0]}", {b[0]})' for b in standard_bijectors] + ) + + print( + f"""standard_bijectors = [ + {standard_str} +] +""", + file=file, + ) + + print( + f"""meta_bijectors = [ + {meta_str} +] +""", + file=file, + ) + + # Rest of code + print(bijectors_code, file=file) + + contents = file.getvalue() + + with open(filename, "w") as real_file: + print( + black.format_file_contents(contents, fast=fast, mode=mode), + file=real_file, + end="", + ) + + +if __name__ == "__main__": + # Create module folders if they don't exist + try: + for m in ["distributions", "bijectors", "parameters"]: + os.makedirs(os.path.join(flowtorch.__path__[0], m)) + except OSError as e: + if e.errno != errno.EEXIST: + raise + + bijectors_init = os.path.join(flowtorch.__path__[0], "bijectors/__init__.py") + distributions_init = os.path.join( + flowtorch.__path__[0], "distributions/__init__.py" + ) + parameters_init = os.path.join(flowtorch.__path__[0], "parameters/__init__.py") + + generate_imports_bijectors(bijectors_init) + generate_imports_plain(distributions_init, list_distributions()) + generate_imports_plain(parameters_init, list_parameters()) diff --git a/setup.cfg b/setup.cfg new file mode 100644 index 00000000..03542267 --- /dev/null +++ b/setup.cfg @@ -0,0 +1,9 @@ +[flake8] +max-line-length = 80 +max-complexity = 12 +ignore = E501 +select = C,E,F,W,B,B9 +extend-ignore = E203, W503 + +[metadata] +license_files = LICENSE.txt diff --git a/setup.py b/setup.py new file mode 100644 index 00000000..e955b270 --- /dev/null +++ b/setup.py @@ -0,0 +1,87 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import os +import sys + +from setuptools import find_packages, setup + +REQUIRED_MAJOR = 3 +REQUIRED_MINOR = 7 + + +TEST_REQUIRES = ["numpy", "pytest", "pytest-cov", "scipy"] +DEV_REQUIRES = TEST_REQUIRES + [ + "black", + "flake8", + "flake8-bugbear", + "mypy", + "usort", +] + + +# Check for python version +if sys.version_info < (REQUIRED_MAJOR, REQUIRED_MINOR): + error = ( + "Your version of python ({major}.{minor}) is too old. You need " + "python >= {required_major}.{required_minor}." + ).format( + major=sys.version_info.major, + minor=sys.version_info.minor, + required_minor=REQUIRED_MINOR, + required_major=REQUIRED_MAJOR, + ) + sys.exit(error) + + +# read in README.md as the long description +with open("README.md", "r") as fh: + long_description = fh.read() + +setup( + name="flowtorch", + description="Normalizing Flows for PyTorch", + author="FlowTorch Development Team", + author_email="info@stefanwebb.me", + license="MIT", + url="https://flowtorch.ai/users", + project_urls={ + "Documentation": "https://flowtorch.ai/users", + "Source": "https://www.github.com/facebookincubator/flowtorch", + }, + keywords=[ + "Deep Learning", + "Bayesian Inference", + "Statistical Modeling", + "Variational Inference", + "PyTorch", + ], + classifiers=[ + "Development Status :: 3 - Alpha", + "Programming Language :: Python :: 3 :: Only", + "License :: OSI Approved :: MIT License", + "Topic :: Scientific/Engineering", + "Intended Audience :: Science/Research", + "Intended Audience :: Developers", + ], + long_description=long_description, + long_description_content_type="text/markdown", + python_requires=">={}.{}".format(REQUIRED_MAJOR, REQUIRED_MINOR), + install_requires=[ + "torch>=1.8.1", + ], + setup_requires=["setuptools_scm"], + use_scm_version={ + "root": ".", + "relative_to": __file__, + "write_to": os.path.join("flowtorch", "version.py"), + }, + packages=find_packages( + include=["flowtorch", "flowtorch.*"], + exclude=["debug", "tests", "website", "examples", "scripts"], + ), + extras_require={ + "dev": DEV_REQUIRES, + "test": TEST_REQUIRES, + }, +) diff --git a/tests/conftest.py b/tests/conftest.py new file mode 100644 index 00000000..5fe5d8a4 --- /dev/null +++ b/tests/conftest.py @@ -0,0 +1,16 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +import random + +import numpy as np +import pytest +import torch + + +@pytest.fixture(scope="function", autouse=True) +def set_seeds_before_every_test(): + torch.manual_seed(42) + np.random.seed(42) + random.seed(42) + + yield # yield control to the test to run diff --git a/tests/test_bijector.py b/tests/test_bijector.py new file mode 100644 index 00000000..dad4eefd --- /dev/null +++ b/tests/test_bijector.py @@ -0,0 +1,130 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +import flowtorch.bijectors as bijectors +import numpy as np +import pytest +import torch +import torch.distributions as dist +import torch.optim +from flowtorch.distributions import Flow + +""" +def test_bijector_constructor(): + param_fn = flowtorch.params.DenseAutoregressive() + b = flowtorch.bijectors.AffineAutoregressive(param_fn=param_fn) + assert b is not None +""" + + +@pytest.fixture(params=[bij_name for _, bij_name in bijectors.standard_bijectors]) +def flow(request): + bij = request.param + event_dim = max(bij.domain.event_dim, 1) + event_shape = event_dim * [3] + base_dist = dist.Independent( + dist.Normal(torch.zeros(event_shape), torch.ones(event_shape)), event_dim + ) + + flow = Flow(base_dist, bij) + return flow + + +def test_jacobian(flow, epsilon=1e-2): + # Instantiate transformed distribution and parameters + bij = flow.bijector + params = bij.params + + # Calculate auto-diff Jacobian + x = torch.randn(*flow.event_shape) + x = torch.distributions.transform_to(bij.domain)(x) + y = bij.forward(x) + if bij.domain.event_dim == 1: + analytic_ldt = bij.log_abs_det_jacobian(x, y).data + else: + analytic_ldt = bij.log_abs_det_jacobian(x, y).sum(-1).data + + # Calculate numerical Jacobian + # TODO: Better way to get all indices of array/tensor? + jacobian = torch.zeros(flow.event_shape * 2) + idxs = np.nonzero(np.ones(flow.event_shape)) + + # Have to permute elements for MADE + count_vars = len(idxs[0]) + if hasattr(params, "permutation"): + inv_permutation = np.zeros(count_vars, dtype=int) + inv_permutation[params.permutation] = np.arange(count_vars) + + # TODO: Vectorize numerical calculation of Jacobian with PyTorch + # TODO: Break this out into flowtorch.numerical.derivatives.jacobian + for var_idx in range(count_vars): + idx = [dim_idx[var_idx] for dim_idx in idxs] + epsilon_vector = torch.zeros(flow.event_shape) + epsilon_vector[(*idx,)] = epsilon + # TODO: Use scipy.misc.derivative or another library's function? + delta = ( + bij.forward(x + 0.5 * epsilon_vector) + - bij.forward(x - 0.5 * epsilon_vector) + ) / epsilon + + for var_jdx in range(count_vars): + jdx = [dim_jdx[var_jdx] for dim_jdx in idxs] + + # Have to account for permutation potentially introduced by MADE network + # TODO: Make this more general with structure abstraction + if hasattr(params, "permutation"): + jacobian[(inv_permutation[idx[0]], inv_permutation[jdx[0]])] = float( + delta[(Ellipsis, *jdx)].data.sum() + ) + else: + jacobian[(*idx, *jdx)] = float(delta[(Ellipsis, *jdx)].data.sum()) + + # For autoregressive flow, Jacobian is sum of diagonal, otherwise need full + # determinate + if hasattr(params, "permutation"): + numeric_ldt = torch.sum(torch.log(torch.diag(jacobian))) + else: + numeric_ldt = torch.log(torch.abs(jacobian.det())) + + ldt_discrepancy = (analytic_ldt - numeric_ldt).abs() + assert ldt_discrepancy < epsilon + + # Test that lower triangular with non-zero diagonal for autoregressive flows + if hasattr(params, "permutation"): + + def nonzero(x): + return torch.sign(torch.abs(x)) + + diag_sum = torch.sum(torch.diag(nonzero(jacobian))) + lower_sum = torch.sum(torch.tril(nonzero(jacobian), diagonal=-1)) + assert diag_sum == float(count_vars) + assert lower_sum == float(0.0) + + +def test_inverse(flow, epsilon=1e-5): + bij = flow.bijector + base_dist = flow.base_dist + + # Test g^{-1}(g(x)) = x + x_true = base_dist.sample(torch.Size([10])) + x_true = torch.distributions.transform_to(bij.domain)(x_true) + + y = bij.forward(x_true) + x_calculated = bij.inverse(y) + assert (x_true - x_calculated).abs().max().item() < epsilon + + # Test that Jacobian after inverse op is same as after forward + J_1 = bij.log_abs_det_jacobian(x_true, y) + J_2 = bij.log_abs_det_jacobian(x_calculated, y) + assert (J_1 - J_2).abs().max().item() < epsilon + + +""" +# TODO +def _test_shape(self, base_shape, transform): + pass + + +# TODO: This tests whether can take autodiff gradient without exception +def _test_autodiff(self, input_dim, transform, inverse=False): + pass +""" diff --git a/tests/test_compose.py b/tests/test_compose.py new file mode 100644 index 00000000..af2a6873 --- /dev/null +++ b/tests/test_compose.py @@ -0,0 +1,41 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import flowtorch.bijectors as bijs +import flowtorch.distributions as dist +import flowtorch.parameters as params +import torch +import torch.distributions +import torch.optim + + +def test_compose(): + transforms = bijs.Compose( + bijectors=[ + bijs.AffineAutoregressive( + params.DenseAutoregressive(), + ), + bijs.AffineAutoregressive( + params.DenseAutoregressive(), + ), + bijs.AffineAutoregressive( + params.DenseAutoregressive(), + ), + ] + ) + + event_shape = (5,) + base_dist = torch.distributions.Independent( + torch.distributions.Normal( + loc=torch.zeros(event_shape), scale=torch.ones(event_shape) + ), + len(event_shape), + ) + flow = dist.Flow(base_dist, transforms) + + optimizer = torch.optim.Adam(flow.parameters()) + assert optimizer.param_groups[0]["params"][0].grad is None + flow.log_prob(torch.randn((100,) + event_shape)).sum().backward() + assert optimizer.param_groups[0]["params"][0].grad.abs().sum().item() > 1e-3 + optimizer.zero_grad() + assert optimizer.param_groups[0]["params"][0].grad.abs().sum().item() < 1e-3 diff --git a/tests/test_distribution.py b/tests/test_distribution.py new file mode 100644 index 00000000..542af70f --- /dev/null +++ b/tests/test_distribution.py @@ -0,0 +1,117 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + +import flowtorch.bijectors as bijs +import flowtorch.distributions as dist +import flowtorch.parameters as params +import scipy.stats +import torch +import torch.distributions +import torch.optim + + +def test_tdist_standalone(): + input_dim = 3 + + def make_tdist(): + # train a flow here + base_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(input_dim), torch.ones(input_dim)), 1 + ) + bijector = bijs.AffineAutoregressive() + tdist = dist.Flow(base_dist, bijector) + return tdist + + tdist = make_tdist() + tdist.log_prob(torch.randn(input_dim)) # should run without error + assert True + + +def test_neals_funnel_vi(): + torch.manual_seed(42) + nf = dist.NealsFunnel() + bijector = bijs.AffineAutoregressive(params=params.DenseAutoregressive()) + + base_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), 1 + ) + flow = dist.Flow(base_dist, bijector) + bijector = flow.bijector + + opt = torch.optim.Adam(flow.parameters(), lr=2e-3) + num_elbo_mc_samples = 200 + for _ in range(100): + z0 = flow.base_dist.rsample(sample_shape=(num_elbo_mc_samples,)) + zk = bijector._forward(z0) + ldj = bijector._log_abs_det_jacobian(z0, zk) + + neg_elbo = -nf.log_prob(zk).sum() + neg_elbo += flow.base_dist.log_prob(z0).sum() - ldj.sum() + neg_elbo /= num_elbo_mc_samples + + if not torch.isnan(neg_elbo): + neg_elbo.backward() + opt.step() + opt.zero_grad() + + nf_samples = dist.NealsFunnel().sample((20,)).squeeze().numpy() + vi_samples = flow.sample((20,)).detach().numpy() + + assert scipy.stats.ks_2samp(nf_samples[:, 0], vi_samples[:, 0]).pvalue >= 0.05 + assert scipy.stats.ks_2samp(nf_samples[:, 1], vi_samples[:, 1]).pvalue >= 0.05 + + +""" +def test_conditional_2gmm(): + context_size = 2 + + flow = bijs.Compose( + bijectors=[ + bijs.AffineAutoregressive(context_size=context_size) + for _ in range(2) + ], + context_size=context_size, + ).inv() + + base_dist = dist.Normal(torch.zeros(2), torch.ones(2)) + new_cond_dist = flow(base_dist) + flow = new_cond_dist.bijector + + target_dist_0 = dist.Independent( + dist.Normal(torch.zeros(2) + 5, torch.ones(2) * 0.5), 1 + ) + target_dist_1 = dist.Independent( + dist.Normal(torch.zeros(2) - 5, torch.ones(2) * 0.5), 1 + ) + + opt = torch.optim.Adam(flow.params.parameters(), lr=1e-3) + + for idx in range(100): + opt.zero_grad() + + if idx % 2 == 0: + target_dist = target_dist_0 + context = torch.ones(context_size) + else: + target_dist = target_dist_1 + context = -1 * torch.ones(context_size) + + marginal = new_cond_dist.condition(context) + y = marginal.rsample((50,)) + loss = -target_dist.log_prob(y) + marginal.log_prob(y) + loss = loss.mean() + + if idx % 100 == 0: + print("epoch", idx, "loss", loss) + + loss.backward() + opt.step() + + assert ( + new_cond_dist.condition(torch.ones(context_size)).sample((1000,)).mean() - 5.0 + ).norm().item() < 1.0 + assert ( + new_cond_dist.condition(-1 * torch.ones(context_size)).sample((1000,)).mean() + + 5.0 + ).norm().item() < 1.0 +""" diff --git a/tests/test_imports.py b/tests/test_imports.py new file mode 100644 index 00000000..d807829b --- /dev/null +++ b/tests/test_imports.py @@ -0,0 +1,66 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT +import inspect + +import flowtorch +import flowtorch.bijectors +import flowtorch.distributions +import flowtorch.parameters +import flowtorch.utils + + +def test_parameters_imports(): + tst_imports( + "Parameters", + [cls for cls, _ in flowtorch.utils.list_parameters()], + [ + c + for c in flowtorch.parameters.__all__ + if inspect.isclass(flowtorch.parameters.__dict__[c]) + ], + ) + + +def test_bijector_imports(): + tst_imports( + "Bijector", + [cls for cls, _ in flowtorch.utils.list_bijectors()], + [ + c + for c in flowtorch.bijectors.__all__ + if inspect.isclass(flowtorch.bijectors.__dict__[c]) + ], + ) + + +def test_distribution_imports(): + tst_imports( + "Distribution", + [cls for cls, _ in flowtorch.utils.list_distributions()], + [ + c + for c in flowtorch.distributions.__all__ + if inspect.isclass(flowtorch.distributions.__dict__[c]) + ], + ) + + +def tst_imports(cls_name, detected, imported): + unimported = set(detected).difference(set(imported)) + undetected = set(imported).difference(set(detected)) + + error_msg = [] + if len(unimported): + error_msg.append( + f'The following {cls_name} classes are declared but not imported: \ +{", ".join(unimported)}' + ) + + if len(undetected): + error_msg.append( + f'The following {cls_name} classes are imported but not detected: \ +{", ".join(undetected)}' + ) + + if len(error_msg): + raise ImportError("\n".join(error_msg)) diff --git a/tests/test_interface.py b/tests/test_interface.py new file mode 100644 index 00000000..e5706d4b --- /dev/null +++ b/tests/test_interface.py @@ -0,0 +1,31 @@ +# Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved +# SPDX-License-Identifier: MIT + + +from flowtorch.docs import sorted_entity_names +from flowtorch.utils import InterfaceError + + +class TestInterface: + def test_documentable_case_insensitivity(self): + """ + Checks whether there are any two entities that are indistinguishable by + case. E.g. "flowtorch.params" the module and "flowtorch.Params" the + class. For producing the API docs on Windows and Mac OS systems, it is + advisable to have entity names that are unique regardless of case. + + """ + equivalence_classes = {} + for n in sorted_entity_names: + equivalence_classes.setdefault(n.lower(), []).append(n) + erroneous_equivalences = [ + f'{{{", ".join(v)}}}' for v in equivalence_classes.values() if len(v) > 1 + ] + + if len(erroneous_equivalences): + error_string = "\t\n".join(erroneous_equivalences) + raise InterfaceError( + f"""Documentable entities must be unique irrespective of case. The \ +following equivalences were found: + {error_string}""" + ) diff --git a/website/.gitignore b/website/.gitignore new file mode 100644 index 00000000..8a11ee68 --- /dev/null +++ b/website/.gitignore @@ -0,0 +1,22 @@ +# Dependencies +/node_modules + +# Production +/build + +# Generated files +.docusaurus +.cache-loader +/docs/api +api.sidebar.js + +# Misc +.DS_Store +.env.local +.env.development.local +.env.test.local +.env.production.local + +npm-debug.log* +yarn-debug.log* +yarn-error.log* diff --git a/website/README.md b/website/README.md new file mode 100644 index 00000000..ed310c91 --- /dev/null +++ b/website/README.md @@ -0,0 +1,41 @@ + + +This website is built using [Docusaurus 2](https://v2.docusaurus.io/), a modern static website generator, and hosted using GitHub pages at [https://flowtorch.ai](https://flowtorch.ai). The source for the website is located in [main/website](https://github.com/facebookincubator/flowtorch/tree/main/website) and it is hosted from the root directory of the [website](https://github.com/facebookincubator/flowtorch/tree/website) branch. + +## Preparation +1. Install [Node.js](https://nodejs.org/). +2. Install [Yarn](https://yarnpkg.com/): +```console +npm install --global yarn +``` +4. Navigate to [main/website](https://github.com/facebookincubator/flowtorch/tree/main/website) and install the dependencies: +```console +cd website +yarn install +``` + +## Local Development + +```console +yarn start +``` + +This command starts a local development server and open up a browser window. Most changes are reflected live without having to restart the server. + +## Build + +```console +yarn build +``` + +This command generates static content into the `website/build` directory, which is deployed by copying into the [gh-pages](https://github.com/facebookincubator/flowtorch/tree/gh-pages) branch. + +## Deployment + +Core developers can deploy the website as follows: + +```console +GIT_USER= USE_SSH=true yarn deploy +``` + +Activity logs for all past deployments to GitHub pages can be viewed [here](https://github.com/facebookincubator/flowtorch/deployments/activity_log?environment=github-pages). diff --git a/website/babel.config.js b/website/babel.config.js new file mode 100644 index 00000000..e00595da --- /dev/null +++ b/website/babel.config.js @@ -0,0 +1,3 @@ +module.exports = { + presets: [require.resolve('@docusaurus/core/lib/babel/preset')], +}; diff --git a/website/docs/dev/about.mdx b/website/docs/dev/about.mdx new file mode 100644 index 00000000..0fec8555 --- /dev/null +++ b/website/docs/dev/about.mdx @@ -0,0 +1,40 @@ +--- +id: about +title: About the Team +sidebar_label: About the Team +--- +:::info +*This could be you!* See [here](/dev) and [here](/dev/overview) for how to make an independent contribution to [FlowTorch](https://flowtorch.ai). We will consider adding new members to the core team for those who are interested and have made previous contributions. +::: + +## Core Team +The Core Developers team comprises [Stefan Webb](https://stefanwebb.me) (Team Leader), [Feynman Liang](https://feynmanliang.com/about/), and [Fritz Obermeyer](http://fritzo.org/). + +export const FlexContainer = ({children}) => ( +
+ {children} +
+); + + +
+
+
+ + +
+
+
+ + +
+
+
+ +## Contributors +:::info +Independent contributors will be recognized here when such contributions begin to flow (no pun intended)! +::: diff --git a/website/docs/dev/bibliography.mdx b/website/docs/dev/bibliography.mdx new file mode 100644 index 00000000..a547004a --- /dev/null +++ b/website/docs/dev/bibliography.mdx @@ -0,0 +1,89 @@ +--- +id: bibliography +title: Bibliography +sidebar_label: Bibliography +--- +:::info + +If you know of a paper or library that ought to be listed in this bibliography please let us know [in the forum](https://github.com/facebookincubator/flowtorch/discussions) or by [starting a pull request](https://github.com/facebookincubator/flowtorch/pulls). + +::: + +Here is a collation of materials related to the research, engineering, and teaching of normalizing flows. A corresponding BibTex file can be found [here](https://github.com/facebookincubator/flowtorch/blob/new_interface/website/static/assets/normalizing-flows.bib). + +## Surveys +
+[bond2021deep] Bond-Taylor, S., Leach, A., Long, Y., and Willcocks, C.G. Deep Generative Modelling: A Comparative Review of VAEs, GANs, Normalizing Flows, Energy-Based and Autoregressive Models. arXiv preprint arXiv:2103.04922, 2021. +
+ +
+[kobyzev2020normalizing] Kobyzev, I., Prince, S., and Brubaker, M. Normalizing flows: An introduction and review of current methods. IEEE Transactions on Pattern Analysis and Machine Intelligence, 2020. +
+ +
+[papamakarios2019normalizing] Papamakarios, G., Nalisnick, E., Rezende, D., Mohamed, S., and Lakshminarayanan, B. Normalizing flows for probabilistic modeling and inference. arXiv preprint arXiv:1912.02762, 2019. +
+ +## Methodology +
+[dinh2014nice] Dinh, L., Krueger, D., and Bengio, Y. NICE: Non-linear Independent Components Estimation. Workshop contribution at the International Conference on Learning Representations (ICLR), 2015. +
+ +
+[dinh2016density] Dinh, L., Sohl-Dickstein, J., and Bengio, S. Density estimation using real NVP. Conference paper at the International Conference on Learning Representations (ICLR), 2017. +
+ +
+[durkan2019neural] Durkan, C., Bekasov, A., Murray, I., and Papamakarios, G. Neural spline flows. 33rd Conference on Neural Information Processing Systems (NeurIPS), 2019. +
+ +
+[germain2015made] Germain, M., Gregor, K., Murray, I., and Larochelle, H. MADE: Masked autoencoder for distribution estimation . International Conference on Machine Learning (ICML), 2015. +
+ +
+[kingma2016improving] Kingma, D.P., Salimans, T., Jozefowicz, R., Chen, X., Sutskever, I., and Welling, M. Improving variational inference with inverse autoregressive flow . 29th Conference on Neural Information Processing Systems (NeurIPS), 2016. +
+ +
+[papamakarios2017masked] Papamakarios, G., and Pavlakou, T., and Murray, I. Masked autoregressive flow for density estimation . 30th Conference on Neural Information Processing Systems (NeurIPS), 2017. +
+ +
+[rezende2015variational] Rezende, D., and Mohamed, S. Variational inference with normalizing flows . International Conference on Machine Learning (ICML), 2015. +
+ +## Applications +
+[jin2019unsupervised] Jin, L., and Doshi-Velez, F., and Miller, T., and Schwartz, L., and Schuler, W. Unsupervised learning of PCFGs with normalizing flow. 57th Annual Meeting of the Association for Computational Linguistics (ACL), 2019. +
+ +
+[kim2020wavenode] Kim, H., and Lee, H., and Kang, W. H., and Cheon, S. J., Choi, B. J., and Kim, N. S. WaveNODE: A Continuous Normalizing Flow for Speech Synthesis. 2nd workshop on Invertible Neural Networks, Normalizing Flows, and Explicit Likelihood Models (ICML 2020), 2020. +
+ +
+[yang2019pointflow] Yang, G., Huang, X., Hao, Z., Liu, M., Belongie, S., and Hariharan, B. Pointflow: 3d point cloud generation with continuous normalizing flows. IEEE/CVF International Conference on Computer Vision, 2019. +
+ +## Libraries +### PyTorch +
+[bingham2018pyro] Bingham, E., and Chen, J.P., Jankowiak, M., Obermeyer, F., Pradhan, N., Karaletsos, T., Singh, R., Szerlip, P., Horsfall, P., and Goodman, N.D. Pyro: Deep Universal Probabilistic Programming. Journal of Machine Learning Research (JMLR), 2018. +
+ +> The majority of [the bijections in Pyro](https://github.com/pyro-ppl/pyro/tree/dev/pyro/distributions/transforms) were written by the core developer, [Stefan Webb](https://stefanwebb.me), and [FlowTorch](https://flowtorch.ai) builds upon this code and the experience gained from it. + +
+[phan2019composable] Phan, D., Pradhan, N., and Jankowiak, M. Composable Effects for Flexible and Accelerated Probabilistic Programming in NumPyro. arXiv preprint arXiv:1912.11554, 2019. +
+ +## Other Related +### Probabilistic Graphical Models +
+[koller2009probabilistic] Koller, D. and Friedman, N. Probabilistic graphical models: principles and techniques. MIT Press, 2009. +
+ +
+[webb2017faithful] Webb, S., Golinski, A., Zinkov, R., Siddharth, N., Rainforth, T., Teh, Y.W., and Wood, F. Faithful inversion of generative models for effective amortized inference. 31th Conference on Neural Information Processing Systems (NeurIPS), 2018. +
diff --git a/website/docs/dev/bijector.mdx b/website/docs/dev/bijector.mdx new file mode 100644 index 00000000..39b76076 --- /dev/null +++ b/website/docs/dev/bijector.mdx @@ -0,0 +1,69 @@ +--- +id: bijector +title: Bijector Interface +sidebar_label: Bijector Interface +--- + +## The Interface +A class satisfying the "Bijector interface" contains the following elements. + +### Parent class +A bijector must inherit from [`flowtorch.bijectors.Bijector`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/bijectors/base.py). This class defines important methods that are common to all bijectors such as `.inv` for defining an equivalent bijector swapping the forward and inverse operations. In the future, this parent class will be responsible for implementing [caching](/users/caching). + +### Domain and Codomain +`self.domain` and `self.codomain` are values of type `torch.distributions.constraint` and specify the range of valid inputs and outputs that a bijector acts upon, as well as the dimensionality of both. FlowTorch does not validate the values of the inputs or outputs using this information - it is mainly intended to be useful for users as documentation. + +However, the `.event_dim` property of both `self.domain` and `self.codomain` is important as it specifies whether a bijector operates over scalars (`event_dim=0`), vectors (`event_dim=1`), matrices (`event_dim=2`), etc., and this determines the shapes of a transformed distribution using the bijector. + +`self.domain` and `self.codomain` are typically *class properties*, although they can be *instance properties* where that makes sense, for example when a bijector operates on a different sized input depending on parameters passed to `__init__`. + +### Other Metadata +Further metadata about a bijector is defined in these properties: +* `autoregressive`: a bijector operating on vectors is autoregressive if $x_i$ is independent of $x_j$ for all $j>i$. Note that the order of autoregression may not be the same order as the PyTorch tensor since the bijector or its conditioning network may apply a permutation. We can generalize this in a straightforward way for bijectors operating on matrices, tensors, and higher-dimensional objects. This property is used by the testing framework. +* `near_identity_initialization`: whether a bijector is initialized to an "almost-identity" operation. In this context, a bijector is defined as being "almost-identity" if $y=f(x)$ does not diverge too much from a standard (multivariate) normal distribution when $x$ has a standard normal distribution. +* `volume_preserving`: a bijector is volume preserving, also known as *homomorphic*, if the volume of $\{f(x)\mid x \in A\}$ is the same as the volume of $A$ for all $A\subseteq\text{domain}(f)$. This is true of many bijections used in normalizing flows (examples to follow when we've moved across all the bijections from Pyro). + +Again, this metadata can be represented by *class properties* and *instance properties* depending on the context. For instance, a bijector may not be volume preserving by default and have a special volume preserving version that is enabled by a flag passed to `__init__`. + +:::info +Further metadata fields may be defined in the future. However, developers are not permitted to define their own without adding a default value to [`flowtorch.bijectors.Bijector`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/bijectors/base.py). +::: + +### Class Methods +Class methods define initialization of the bijector, the forward ($y=f(x)$) and inverse ($x=f^{-1}(y)$) operators, the log absolute determinant Jacobian ($\log(|\det(dy/dx)|)$), and methods that define the shapes of $f$, $f^{-1}$, and its parameters. All methods are optional save for `._forward` and `._inverse` - the defaults for the others are the same as those of the identity operation (see [`flowtorch.bijectors.Bijector`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/bijectors/base.py)). + +#### `.__init__(self, param_fn: flowtorch.params.Params, *, **kwargs)` +This optional method initializes a bijector, taking a `flowtorch.params.Params` object and an arbitrary number of keyword arguments specific to the bijector. It must call the parent initializer, passing the value of `param_fn`, that is, `super().__init__(param_fn=param_fn)`. Typically, the initializer is used to store parameters of the bijector and sometimes modify its metadata. + +*`__init__` must have sensible default values for all its arguments so that one can instantiate a bijector with, for example, `b = MyBijector()`.* This design allows both easy creation and testing of bijectors. + +#### `._forward(self, x: torch.Tensor, params: Optional[flowtorch.params.ParamsModule])` +#### `._inverse(self, y: torch.Tensor, params: Optional[flowtorch.params.ParamsModule])` +These methods defines the forward, $y=f_\theta(x)$, and inverse, $x=f^{-1}_\theta(y)$, operations of a bijector, respectively. + +By convention, when a bijector has either a forward or inverse operation that does not have an explicit formula or that is intractable, the forward operation will be defined by the tractable operation and the inverse will be left undefined (and you can obtain the inverted bijector with `.inv`). [Caching](/users/caching) is useful in these circumstances to apply the intractable operation to inputs that have previously been used with the bijector. + +#### `._log_abs_det_jacobian(self, x: torch.Tensor, y: torch.Tensor, params: Optional[flowtorch.params.ParamsModule])` +This methods defines the log absolute determinant Jacobian, $\log(|\det(dy/dx)|)$, that determines how the functional form of the bijector warps an infinitesimally small volume of space. Since it may be easier to calculate this using one of either $x$ or $y$, both are given as arguments - it is up to the caller to ensure that $y=f(x)$. + +If this method is undefined, it will default to a tensor of zeros, that is, the quantity in question for a volume preserving bijector. + +#### `.forward_shape(self, event_shape)` +#### `.inverse_shape(self, event_shape)` +`.forward_shape` defines the event shape of $y=f(x)$ given the event shape of $x$. Similarly, `.inverse_shape` defines the event shape of $x=f^{-1}(y)$ given the event shape of $y$. These methods provide additional flexibility for defining bijectors, although in most cases will be left undefined in the derived class so that the default of the identity function is used. One example of where these methods differ from the identity is in [flowtorch.Reshape](/users/composing). + +:::info +It must be the case that `len(event_shape) == self.domain.event_dim` for `.forward_shape` and `len(event_shape) == self.codomain.event_dim` for `.inverse_shape`. Likewise, the outputs of these two methods must match the corresponding `event_dim`. +::: + +#### `.params_shape(self, event_shape)` +This method defines the shapes of the parameters for a given event shape. It returns a tuple of shapes of type `torch.Size()`. For instance, if there are two separate scalar parameters for each event dimension, we could implement this method as: + +```python + def params_shape(self, event_shape:torch.Size) -> Tuple[torch.Size]: + return (event_shape, event_shape) +``` + +:::note +Yet to be decided: what do we want as the convention when a `Bijector` does not use any parameters? Should it return `None` or a single `torch.Size()`? +::: diff --git a/website/docs/dev/contributing.md b/website/docs/dev/contributing.md new file mode 100644 index 00000000..d39e8b55 --- /dev/null +++ b/website/docs/dev/contributing.md @@ -0,0 +1,28 @@ +--- +id: contributing +title: Help Wanted! +sidebar_label: Help Wanted! +slug: /dev +--- +:::info +Please contact us in [the forum](https://github.com/facebookincubator/flowtorch/discussions) if you are interested in becoming an independent contributor and tag your discussion with ":bulb: Ideas" - the process is outlined [here](/dev/overview). +::: + +## Call for Contributions +We are looking for independent collaborators to: +* add new bijectors (i.e., normalizing flow transforms) and parameters (i.e., conditioning networks); +* [discover and fix bugs](https://github.com/facebookincubator/flowtorch/issues/new/choose); and, +* write tutorials on [applications of Normalizing Flow methodology](/dev/bibliography#applications). + +The [Core Developers](/dev/about) are able to help smooth [the process of making a contribution](/dev/overview). + + +## Why Contribute? +Why would you freely give up your labour and contribute to an open-source project? Firstly, contributing to an open-source project is excellent for one's professional development as a software engineer. Take the example of the author, who started contributing to an open-source project during his PhD and developed important DevOp skills. It was greatly responsible for his scoring an internship in Industry and kickstarting his career. Contributors will be recognised both [here](/dev/about#contributors) and [here](https://github.com/facebookincubator/flowtorch/graphs/contributors). + +Another reason is that [it is inherently satisfying to make](https://en.wikipedia.org/wiki/Maker_culture). By contributing to [FlowTorch](https://flowtorch.ai) you will be creating useful components that will have a concrete impact. Finally, a main motivation behind [FlowTorch](https://flowtorch.ai) is to advance scientific knowledge around representing probability distributions and their applications - by contributing to [FlowTorch](https://flowtorch.ai) you are contributing to the advancement of science! + +See [here](https://opensource.guide/how-to-contribute/) for a more detailed essay on the philosophy of open-source. + +## Code of Conduct +As a contributor, you agree to abide by the [Contributor Covenant Code of Conduct](https://github.com/facebookincubator/flowtorch/blob/main/CODE_OF_CONDUCT.md). In a nutshell, the code says [Be Excellent to Each Other!](https://www.youtube.com/watch?v=rph_1DODXDU) Please report any suspected violations to the Core Developers. diff --git a/website/docs/dev/docs.md b/website/docs/dev/docs.md new file mode 100644 index 00000000..cb3b7b19 --- /dev/null +++ b/website/docs/dev/docs.md @@ -0,0 +1,15 @@ +--- +id: docs +title: Docs +sidebar_label: Docs +--- +:::info +The easiest way to write a docstring that adheres to FlowTorch conventions is to copy one from a pre-existing class and adapt it to your case. +::: + +## Docstrings +It is crucial to add an informative [docstring](https://www.python.org/dev/peps/pep-0257/#id15) to new `bij.Bijector` and `params.Parameters` classes. This docstring should detail what the class does, its functional form, the meaning of *all* input arguments and returned values, and references to any relevant literature. + +References should link to their citation in the [bibliography](/dev/bibliography), for example, with [https://flowtorch.ai/dev/bibliography#dinh2014nice](https://flowtorch.ai/dev/bibliography#dinh2014nice). This means you may need to add additional citations to the website with your `Bijector` or `Parameters` implementation. + +Be sure to test the formatting of the docstring in the docs using the workflow detailed [here](/dev/ops). diff --git a/website/docs/dev/ops.md b/website/docs/dev/ops.md new file mode 100644 index 00000000..68871918 --- /dev/null +++ b/website/docs/dev/ops.md @@ -0,0 +1,91 @@ +--- +id: ops +title: Continuous Integration +sidebar_label: Continuous Integration +--- +:::info +Please do not feel intimidated by the thought of having to make your code pass the CI tests! The core developer team is happy to work closely with contributors to integrate their code and merge PRs. +::: + +FlowTorch uses [GitHub Actions](https://docs.github.com/en/actions) to run code quality tests on pushes or pull requests to the `main` branch, a process known as [continuous integration](https://en.wikipedia.org/wiki/Continuous_integration) (CI). The tests are run for Python versions 3.7, 3.8, and 3.9, and must be successful for a PR to be merged into `main`. All workflow runs can be viewed [here](https://github.com/facebookincubator/flowtorch/actions), or else viewed from the link at the bottom of the [PR](https://github.com/facebookincubator/flowtorch/pulls) in question. + + +## Workflow Steps +The definition of the steps performed in the build workflow is found [here](https://github.com/facebookincubator/flowtorch/blob/main/.github/workflows/python-package.yml) and is as follows: + +1. The version of Python (3.7, 3.8, or 3.9) is installed along with the developer dependencies of FlowTorch; +```bash +python -m pip install --upgrade pip +python -m pip install flake8 black usort pytest mypy +pip install numpy +pip install --pre torch torchvision torchaudio +pip install -e .[dev] +``` + +2. Each Python source is checked for containing the mandatory copyright header by a [custom script](https://github.com/facebookincubator/flowtorch/blob/main/scripts/copyright_headers.py): +```bash +python scripts/copyright_headers.py --check flowtorch tests scripts examples +``` + +3. The formatting of the Python code in the [library](https://github.com/facebookincubator/flowtorch/tree/main/flowtorch) and [tests](https://github.com/facebookincubator/flowtorch/tree/main/tests) is checked to ensure it follows a standard using [`black`](https://black.readthedocs.io/en/stable/); +```bash +black --check flowtorch tests +``` +4. Likewise, the order and formatting of Python `import` statements in the same folders is checked to ensure it follows a standard using [`usort`](https://usort.readthedocs.io/en/stable/); +```bash +usort check flowtorch tests +``` +5. A [static code analysis](https://en.wikipedia.org/wiki/Static_program_analysis), or rather, linting, is performed by [`flake8`](https://flake8.pycqa.org/en/latest/) to find potential bugs; +```bash +flake8 . tests --count --show-source --statistics +``` +6. FlowTorch makes use of type hints, which we consider mandatory for all contributed code, and static types are checked with [`mypy`](https://github.com/python/mypy); +```bash +mypy --disallow-untyped-defs flowtorch +``` +7. Unit tests: + +pytest + XML coverage report +```bash +pytest --cov=tests --cov-report=xml -W ignore::DeprecationWarning tests/ +``` + +8. The coverage report is uploaded to [Codecov](https://about.codecov.io/) with a [GitHub Action](https://github.com/codecov/codecov-action). This allows us to analyze the results and produce the percentage of code covered badge. + +If any step fails, the workflow fails and you will not be able to merge the PR into `main`. + +## Successful Commits +To ensure your PR passes, you should perform these steps *before pushing your local commits to the remote branch*. + +### Run Tests +Run the tests first so that you can do the code formatting just once as the final step: +```bash +pytest tests -W ignore::DeprecationWarning +``` +Fix any failing tests until the above command succeeds. + +### Check Types +Check that there are no errors with the type hints: +```bash +mypy --disallow-untyped-defs flowtorch +``` +I find this is one of the most difficult steps to make pass - if you require assistance, comment on your PR, tagging the core developers. + +### Formatting and Linting +Having ensured the tests and docs are correct, run the following commands to standardize your code's formatting: +```bash +black flowtorch tests +usort format flowtorch tests +``` +Now, run these commands in check mode to ensure there are no errors: +```bash +black --check flowtorch tests +usort check flowtorch tests +``` +It is possible you may need to fix some errors by hand. + +Finally, run the linter and fix any resulting errors: +```bash +flake8 flowtorch tests +``` +At this point, you are ready to commit your changes and push to the remote branch - you're a star! :star: From there, your PR will be reviewed by the core developers and after any modifications are made, merged to the `main` branch. diff --git a/website/docs/dev/overview.md b/website/docs/dev/overview.md new file mode 100644 index 00000000..f18c6c35 --- /dev/null +++ b/website/docs/dev/overview.md @@ -0,0 +1,40 @@ +--- +id: overview +title: Overview +sidebar_label: Overview +--- +:::info +If you are having trouble getting the CI tests to pass, you may create a PR, regardless, in order to get a review and help from the core developers. +::: + +:::info +It is preferable to write smaller, incremental PRs as opposed to larger, monolithic ones. Aim to modify only a few files and add less than 500 lines of code. +::: + +[FlowTorch](https://flowtorch.ai) is designed with easy extensibility in mind. In this section, we detail the interfaces for normalizing flow bijections and conditioning networks, as well as the software practices that must be followed. First, however, let us explain the process for making a contribution to [FlowTorch](https://flowtorch.ai). + +## How to Make a Contribution +### Ideation +New features begin with a discussion between users, independent contributors (that's you!), and the core development team. If you would like to see a new feature or are interested in contributing it yourself, please start a new thread on the forum, tagging it with "new feature." + +### Development +After this discussion has taken place and the details of new feature has been decided upon, the next step is to fork the [flowtorch repo](https://github.com/facebookincubator/flowtorch) using the "Fork" button in the upper right corner. + +Next, clone your forked repository locally and create a feature branch: + +```bash +git clone https://github.com//flowtorch.git +cd flowtorch +git checkout -b +``` + +Create your new feature. Ensure you have [added a docstring](/dev/docs) to your new class. + +Follow the steps [here](/dev/ops#successful-commits) to ensure that your code is formatted correctly, passes type checks, unit tests, the docs build, and so on. + +Assuming it passes these tests, commit the changes to your local repo and push to your remote fork. + +### Review +Finally, create a [pull request](https://docs.github.com/en/github/collaborating-with-issues-and-pull-requests/about-pull-requests) (PR) to merge your forked feature branch into the [main branch](https://github.com/facebookincubator/flowtorch). Give an informative name to your PR and include in the description of your PR the details of which features are added. Ensure your feature branch contains the latest commits from the main branch so as to avoid merge conflicts. + +The core developers will review your PR and most likely suggest changes to the code. After edits have been made, pushing to the feature branch of your forked remote will update the existing PR that you have opened. After all edits have been made and the tests pass, the core developers will merge your code into the main branch! diff --git a/website/docs/dev/params.mdx b/website/docs/dev/params.mdx new file mode 100644 index 00000000..2da2989f --- /dev/null +++ b/website/docs/dev/params.mdx @@ -0,0 +1,45 @@ +--- +id: params +title: Params Interface +sidebar_label: Params Interface +--- + +## The Interface +A class satisfying the "Params interface" contains the following elements. + +### Parent class +A bijector must inherit from [`flowtorch.params.Params`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/params/base.py). This class defines important methods that are common to all parameter objects, such as `.__call__` for instantiating a [`flowtorch.params.ParamsModule`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/params/base.py) given shape information. + +### Metadata +The following property is the only one currently used: +* `autoregressive`: a parameter object operating on vectors is autoregressive if the output $x_i$ is not a function of any $x_j$ with $j>i$ (with a straightforward generalization to higher-dimensional objects). This property is used by the testing framework. + +:::info +In the near future, the `autoregressive` property is likely to be removed, and a [structured representation](/users/structure) API used instead. +::: + +:::info +Further metadata fields may be defined in the future. However, developers are not permitted to define their own without adding a default value to [`flowtorch.params.Params`](https://github.com/facebookincubator/flowtorch/blob/main/flowtorch/params/base.py). +::: + +### Class Methods +Class methods define the initization of the lazy parameter object, how to instantiate the parameter object - that is, create any parameter vectors and neural networks, given shape information - and how to calculate the value of the parameters given a value from the distribution and possibly a context variable that is conditioned upon. + +#### `.__init__(self, *, **kwargs)` +This optional method initializes a lazy parameter object, taking an arbitrary number of keyword arguments specific to the class. It must call the parent initializer as `super().__init__()`. Typically, the initializer is used to store settings and sometimes modify metadata. + +*`__init__` must have sensible default values for all its arguments so that one can instantiate a params object with, for example, `p = Params()`.* This design allows both easy creation and testing of params. + +#### `._build(self, input_shape: torch.Size, param_shapes: Sequence[torch.Size], context_dims: int) -> Tuple[nn.ModuleList, Dict[str, Any]]` +This method builds any necessary `nn.Parameters` or `nn.Module`s as well as buffer objects, given the shape of an input, `input_shape`, the output shapes, `param_shapes`, and the number of dimensions, `context_dims`, of an optional context variable. It returns a tuple consisting of an `nn.ModuleList` for the learnable parameters and an optional `Dict[str, Any]` mapping strings to buffer objects. + +Buffer objects differ from learnable parameters in that they do not partake in gradient descent updates, but share with parameters that they are serialized when the object is saved and loaded to disk. Buffers are typically used to store tensors that are convenient to calculate and cache during the construction of the object, such as masking matrices. + +The `._build` method is called by `Params.__call__` during the process of instantiating a non-lazy `flowtorch.params.ParamsModule` using the lazy `flowtorch.params.Params` and specified shapes. `._build` should operate on any arbitrary input and parameter shapes. + +#### `._forward(self, x: torch.Tensor, context: torch.Tensor, modules: nn.ModuleList) -> Sequence[torch.Tensor]` +This method evaluates the parameters, $\theta=f(x;z,\{\alpha_i\})$, which in general are a function of the input, $x$, context variable, $z$, and a list of modules, $\{\alpha_i\}$. Note that this may not always be the case, for instance, when the parameters are `nn.Parameter` tensors that do not depend on $x$, or when the `Params` object is a placeholder for no parameters. + +:::note +Certain `Params` are incompatible with certain `Bijector`s. For example, an autoregressive bijector requires an autoregressive params. We are currently deciding on a solution to enforce/check this and will likely release with v2 of the library. +::: diff --git a/website/docs/dev/releases.md b/website/docs/dev/releases.md new file mode 100644 index 00000000..937d5ac9 --- /dev/null +++ b/website/docs/dev/releases.md @@ -0,0 +1,15 @@ +--- +id: releases +title: Releases +sidebar_label: Releases +--- + +A list of FlowTorch releases is to be found [here](https://github.com/facebookincubator/flowtorch/releases). In this section, we detail the process of making a release. + +## Versioning Scheme +The versioning scheme we use is a simple system with versions of the form *<major>.<minor>[.dev<build>]*. Some examples are: +* `0.5`; +* `1.4`; and, +* `0.0.dev1`. + +We use [`setuptools_scm`](https://github.com/pypa/setuptools_scm) to automatically handle versions, and it is able to bump the version for builds without `.dev`. A description of how [`setuptools_scm`](https://github.com/pypa/setuptools_scm) handles versioning can be found [here](https://github.com/pypa/setuptools_scm/#default-versioning-scheme). diff --git a/website/docs/dev/tests.md b/website/docs/dev/tests.md new file mode 100644 index 00000000..635e64ce --- /dev/null +++ b/website/docs/dev/tests.md @@ -0,0 +1,9 @@ +--- +id: tests +title: Tests +sidebar_label: Tests +--- + +All `bijector.Bijector` and `params.Parameters` classes are covered by unit tests that test that the interface is satisfied, correct shape information is produced, and in the case of bijectors, that the log determinate absolute Jacobian is correct, amongst other things. + +In general, you will not need to write new unit tests. When you implement a new component it will be detected by the library and included in existing tests. diff --git a/website/docs/users/bijectors.md b/website/docs/users/bijectors.md new file mode 100644 index 00000000..230c0f5c --- /dev/null +++ b/website/docs/users/bijectors.md @@ -0,0 +1,11 @@ +--- +id: bijectors +title: Bijectors +sidebar_label: Bijectors +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/caching.md b/website/docs/users/caching.md new file mode 100644 index 00000000..a8f7bd33 --- /dev/null +++ b/website/docs/users/caching.md @@ -0,0 +1,13 @@ +--- +id: caching +title: Caching +sidebar_label: Caching +--- + +:::caution + +This document is under construction! + +::: + +* Issue of cache being invalidated when you update the parameters! diff --git a/website/docs/users/composing.md b/website/docs/users/composing.md new file mode 100644 index 00000000..e0192972 --- /dev/null +++ b/website/docs/users/composing.md @@ -0,0 +1,17 @@ +--- +id: composing +title: Composing Bijectors +sidebar_label: Composing Bijectors +--- +:::caution + +This document is under construction! + +::: + +There are several ways to compose `Bijector`s to form new ones. + +* `flowtorch.Cat`: ? +* `flowtorch.Compose`: function composition +* `flowtorch.Reshape`: change the event shape of a bijector +* `flowtorch.Stack`: ? diff --git a/website/docs/users/conditional.mdx b/website/docs/users/conditional.mdx new file mode 100644 index 00000000..0cc20e48 --- /dev/null +++ b/website/docs/users/conditional.mdx @@ -0,0 +1,29 @@ +--- +id: conditional +title: Conditional Bijections +sidebar_label: Conditional Bijections +--- + +## Background + +In many cases, we wish to represent conditional rather than joint distributions. For instance, in performing variational inference, the variational family is a class of conditional distributions, + +$$ +\begin{aligned} +\{q_\psi(\mathbf{z}\mid\mathbf{x})\mid\theta\in\Theta\}, +\end{aligned} +$$ + +where $\mathbf{z}$ is the latent variable and $\mathbf{x}$ the observed one, that hopefully contains a member close to the true posterior of the model, $p(\mathbf{z}\mid\mathbf{x})$. In other cases, we may wish to learn to generate an object $\mathbf{x}$ conditioned on some context $\mathbf{c}$ using $p_\theta(\mathbf{x}\mid\mathbf{c})$ and observations $\{(\mathbf{x}_n,\mathbf{c}_n)\}^N_{n=1}$. For instance, $\mathbf{x}$ may be a spoken sentence and $\mathbf{c}$ a number of speech features. + +The theory of Normalizing Flows is easily generalized to conditional distributions. We denote the variable to condition on by $C=\mathbf{c}\in\mathbb{R}^M$. A simple multivariate source of noise, for example a standard i.i.d. normal distribution, $X\sim\mathcal{N}(\mathbf{0},I_{D\times D})$, is passed through a vector-valued bijection that also conditions on C, $g:\mathbb{R}^D\times\mathbb{R}^M\rightarrow\mathbb{R}^D$, to produce the more complex transformed variable $Y=g(X;C=\mathbf{c})$. In practice, this is usually accomplished by making the parameters for a known normalizing flow bijection $g$ the output of a hypernet neural network that inputs $\mathbf{c}$. + +Sampling of conditional transforms simply involves evaluating $Y=g(X; C=\mathbf{c})$. Conditioning the bijections on $\mathbf{c}$, the same formula holds for scoring as for the joint multivariate case. + +## Conditioning Transformed Distributions +:::caution +The examples in this section makes use of `Bijector` classes that are not yet available - they will be added in the `v0.3` release. The concepts are still relevant. +::: +[`Bijector`s have no notion of conditionality!] + +## Next Steps diff --git a/website/docs/users/conditioning.md b/website/docs/users/conditioning.md new file mode 100644 index 00000000..b7813bd0 --- /dev/null +++ b/website/docs/users/conditioning.md @@ -0,0 +1,11 @@ +--- +id: conditioning +title: Conditioning +sidebar_label: Conditioning +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/constraints.md b/website/docs/users/constraints.md new file mode 100644 index 00000000..7a4f1adc --- /dev/null +++ b/website/docs/users/constraints.md @@ -0,0 +1,11 @@ +--- +id: constraints +title: Constraints +sidebar_label: Constraints +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/gpu_support.md b/website/docs/users/gpu_support.md new file mode 100644 index 00000000..7c96e582 --- /dev/null +++ b/website/docs/users/gpu_support.md @@ -0,0 +1,9 @@ +--- +id: gpu_support +title: GPU Support +sidebar_label: GPU Support +--- + +:::info +FlowTorch bijectors, conditioning networks, and transformed distributions are likely to work on GPUs but have not yet been fully tested. Full GPU support is expected for our `v0.2` release and until then this page will serve as a placeholder. +::: diff --git a/website/docs/users/initialization.md b/website/docs/users/initialization.md new file mode 100644 index 00000000..39352147 --- /dev/null +++ b/website/docs/users/initialization.md @@ -0,0 +1,11 @@ +--- +id: initialization +title: Initialization +sidebar_label: Initialization +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/installation.md b/website/docs/users/installation.md new file mode 100644 index 00000000..c2f1952f --- /dev/null +++ b/website/docs/users/installation.md @@ -0,0 +1,35 @@ +--- +id: installation +title: Installation +sidebar_label: Installation +--- + +[FlowTorch](https://flowtorch.ai) can be installed as a package or directly from source. + +## Requirements + +Python 3.7 or later is required. Other requirements will be downloaded by `pip` according to [setup.py](https://github.com/facebookincubator/flowtorch/blob/main/setup.py). + +## Pre-release + +As [FlowTorch](https://flowtorch.ai) is currently under rapid development, we recommend installing the [latest commit](https://github.com/facebookincubator/flowtorch/commits/main) from GitHub: + + git clone https://github.com/facebookincubator/flowtorch.git + cd flowtorch + pip install -e . + +Updates can then be performed by navigating to the directory where you cloned [FlowTorch](https://flowtorch.ai) and running: + + git pull + +## Latest Release + +Alternatively, the [latest release](https://github.com/facebookincubator/flowtorch/releases) is installed from [PyPI](https://pypi.org/project/flowtorch/): + + pip install flowtorch + +## Developers + +[Additional libraries](https://github.com/facebookincubator/flowtorch/blob/main/setup.py#L14) required for development are installed by replacing the above `pip` command with: + + pip install -e .[dev] diff --git a/website/docs/users/intro.mdx b/website/docs/users/intro.mdx new file mode 100644 index 00000000..33b703e5 --- /dev/null +++ b/website/docs/users/intro.mdx @@ -0,0 +1,27 @@ +--- +id: introduction +title: Introduction +sidebar_label: Introduction +slug: /users +--- + +## What is a Normalizing Flow? +Normalizing Flows are a family a methods for representing and learning high-dimensional probability distributions. They have found [state-of-the-art applications](/dev/bibliography#applications) in modeling complex distributions over images, [speech](/dev/bibliography#kim2020wavenode), [syntactic structure](/dev/bibliography#jin2019unsupervised), and molecules, to name a few. *Simply put, a Normalizing Flow is a composition of learnable functions that inputs samples from a simple random distribution, typically Gaussian noise, and outputs samples from a more complex target distribution.* Here is an illustration ([taken with permission from here](https://github.com/janosh/awesome-normalizing-flows)): + +

+ +

+ +A simple source of noise, $z_0$, is passed through a number of invertible functions, $f_1,f_2,\ldots,f_k$ to produce a more complex random variable, $z_k$. The invertible functions are constructed in a clever way so that we can easily sample from $z_k$ and calculate its density function, $p_k(\cdot)$. The field of Normalizing Flows can be seen as a modern take on the [change of variables method for random distributions](https://en.wikipedia.org/wiki/Probability_density_function#Function_of_random_variables_and_change_of_variables_in_the_probability_density_function), where the transformations are *high-dimensional*, often employing *neural networks*, and are designed for *effective stochastic optimization*. + +We believe, although still a nascent field, that Normalizing Flows are a fundamental component of the modern Bayesian statistics and probabilistic computing toolkit, and we will likely see many more exciting applications in the near future. + +## What is FlowTorch? +[FlowTorch](https://flowtorch.ai) is a library that provides PyTorch components for constructing Normalizing Flows using the latest research in the field. It builds on an earlier sub-library of code from [Pyro](https://github.com/pyro-ppl/pyro/tree/dev/pyro/distributions/transforms) developed by the author since 2018. The main goals behind creating a new library for Normalizing Flows are to: +* define an elegant interface for Normalizing Flow methodology, building on our experience with Pyro, so that practitioners can easily utilize these methods and researchers can easily contribute their own implementations; +* develop robust unit tests and other code quality practices to guarantee production quality code; +* promote the methods in applied settings by fostering a community of Normalizing Flow practioners and linking them with researchers; +* accelerate research in Normalizing Flows by providing standard implementations, benchmarking, and a comprehensive literature survey. + +## Where to From Here? +We recommend reading the next two sections to [install FlowTorch](/users/installation) and [train your first Normalizing Flow](/users/start). For more theoretical background on Normalizing Flows and information about their applications, see the primer [here](/users/univariate) and the list of survey papers [here](/dev/bibliography#surveys). diff --git a/website/docs/users/methods.md b/website/docs/users/methods.md new file mode 100644 index 00000000..4e230c31 --- /dev/null +++ b/website/docs/users/methods.md @@ -0,0 +1,11 @@ +--- +id: methods +title: Table of Methods +sidebar_label: Table of Methods +--- + +:::info + +This page will contain a table of `Bijector` and `Params` classes in FlowTorch with information such as their functional form, domain, range, computation/memory complexity, and literature references. Until our `v0.2` release, this is a placeholder. + +::: diff --git a/website/docs/users/multivariate.mdx b/website/docs/users/multivariate.mdx new file mode 100644 index 00000000..e961d94f --- /dev/null +++ b/website/docs/users/multivariate.mdx @@ -0,0 +1,133 @@ +--- +id: multivariate +title: Multivariate Bijections +sidebar_label: Multivariate Bijections +--- + +## Background +The fundamental idea of normalizing flows also applies to multivariate random variables, and this is where its value is clearly seen - *representing complex high-dimensional distributions*. In this case, a simple multivariate source of noise, for example a standard i.i.d. normal distribution, $X\sim\mathcal{N}(\mathbf{0},I_{D\times D})$, is passed through a vector-valued bijection, $g:\mathbb{R}^D\rightarrow\mathbb{R}^D$, to produce the more complex transformed variable $Y=g(X)$. + +Sampling $Y$ is again trivial and involves evaluation of the forward pass of $g$. We can score $Y$ using the multivariate substitution rule of integral calculus, + +$$ +\begin{aligned} + \mathbb{E}_{p_X(\cdot)}\left[f(X)\right] &= \int_{\text{supp}(X)}f(\mathbf{x})p_X(\mathbf{x})d\mathbf{x}\\ + &= \int_{\text{supp}(Y)}f(g^{-1}(\mathbf{y}))p_X(g^{-1}(\mathbf{y}))\det\left|\frac{d\mathbf{x}}{d\mathbf{y}}\right|d\mathbf{y}\\ + &= \mathbb{E}_{p_Y(\cdot)}\left[f(g^{-1}(Y))\right], + \end{aligned} +$$ + +where $d\mathbf{x}/d\mathbf{y}$ denotes the Jacobian matrix of $g^{-1}(\mathbf{y})$. Equating the last two lines we get, + +$$ +\begin{aligned} + \log(p_Y(y)) &= \log(p_X(g^{-1}(y)))+\log\left(\det\left|\frac{d\mathbf{x}}{d\mathbf{y}}\right|\right)\\ + &= \log(p_X(g^{-1}(y)))-\log\left(\det\left|\frac{d\mathbf{y}}{d\mathbf{x}}\right|\right). +\end{aligned} +$$ + +Inituitively, this equation says that the density of $Y$ is equal to the density at the corresponding point in $X$ plus a term that corrects for the warp in volume around an infinitesimally small volume around $Y$ caused by the transformation. For instance, in $2$-dimensions, the geometric interpretation of the absolute value of the determinant of a Jacobian is that it represents the area of a parallelogram with edges defined by the columns of the Jacobian. In $n$-dimensions, the geometric interpretation of the absolute value of the determinant Jacobian is that is represents the hyper-volume of a parallelepiped with $n$ edges defined by the columns of the Jacobian (see a calculus reference such as \[7\] for more details). + +Similar to the univariate case, we can compose such bijective transformations to produce even more complex distributions. By an inductive argument, if we have $L$ transforms $g_{(0)}, g_{(1)},\ldots,g_{(L-1)}$, then the log-density of the transformed variable $Y=(g_{(0)}\circ g_{(1)}\circ\cdots\circ g_{(L-1)})(X)$ is + +$$ +\begin{aligned} + \log(p_Y(y)) &= \log\left(p_X\left(\left(g_{(L-1)}^{-1}\circ\cdots\circ g_{(0)}^{-1}\right)\left(y\right)\right)\right)+\sum^{L-1}_{l=0}\log\left(\left|\frac{dg^{-1}_{(l)}(y_{(l)})}{dy'}\right|\right), +\end{aligned} +$$ + +where we've defined $y_{(0)}=x$, $y_{(L-1)}=y$ for convenience of notation. + +The main challenge is in designing parametrizable multivariate bijections that have closed form expressions for both $g$ and $g^{-1}$, a tractable Jacobian whose calculation scales with $O(D)$ or $O(1)$ rather than $O(D^3)$, and can express a flexible class of functions. + +## Multivariate `Bijector`s +In this section, we show how to use `bij.SplineAutoregressive` to learn the bivariate toy distribution from our running example. Making a simple change we can represent bivariate distributions of the form, $p(x_1,x_2)=p(x_1)p(x_2|x_1)$: + +```python +dist_x = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), + 1 +) +bijector = bij.SplineAutoregressive() +dist_y = dist.Flow(dist_x, bijector) +``` + +The `bij.SplineAutoregressive` bijector extends `bij.Spline` so that the spline parameters are the output of an autoregressive neural network. See [[durkan2019neural]](/dev/bibliography#durkan2019neural) and [[germain2015made]](/dev/bibliography#germain2015made) for more details. + +Similarly to before, we train this distribution on the toy dataset and plot the results: + +```python +dataset = torch.tensor(X, dtype=torch.float) +optimizer = torch.optim.Adam(spline_transform.parameters(), lr=5e-3) +for step in range(steps): + optimizer.zero_grad() + loss = -dist_y.log_prob(dataset).mean() + loss.backward() + optimizer.step() + + if step % 500 == 0: + print('step: {}, loss: {}'.format(step, loss.item())) +``` + +``` +step: 0, loss: 8.446191787719727 +step: 500, loss: 2.0197808742523193 +step: 1000, loss: 1.794958472251892 +step: 1500, loss: 1.73616361618042 +step: 2000, loss: 1.7254879474639893 +step: 2500, loss: 1.691617488861084 +step: 3000, loss: 1.679549217224121 +step: 3500, loss: 1.6967085599899292 +step: 4000, loss: 1.6723777055740356 +step: 4500, loss: 1.6505967378616333 +step: 5000, loss: 1.8024061918258667 +``` + +```python +X_flow = dist_y.sample(torch.Size([1000,])).detach().numpy() +plt.title(r'Joint Distribution') +plt.xlabel(r'$x_1$') +plt.ylabel(r'$x_2$') +plt.scatter(X[:,0], X[:,1], label='data', alpha=0.5) +plt.scatter(X_flow[:,0], X_flow[:,1], color='firebrick', label='flow', alpha=0.5) +plt.legend() +plt.show() +``` + +

+ +

+ +```python +plt.subplot(1, 2, 1) +sns.distplot(X[:,0], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='data') +sns.distplot(X_flow[:,0], hist=False, kde=True, + bins=None, color='firebrick', + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='flow') +plt.title(r'$p(x_1)$') +plt.subplot(1, 2, 2) +sns.distplot(X[:,1], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='data') +sns.distplot(X_flow[:,1], hist=False, kde=True, + bins=None, color='firebrick', + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='flow') +plt.title(r'$p(x_2)$') +plt.show() +``` + +

+ +

+ +We see from the output that this normalizing flow has successfully learnt both the univariate marginals *and* the bivariate distribution. diff --git a/website/docs/users/parameters.md b/website/docs/users/parameters.md new file mode 100644 index 00000000..e3373624 --- /dev/null +++ b/website/docs/users/parameters.md @@ -0,0 +1,11 @@ +--- +id: parameters +title: Parameters +sidebar_label: Parameters +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/serialization.md b/website/docs/users/serialization.md new file mode 100644 index 00000000..04653b30 --- /dev/null +++ b/website/docs/users/serialization.md @@ -0,0 +1,9 @@ +--- +id: serialization +title: Serialization +sidebar_label: Serialization +--- + +:::info +Serialization of `flowtorch.distributions.TransformedDistribution` objects is likely to work but has not yet been tested. We expect this to be completed for our `v0.2` release and until then this page will serve as a placeholder. +::: diff --git a/website/docs/users/shapes.mdx b/website/docs/users/shapes.mdx new file mode 100644 index 00000000..3f611306 --- /dev/null +++ b/website/docs/users/shapes.mdx @@ -0,0 +1,106 @@ +--- +id: shapes +title: Shapes +sidebar_label: Shapes +--- + +One of the advantages of using FlowTorch is that we have carefully thought out how shape information is propagated from the base distribution through the sequence of bijective transforms. Before we explain how shapes are handled in FlowTorch, let us revisit the shape conventions shared across PyTorch and TensorFlow. + +## Shape Conventions +FlowTorch shares the shape conventions of PyTorch's [`torch.distributions.Distribution`](https://pytorch.org/docs/stable/distributions.html#distribution) and TensorFlow's [`tfp.distributions.Distribution`](https://www.tensorflow.org/probability/api_docs/python/tfp/distributions/Distribution) for representing random distributions. In these conventions, the shape of a tensor sampled from a random distribution is divided into three parts: the *sample shape*, the *batch shape*, and the *event shape*. + +As described in the [TensorFlow documentation](https://www.tensorflow.org/probability/examples/Understanding_TensorFlow_Distributions_Shapes#basics), + +* Event shape describes the shape of a single draw from the distribution, which may or may not be dependent across dimensions. +* Batch shape describes independent, not identically distributed draws, that is, a "batch" of distributions. +* Sample shape describes independent, identically distributed draws of batches from the distribution family. + +## Examples +This is best illustrated with some simple examples. Let's begin with a standard normal distribution: + +```python +import torch +import torch.distributions as dist +d = dist.Normal(loc=0, scale=1) +sample_shape = torch.Size([]) + +assert d.event_shape == torch.Size([]) +assert d.batch_shape == torch.Size([]) +assert d.sample(sample_shape).shape == torch.Size([]) +``` + +In this example, we have a single scalar normal distribution from which we draw a scalar sample. Since it is a scalar distribution, the `event_shape == torch.Size([])`. Since it is a single distribution, `batch_shape == torch.Size([])`. And we draw a scalar sample since `sample_shape == torch.Size([])`. + +Note that *the event shape and batch shape are properties of the distribution itself*, whereas the sample shape depends on the size argument passed to [`Distribution.sample`](https://pytorch.org/docs/stable/distributions.html#torch.distributions.distribution.Distribution.sample) or [`Distribution.rsample`](https://pytorch.org/docs/stable/distributions.html#torch.distributions.distribution.Distribution.rsample). Also, the shape of `d.sample(sample_shape)` is the concatenation of the `sample_shape`, `batch_shape`, and `event_shape`, in that order. + +Let's look at another example: + +```python +d = dist.Normal(loc=torch.zeros(1), scale=torch.ones(1)) +sample_shape = torch.Size([2]) + +assert d.event_shape == torch.Size([]) +assert d.batch_shape == torch.Size([1]) +assert d.sample(sample_shape).shape == torch.Size([2, 1]) +``` + +In this case, `event_shape = torch.Size([])` since we have a scalar distribution, but `batch_shape = torch.Size([1])` since we have tensor of parameters of that shape defining the distribution. Also, `sample_shape = torch.Size([2])` so that `d.sample(sample_shape).shape = torch.Size([2, 1])`. + +A further example: + +```python +d = dist.Normal(loc=torch.zeros(2, 5), scale=torch.ones(2, 5)) +sample_shape = torch.Size([3, 4]) + +assert d.event_shape == torch.Size([]) +assert d.batch_shape == torch.Size([2, 5]) +assert d.sample(sample_shape).shape == torch.Size([3, 4, 2, 5]) +``` + +We see that batch shapes, sample shapes (and event shapes) can have an arbitrary number of dimensions are are not restricted to being vectors. + +Is the event shape always `torch.Size([])`? This is not true for *multivariate* distributions, that is, distributions over vectors, matrices, and higher-order tensors that can have dependencies across their dimensions. For example: + +```python +d = dist.MultivariateNormal(loc=torch.zeros(2, 5), covariance_matrix=torch.eye(5)) +sample_shape = torch.Size([3, 4]) + +assert d.event_shape == torch.Size([5]) +assert d.batch_shape == torch.Size([2]) +assert d.sample(sample_shape).shape == torch.Size([3, 4, 2, 5]) +``` + +Note that the `covariance_matrix` tensor will be broadcast across `loc`. *Whereas the previous example defined a matrix batch of scalar normal distributions, this example defines a vector batch of multivariate normal distributions.* This is an important distinction! + +See [this page](https://ericmjl.github.io/blog/2019/5/29/reasoning-about-shapes-and-probability-distributions/) for further explanation on shape conventions. + +## Non-conditional Transformed Distributions +How do shapes work for transformed distributions that do not condition on a context variable, that is, distributions of the form $p_\theta(\mathbf{x})$? The sample shape depends strictly on the input to `.sample` or `.rsample` and so we restrict our attention to the batch and event shapes. + +Returning to the diagram on the [intro page](/users), suppose the base distribution is $p_0$, and the distribution after applying the the initial bijection, $f_1$, is $p_1$. Denote by $z_0$ a sample from the base distribution and $z_1=f_1(z_0)$. We make a few observations: + +Firstly, since $f_1$ is a bijection, $z_0$ must have the same number of dimensions as $z_1$. In our shape terminology, the sum of the event shape of the base distribution must be the same as the sum of the event shape of the transformed one. + +Secondly, the batch shape is preserved from the base distribution to transformed one. *By convention, we assume that a single bijection, $f_1$, is applied to a batch of base distributions, $\{p_{0,i}\}$, to produce a batch of the same shape of transformed distributions, $\{p_{1,i}\}$.* + +Thirdly, the event shape of the base distribution must be compatible with the domain of the bijection. For instance, if the base distribution has event shape `torch.Size([])` and is a scalar, it does not make sense to applied a bijection on matrices with, e.g., $\text{Dom}[f_1]\subseteq \mathbb{R}^{n\times m}$. + +Given a base distribution, `base`, and a non-conditional bijector `bijector`, the pseudo-code to calculate the batch and event shape of the transformed distribution, `flow`, looks like this: + +```python +# Input event shape must have at least as many dimensions as that which bijector operates over +assert len(base.event_shape) >= bijector.domain.event_dim + +flow.batch_shape = base.batch_shape +flow.event_shape = bijector.forward_shape(base.event_shape) + +# bijector.forward_shape and bijector.codomain.event_dim must be consistent +assert len(flow.event_shape) >= bijector.codomain.event_dim + +# bijectors preserve dimensions +assert sum(flow.event_shape) == sum(base.event_shape) +``` + +The `bijector` class defines the number of dimensions that it operates over in `bijector.domain.event_dim` and `bijector.codomain.event_dim`, and has a method `bijector.forward_shape` that specifies how the event shape of the input relates to that of the output. (In most cases, this will be the identity function.) + +This information is sufficient to construct the batch and event shapes of the transformed distribution from the base. For a Normalizing Flow that is the composition of multiple bijections, we apply this logic in succession, using the transformed distribution of the previous step as the base distribution of the next. \ No newline at end of file diff --git a/website/docs/users/start.mdx b/website/docs/users/start.mdx new file mode 100644 index 00000000..1f8180bf --- /dev/null +++ b/website/docs/users/start.mdx @@ -0,0 +1,136 @@ +--- +id: start +title: Your First Flow +sidebar_label: Your First Flow +--- + +## The Task +Let's begin training our first Normalizing Flow with a simple example! The target distribution that we intend to learn is, +$$ +\begin{aligned} + Y' &\sim \mathcal{N}\left(\mu=\begin{bmatrix} + 5 \\ + 5 +\end{bmatrix}, \Sigma=\begin{bmatrix} + 0.5 & 0 \\ + 0 & 0.5 +\end{bmatrix} \right) +\end{aligned}, +$$ +that is, a linear transformation of an standard multivariate normal distribution. The base distribution is, +$$ +\begin{aligned} + X &\sim \mathcal{N}\left(\mu=\begin{bmatrix} + 0 \\ + 0 +\end{bmatrix}, \Sigma=\begin{bmatrix} + 1 & 0 \\ + 0 & 1 +\end{bmatrix} \right) +\end{aligned}, +$$ +that is, standard normal noise (which is typical for Normalizing Flows). The task is to learn some bijection $g_\theta$ so that +$$ +\begin{aligned} + Y &\triangleq g_\theta(X) \\ + &\sim Y' +\end{aligned} +$$ +approximately holds. We will define our Normalizing Flow, $g_\theta$ by a single affine transformation, +$$ +\begin{aligned} + g_\theta(\mathbf{x}) &\triangleq \begin{bmatrix} + \mu_1 \\ + \mu_2(x_1) +\end{bmatrix} + \begin{bmatrix} + \sigma_1 \\ + \sigma_2(x_1) +\end{bmatrix}\otimes\begin{bmatrix} + x_1 \\ + x_2 +\end{bmatrix}. +\end{aligned} +$$ +In this notation, $\mathbf{x}=(x_1,x_2)^T$, $\otimes$ denotes element-wise multiplication, and the parameters are the scalars $\mu_1,\sigma_1$ and the parameters of the neural networks $\mu_2(\cdot)$ and $\sigma_2(\cdot)$. (Think of the NNs as very simple shallow feedforward nets in this example.) This is an example of [Inverse Autoregressive Flow](/dev/bibliography#kingma2016improving). + +There are several metrics we could use to train $Y$ to be close in distribution to $Y'$. First, let us denote the target distribution of $Y'$ by $p(\cdot)$ and the learnable distribution of the normalizing flow, $Y$, as $q_\theta(\cdot)$ (in the following sections, we will explain how to calculate $q_\theta$ from $g_\theta$). Let's use the forward [KL-divergence](https://en.wikipedia.org/wiki/Kullback%E2%80%93Leibler_divergence), +$$ +\begin{aligned} +\text{KL}\{p\ ||\ q_\theta\} &\triangleq \mathbb{E}_{p(\cdot)}\left[\log\frac{p(Y')}{q_\theta(Y')}\right] \\ +&= -\mathbb{E}_{p(\cdot)}\left[\log q_\theta(Y')\right] + C, +\end{aligned} +$$ +where C is a constant that does not depend on $\theta$. In practice, we draw a finite sample, $\{y_1,\ldots,y_M\}$, from $p$ and optimize a [Monte Carlo estimate](https://en.wikipedia.org/wiki/Monte_Carlo_integration) of the KL-divergence with stochastic gradient descent so that the loss is, +$$ +\begin{aligned} + \mathcal{L}(\theta) &= -\frac{1}{M}\sum^M_{m=1}\log(q_\theta(y_m)) +\end{aligned} +$$ + +*So, to summarize, the task at hand is to learn how to transform standard bivariate normal noise into another bivariate normal distribution using an affine transformation, and we will do so by matching distributions with the KL-divergence metric.* + +## Implementation +First, we import the relevant libraries: +```python +import torch +import flowtorch.bijectors as bij +import flowtorch.distributions as dist +``` +The base and target distributions are defined using standard PyTorch: +```python +base_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), + 1 +) +target_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2)+5, torch.ones(2)*0.5), + 1 +) +``` +Note the use of [`torch.distributions.Independent`](https://pytorch.org/docs/stable/distributions.html#independent) so that our base and target distributions are *vector valued*. + +We can visualize samples from the base and target: +

+ +

+ +A Normalizing Flow is created in two steps. First, we create a "plan" for the flow as a `flowtorch.bijectors.Bijector` object, +```python +# Lazily instantiated flow +bijectors = bij.AffineAutoregressive() +``` +This plan is then made concrete by combining it with the base distributions, which provides the input shape, and constructing a `flowtorch.distributions.Flow` object, and extension of `torch.distributions.Distribution`: +```python +# Instantiate transformed distribution and parameters +flow = dist.Flow(base_dist, bijectors) +``` +At this point, we have an object, `flow`, for the distribution, $q_\theta(\cdot)$, that follows the standard PyTorch interface. Therefore, it can be trained with the following code, which will be familiar for readers who have used `torch.distributions` before: +```python +# Training loop +opt = torch.optim.Adam(flow.parameters(), lr=5e-3) +for idx in range(3001): + opt.zero_grad() + + # Minimize KL(p || q) + y = target_dist.sample((1000,)) + loss = -flow.log_prob(y).mean() + + if idx % 500 == 0: + print('epoch', idx, 'loss', loss) + + loss.backward() + opt.step() +``` +Note how we obtain the learnable parameters of the normalizing flow from the `flow` object, which is a `torch.nn.Module`. Visualizing samples after learning, we see that we have been successful in matching the target distribution: +

+ +

+Congratulations on training your first flow! + +## Discussion + +This simple example illustrates a few important points of FlowTorch's design: + +Firstly, `Bijector` objects are agnostic to their shape. A `Bijector` object specifies *how the shape is changed* by the forward and inverse operations, and then calculates the exact shapes when it obtains knowledge of the base distribution, when `flow = dist.Flow(base_dist, bijectors)` is run. Any neural networks or other parametrized functions, which also require this shape information, are not instantiated until the same moment. In this sense, a `Bijector` can be thought of as a lazy plan for creating a normalizing flow. The advantage of doing things this way is that the shape information can be "type checked" and does not need to be specified in multiple locations (ensuring these quantities are consistent). + +Secondly, all objects are designed to have sensible defaults. We do not need to define the conditioning network for `bijectors.AffineAutoregressive`, it will use a [MADE network](/dev/bibliography#germain2015made) with sensible hyperparameters and defer initialization until it later receives shape information. Thirdly, there is compatibility, in as far as is possible, with standard PyTorch interfaces such as `torch.distributions`. diff --git a/website/docs/users/structure.md b/website/docs/users/structure.md new file mode 100644 index 00000000..135c4ef4 --- /dev/null +++ b/website/docs/users/structure.md @@ -0,0 +1,56 @@ +--- +id: structure +title: Structured Representations +sidebar_label: Structured Representations +--- + +## Bayesian Networks +The *structure* of a distribution refers to the set of independence relationships that hold for the distribution. Suppose we have a distribution over variables, $\{x_1,x_2,\ldots,x_N\}$. At one extreme, the variables are fully *independent* and the distribution can be written as, +$$ +p(\mathbf{x}) = \prod^N_{n=1}p(x_n). +$$ +At the other extreme, the variables are fully *dependent* and the distribution can be written as, +$$ +p(\mathbf{x}) = \prod^N_{n=1}p(x_n\mid x_1,x_2\ldots,x_{n-1}). +$$ +In between these two extremes, a distribution will have a factorization with factors that condition on some but not all of the previous variables under a given ordering. + +The field of *Probabilistic Graphical Models* studies graphical representations that express these structural relationships within distributions, as well as inference algorithms that operate directly on the graphical structures. For instance, we say that the fully independent distribution factors over the following *directed acyclic graph* (DAG), also known as a *Bayesian network (BN) structure*, + +[insert graph svg] + +And the fully dependent distribution factors over the fully connected BN structure, + +[insert graph svg] + +Therefore, full independence corresponds to zero edges, full dependence corresponds to the maximum number of edges in a DAG (that is, $N\ \text{choose}\ 2$), and it seems reasonable that distributions between these two extremes factor according to some graph with an intermediate number of edges. For instance, a graph over $x_1,\ldots,x_7$ might factor according to this graph, + +[insert graph svg] + +For Bayesian networks, that is, *directed* graphical models (there are other formalisms for *undirected*, *bidirected*, and graphs with mixed directionality), the semantics of the graphical structure are that ... It can be shown that a graph factors according to a BN structure if and only if ... + +So in a sense, the BN structure and a distribution's factorization are equivalent, and both express the conditional independence relationships that hold in the distribution. + +## Faithfulness and Minimality +One important point to note is that the BN structure for which a distribution factors over may fail to express some of the independence relationships that hold in a distribution - it must not, however, express independence relationships that *do not* hold in the distribution. For instance, any distribution factors according to $\prod^N_{n=1}p(x_n\mid x_1,x_2\ldots,x_{n-1})$, by the chain rule of probability. So, the fully connected DAG is a valid BN structure for the fully *independent* distribution. + +* Lack of edges express conditional independent relationships that hold in a distribution, whereas the presence of edges is non-informative. + +* Definition of I-map, minimal I-Map, and faithful + +* Non-uniqueness of the minimal I-map. Also can have varying number of edges! + +## Structure of Normalizing Flows +The dependency structure of normalizing flows is not something that has been considered in the literature, save for a few papers (for example, ?). Typically, they input a fully independent . + +However, it can be advantageous to represent some structure in the distribution and use this as an inductive prior for learning. [Cite my work!] showed ... + +## Abstractions for Expressing Structure + +:::info + +Keeping this discussion in mind, we are developing an abstraction for expressing structure in a normalizing flow for the `v0.2` release. This abstraction is likely to belong to both `Params` and `Bijector`s, and analogously to the `.forward_shape` and `.backward_shape` methods, informs the `TransformedDistribution` class how the dependency structure is effected by each layer of the normalizing flow. + +There will likely be two methods exposed to the user on `TransformedDistribution`: `.factorization`, and `.topological_order`. The first, `.factorization` might return a dictionary from variable indices to the parents of that variable in a minimal I-map. Another possibility is for `.factorization` to input a variable indices and return the array of parent indices (in which case, perhaps it should be called `.parents` and perhaps there should be a `.children` too?). This may be better if calculating and returning the whole object is an expensive operation. The second, `.topological_order`, returns an array of indices in topological ordering, possibly only calculating this lazily the first time it is requested. + +::: diff --git a/website/docs/users/torchscript.md b/website/docs/users/torchscript.md new file mode 100644 index 00000000..6ea1dc03 --- /dev/null +++ b/website/docs/users/torchscript.md @@ -0,0 +1,9 @@ +--- +id: torchscript +title: TorchScript Support +sidebar_label: TorchScript +--- + +:::info +FlowTorch bijectors, conditioning networks, and transformed distributions are likely to work with [TorchScript](https://pytorch.org/docs/stable/jit.html) constructions but have not yet been fully tested. TorchScript support is expected for our `v0.2` release and until then this page will serve as a placeholder. +::: diff --git a/website/docs/users/transformed_distributions.md b/website/docs/users/transformed_distributions.md new file mode 100644 index 00000000..928d93ad --- /dev/null +++ b/website/docs/users/transformed_distributions.md @@ -0,0 +1,11 @@ +--- +id: transformed_distributions +title: Transformed Distributions +sidebar_label: Transformed Distributions +--- + +:::caution + +This document is under construction! + +::: diff --git a/website/docs/users/univariate.mdx b/website/docs/users/univariate.mdx new file mode 100644 index 00000000..378e35b7 --- /dev/null +++ b/website/docs/users/univariate.mdx @@ -0,0 +1,254 @@ +--- +id: univariate +title: Univariate Bijections +sidebar_label: Univariate Bijections +--- +## Background +[Normalizing Flows](/dev/bibliography#surveys) are a family of methods for constructing flexible distributions. As mentioned in [the introduction](/users), Normalizing Flows can be seen as a modern take on the [change of variables method for random distributions](https://en.wikipedia.org/wiki/Probability_density_function#Function_of_random_variables_and_change_of_variables_in_the_probability_density_function), and this is most apparent for univariate bijections. Thus, in this first section we restrict our attention to representing univariate distributions with bijections. + +The basic idea is that a simple source of noise, for example a variable with a standard normal distribution, $X\sim\mathcal{N}(0,1)$, is passed through a bijective (i.e. invertible) function, $g(\cdot)$ to produce a more complex transformed variable $Y=g(X)$. For such a random variable, we typically want to perform two operations: sampling and scoring. Sampling $Y$ is trivial. First, we sample $X=x$, then calculate $y=g(x)$. Scoring $Y$, or rather, evaluating the log-density $\log(p_Y(y))$, is more involved. How does the density of $Y$ relate to the density of $X$? We can use the substitution rule of integral calculus to answer this. Suppose we want to evaluate the expectation of some function of $X$. Then, + +$$ +\begin{aligned} +\mathbb{E}_{p_X(\cdot)}\left[f(X)\right] &= \int_{\text{supp}(X)}f(x)p_X(x)dx\\ + &= \int_{\text{supp}(Y)}f(g^{-1}(y))p_X(g^{-1}(y))\left|\frac{dx}{dy}\right|dy \\ + &= \mathbb{E}_{p_Y(\cdot)}\left[f(g^{-1}(Y))\right], +\end{aligned} +$$ + +where $\text{supp}(X)$ denotes the support of $X$, which in this case is $(-\infty,\infty)$. Crucially, we used the fact that $g$ is bijective to apply the substitution rule in going from the first to the second line. Equating the last two lines we get, + +$$ +\begin{aligned} + \log(p_Y(y)) &= \log(p_X(g^{-1}(y)))+\log\left(\left|\frac{dx}{dy}\right|\right)\\ + &= \log(p_X(g^{-1}(y)))-\log\left(\left|\frac{dy}{dx}\right|\right). +\end{aligned} +$$ + +Inituitively, this equation says that the density of $Y$ is equal to the density at the corresponding point in $X$ plus a term that corrects for the warp in volume around an infinitesimally small length around $Y$ caused by the transformation. + +If $g$ is cleverly constructed (and we will see several examples shortly), we can produce distributions that are more complex than standard normal noise and yet have easy sampling and computationally tractable scoring. Moreover, we can compose such bijective transformations to produce even more complex distributions. By an inductive argument, if we have $L$ transforms $g_{(0)}, g_{(1)},\ldots,g_{(L-1)}$, then the log-density of the transformed variable $Y=(g_{(0)}\circ g_{(1)}\circ\cdots\circ g_{(L-1)})(X)$ is + +$$ +\begin{aligned} + \log(p_Y(y)) &= \log\left(p_X\left(\left(g_{(L-1)}^{-1}\circ\cdots\circ g_{(0)}^{-1}\right)\left(y\right)\right)\right)+\sum^{L-1}_{l=0}\log\left(\left|\frac{dg^{-1}_{(l)}(y_{(l)})}{dy'}\right|\right), +\end{aligned} +$$ + +where we've defined $y_{(0)}=x$, $y_{(L-1)}=y$ for convenience of notation. In the [following section](/users/multivariate), we will see how to generalize this method to multivariate $X$. + +## Fixed Univariate `Bijector`s +[FlowTorch](https://flowtorch.ai) contains classes for representing *fixed* univariate bijective transformations. These are particularly useful for restricting the range of transformed distributions, for example to lie on the unit hypercube. (In the following sections, we will explore how to represent learnable bijectors.) + +Let us begin by showing how to represent and manipulate a simple transformed distribution, + +$$ +\begin{aligned} + X &\sim \mathcal{N}(0,1)\\ + Y &= \text{exp}(X). +\end{aligned} +$$ + +You may have recognized that this is by definition, $Y\sim\text{LogNormal}(0,1)$. + +We begin by importing the relevant libraries: + +```python +import torch +import flowtorch.bijectors as bij +import flowtorch.distributions as dist +import matplotlib.pyplot as plt +import seaborn as sns +``` +A variety of bijective transformations live in the `flowtorch.bijectors` module, and the classes to define transformed distributions live in `flowtorch.distributions`. We first create the base distribution of $X$ and the class encapsulating the transform $\text{exp}(\cdot)$: +```python +dist_x = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(1), torch.ones(1)), + 1 +) +bijector = bij.Exp() +``` +The class `bij.Exp` derives from `bij.Fixed` and defines the forward, inverse, and log-absolute-derivative operations for this transform, + +$$ +\begin{aligned} + g(x) &= \text{exp(x)}\\ + g^{-1}(y) &= \log(y)\\ + \log\left(\left|\frac{dg}{dx}\right|\right) &= y. +\end{aligned} +$$ + +In general, a bijector class defines these three operations, from which it is sufficient to perform sampling and scoring. *We should think of a bijector as a plan to construct a normalizing flow rather than the normalizing flow itself* - it requires being instantiated with a concrete base distribution supplying the relevant shape information, +```python +dist_y = dist.Flow(dist_x, bijector) +``` +This statement returns the object `dist_y` of type `flowtorch.distributions.Flow` representing an object that has an interface compatible with `torch.distributions.Distribution`. We are able to sample and score from `dist_y` object using its methods `.sample`, `.rsample`, and `.log_prob`. + +Now, plotting samples from both the base and transformed distributions to verify that we that have produced the log-normal distribution: +```python +plt.subplot(1, 2, 1) +plt.hist(dist_x.sample([1000]).numpy(), bins=50) +plt.title('Standard Normal') +plt.subplot(1, 2, 2) +plt.hist(dist_y.sample([1000]).numpy(), bins=50) +plt.title('Standard Log-Normal') +plt.show() +``` +

+ +

+Our example uses a single transform. However, we can compose transforms to produce more expressive distributions. For instance, if we apply an affine transformation we can produce the general log-normal distribution, + +$$ +\begin{aligned} + X &\sim \mathcal{N}(0,1)\\ + Y &= \text{exp}(\mu+\sigma X). +\end{aligned} +$$ + +or rather, $Y\sim\text{LogNormal}(\mu,\sigma^2)$. In FlowTorch this is accomplished, e.g. for $\mu=3, \sigma=0.5$, as follows: +```python +bijectors = bij.Compose([ + bij.AffineFixed(loc=3, scale=0.5), + bij.Exp()]) +dist_y = dist.Flow(dist_x, bijector) + +plt.subplot(1, 2, 1) +plt.hist(dist_x.sample([1000]).numpy(), bins=50) +plt.title('Standard Normal') +plt.subplot(1, 2, 2) +plt.hist(dist_y.sample([1000]).numpy(), bins=50) +plt.title('Log-Normal') +plt.show() +``` +

+ +

+ +The class `bij.Compose` combines multiple `Bijector`s with [function composition](https://en.wikipedia.org/wiki/Function_composition) to produce a single *plan* for a Normalizing Flow, which is then intiated in the regular way. For the forward operation, transformations are applied in the order of the list. In this case, first `AffineFixed` is applied to the base distribution and then `Exp`. + +## Learnable Univariate `Bijector`s +Having introduced the interface for bijections and transformed distributions, we now show how to represent *learnable* transforms and use them for density estimation. Our dataset in this section and the next will comprise samples along two concentric circles. Examining the joint and marginal distributions: +```python +import numpy as np +from sklearn import datasets +from sklearn.preprocessing import StandardScaler + +n_samples = 1000 +X, y = datasets.make_circles(n_samples=n_samples, factor=0.5, noise=0.05) +X = StandardScaler().fit_transform(X) + +plt.title(r'Samples from $p(x_1,x_2)$') +plt.xlabel(r'$x_1$') +plt.ylabel(r'$x_2$') +plt.scatter(X[:,0], X[:,1], alpha=0.5) +plt.show() +``` + +

+ +

+ +```python +plt.subplot(1, 2, 1) +sns.distplot(X[:,0], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}) +plt.title(r'$p(x_1)$') +plt.subplot(1, 2, 2) +sns.distplot(X[:,1], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}) +plt.title(r'$p(x_2)$') +plt.show() +``` + +

+ +

+ +We will learn the marginals of the above distribution using a learnable transform, `bij.Spline`, defined on a two-dimensional input: + +```python +dist_x = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), + 1 +) +bijector = bij.Spline() +dist_y = dist.Flow(dist_x, bijector) +``` + +`bij.Spline` passes each dimension of its input through a separate monotonically increasing function known as a spline. From a high-level, a spline is a complex parametrizable curve for which we can define specific points known as knots that it passes through and the derivatives at the knots. The knots and their derivatives are parameters that can be learnt, e.g., through stochastic gradient descent on a maximum likelihood objective, as we now demonstrate: + +```python +optimizer = torch.optim.Adam(dist_y.parameters(), lr=1e-2) +for step in range(steps): + optimizer.zero_grad() + loss = -dist_y.log_prob(X).mean() + loss.backward() + optimizer.step() + + if step % 200 == 0: + print('step: {}, loss: {}'.format(step, loss.item())) +``` + +``` +step: 0, loss: 2.682476758956909 +step: 200, loss: 1.278384804725647 +step: 400, loss: 1.2647961378097534 +step: 600, loss: 1.2601449489593506 +step: 800, loss: 1.2561875581741333 +step: 1000, loss: 1.2545257806777954 +``` + +Plotting samples drawn from the transformed distribution after learning: +```python +X_flow = dist_y.sample(torch.Size([1000,])).detach().numpy() +plt.title(r'Joint Distribution') +plt.xlabel(r'$x_1$') +plt.ylabel(r'$x_2$') +plt.scatter(X[:,0], X[:,1], label='data', alpha=0.5) +plt.scatter(X_flow[:,0], X_flow[:,1], color='firebrick', label='flow', alpha=0.5) +plt.legend() +plt.show() +``` + +

+ +

+ +```python +plt.subplot(1, 2, 1) +sns.distplot(X[:,0], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='data') +sns.distplot(X_flow[:,0], hist=False, kde=True, + bins=None, color='firebrick', + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='flow') +plt.title(r'$p(x_1)$') +plt.subplot(1, 2, 2) +sns.distplot(X[:,1], hist=False, kde=True, + bins=None, + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='data') +sns.distplot(X_flow[:,1], hist=False, kde=True, + bins=None, color='firebrick', + hist_kws={'edgecolor':'black'}, + kde_kws={'linewidth': 2}, + label='flow') +plt.title(r'$p(x_2)$') +plt.show() +``` + +

+ +

+ +As we can see, we have learnt close approximations to the marginal distributions, $p(x_1),p(x_2)$. *It would have been challenging to fit the irregularly shaped marginals with standard methods, for example, a mixture of normal distributions*. As expected, since there is a dependency between the two dimensions, we do not learn a good representation of the joint, $p(x_1,x_2)$. In the next section, we explain how to learn multivariate distributions whose dimensions are not independent. diff --git a/website/docusaurus.config.js b/website/docusaurus.config.js new file mode 100644 index 00000000..d63e612f --- /dev/null +++ b/website/docusaurus.config.js @@ -0,0 +1,169 @@ +const math = require('remark-math'); +const katex = require('rehype-katex'); + +module.exports = { + title: 'FlowTorch', + tagline: 'Easily learn and sample complex probability distributions with PyTorch', + url: 'https://flowtorch.ai', + baseUrl: '/', + onBrokenLinks: 'throw', + onBrokenMarkdownLinks: 'warn', + favicon: 'img/favicon.png', + organizationName: 'facebookincubator', + projectName: 'flowtorch', + stylesheets: [ + { + href: 'https://cdn.jsdelivr.net/npm/katex@0.12.0/dist/katex.min.css', + type: 'text/css', + integrity: + 'sha384-AfEj0r4/OFrOo5t7NnNe46zW/tFgW6x/bCJG8FqQCEo3+Aro6EYUG4+cU+KJWu/X', + crossorigin: 'anonymous', + }, + ], + baseUrlIssueBanner: true, + themeConfig: { + announcementBar: { + id: 'supportus', + content: + '⭐️ If you like FlowTorch, give it a star on GitHub! ⭐️', + }, + prism: { + theme: require("prism-react-renderer/themes/github"), + darkTheme: require("prism-react-renderer/themes/dracula"), + }, + navbar: { + title: 'FlowTorch', + logo: { + alt: 'FlowTorch Logo', + src: 'img/logo.svg', + }, + items: [ + { + to: 'users', + activeBasePath: 'users', + label: 'Users', + position: 'left', + }, + { + to: 'dev', + activeBasePath: 'dev', + label: 'Developers', + position: 'left', + }, + { + to: 'api', + activeBasePath: 'api', + label: 'Reference', + position: 'left', + }, + { + href: 'https://github.com/facebookincubator/flowtorch/discussions', + label: 'Discussions', + position: 'right', + }, + { + href: 'https://github.com/facebookincubator/flowtorch/releases', + label: 'Releases', + position: 'right', + }, + { + href: 'https://github.com/facebookincubator/flowtorch', + label: 'GitHub', + position: 'right', + }, + ], + }, + footer: { + style: 'dark', + links: [ + { + title: 'Docs', + items: [ + { + label: 'Users Guide', + to: 'users', + }, + { + label: 'Developers Guide', + to: 'dev', + }, + { + label: 'API Reference', + to: 'api', + }, + { + label: 'Roadmap', + href: 'https://github.com/facebookincubator/flowtorch/projects', + }, + ], + }, + { + title: 'Community', + items: [ + { + label: 'Raise an issue', + href: 'https://github.com/facebookincubator/flowtorch/issues/new/choose', + }, + { + label: 'Ask for help', + href: 'https://github.com/facebookincubator/flowtorch/discussions/new', + }, + { + label: 'Give us feedback', + href: 'https://github.com/facebookincubator/flowtorch/discussions/categories/feedback', + }, + { + label: 'Fork the repo', + href: 'https://github.com/facebookincubator/flowtorch/fork', + }, + ], + }, + { + title: 'Legal', + items: [ + { + label: 'MIT Open Source License', + href: 'https://github.com/facebookincubator/flowtorch/blob/main/LICENSE.txt', + }, + { + label: 'Code of Conduct', + href: 'https://www.contributor-covenant.org/version/1/4/code-of-conduct/', + }, + // Please do not remove the privacy and terms, it's a legal requirement. + { + label: 'Privacy', + href: 'https://opensource.facebook.com/legal/privacy/', + target: '_blank', + rel: 'noreferrer noopener', + }, + { + label: 'Terms', + href: 'https://opensource.facebook.com/legal/terms/', + target: '_blank', + rel: 'noreferrer noopener', + }, + ], + }, + ], + copyright: `Copyright © ${new Date().getFullYear()} Facebook, Inc. and its affiliates. All Rights Reserved.`, + }, + }, + presets: [ + [ + '@docusaurus/preset-classic', + { + docs: { + sidebarPath: require.resolve('./sidebars.js'), + editUrl: 'https://github.com/facebookincubator/flowtorch/edit/main/website/', + routeBasePath: '/', + remarkPlugins: [math], + rehypePlugins: [katex], + }, + blog: false, + theme: { + customCss: require.resolve('./src/css/custom.css'), + }, + }, + ], + ], +}; diff --git a/website/flowtorch-ai.png b/website/flowtorch-ai.png new file mode 100644 index 0000000000000000000000000000000000000000..ce3dadab15563392a826523247ccb0e444adb467 GIT binary patch literal 168097 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%0D=`R{^ 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≤6BREl}KTDHiGCN2wT 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 literal 0 HcmV?d00001 diff --git a/website/package.json b/website/package.json new file mode 100644 index 00000000..0ac06e12 --- /dev/null +++ b/website/package.json @@ -0,0 +1,45 @@ +{ + "name": "flowtorch", + "version": "0.0.0", + "private": true, + "scripts": { + "docusaurus": "docusaurus", + "start": "docusaurus start", + "build": "docusaurus build", + "swizzle": "docusaurus swizzle", + "deploy": "docusaurus deploy", + "serve": "docusaurus serve", + "clear": "docusaurus clear" + }, + "dependencies": { + "@docusaurus/core": "2.0.0-alpha.70", + "@docusaurus/preset-classic": "2.0.0-alpha.70", + "@fortawesome/fontawesome-free": "^5.15.1", + "@fortawesome/fontawesome-svg-core": "^1.2.32", + "@fortawesome/free-brands-svg-icons": "^5.15.1", + "@fortawesome/free-regular-svg-icons": "^5.15.1", + "@fortawesome/free-solid-svg-icons": "^5.15.1", + "@fortawesome/react-fontawesome": "^0.1.14", + "@mdx-js/react": "^1.6.21", + "clsx": "^1.1.1", + "node-fetch": "^2.6.1", + "node-gyp": "^7.1.2", + "react": "^16.8.4", + "react-dom": "^16.8.4", + "react-icons": "^4.2.0", + "rehype-katex": "^4.0.0", + "remark-math": "^3.0.1" + }, + "browserslist": { + "production": [ + ">0.5%", + "not dead", + "not op_mini all" + ], + "development": [ + "last 1 chrome version", + "last 1 firefox version", + "last 1 safari version" + ] + } +} diff --git a/website/sidebars.js b/website/sidebars.js new file mode 100644 index 00000000..8a16e5e1 --- /dev/null +++ b/website/sidebars.js @@ -0,0 +1,15 @@ +var apiSideBar = require('./api.sidebar.js'); + +module.exports = { + usersSidebar: { + "Getting Started": ['users/introduction', 'users/installation', 'users/start'], + "Tutorials": ['users/univariate', 'users/multivariate'], + "Basic Concepts": ['users/shapes'], + }, + devsSidebar: { + "General": ['dev/contributing', 'dev/releases', 'dev/about'], + "Extending the Library": ['dev/overview', 'dev/ops', 'dev/docs', 'dev/tests'], + "Resources": ['dev/bibliography'], + }, + apiSidebar: apiSideBar, +}; diff --git a/website/src/css/custom.css b/website/src/css/custom.css new file mode 100644 index 00000000..32259d4a --- /dev/null +++ b/website/src/css/custom.css @@ -0,0 +1,528 @@ +/* stylelint-disable docusaurus/copyright-header */ +/** + * Any CSS included here will be global. The classic template + * bundles Infima by default. Infima is a CSS framework designed to + * work well for content-centric websites. + */ + +/* You can override the default Infima variables here. */ +:root { + --ifm-background-color: #fff; + --ifm-background-surface-color: #f2f6fa; + --ifm-color-primary: #ff6344; + --ifm-color-primary-dark: #ff4824; + --ifm-color-primary-darker: #ff3b14; + --ifm-color-primary-darkest: #e22500; + --ifm-color-primary-light: #ff7e64; + --ifm-color-primary-lighter: #ff8b74; + --ifm-color-primary-lightest: #ffb4a5; + --ifm-card-background-color: var(--ifm-background-color); + --ifm-footer-background-color: var(--ifm-background-surface-color); + --ifm-hero-background-color: rgb(36, 37, 38); /*var(--ifm-background-surface-color); /* rgb(43, 49, 55);*/ + --ifm-hero-text-color: white; /* rgb(68, 73, 80);*/ + --ifm-navbar-background-color: white; + --color-button-primary: rgb(255, 99, 68); + --color-button-primary-hover: rgb(255, 81, 46); + --color-button-secondary: rgb(255, 186, 0); + --color-button-secondary-hover: rgb(233, 169, 0); + --ifm-footer-link-hover-color: var(--ifm-navbar-link-hover-color) +} + +div[class^='announcementBarContent'] { + background-color: var(--ifm-background-surface-color); + color: var(--ifm-font-color-base); +} + +html[data-theme="dark"] { + --ifm-color-primary: #ffcf5d; + --ifm-color-primary-dark: #ffc53a; + --ifm-color-primary-darker: #ffc029; + --ifm-color-primary-darkest: #f4ab00; + --ifm-color-primary-light: #ffd980; + --ifm-color-primary-lighter: #ffde91; + --ifm-color-primary-lightest: #ffeec5; + --ifm-navbar-background-color: rgb(36, 37, 38); + --ifm-hero-text-color: white; +} + +.hero__subtitle { + font-size: 3.75rem; + text-align: left; +} + +.hero__subtitle .hero__primary { + color:var(--color-button-primary); + text-shadow: 1px 1px 2px var(--color-button-primary-hover); +} + +.hero__subtitle .hero__secondary { + color: var(--color-button-secondary); + text-shadow: 1px 1px 2px var(--color-button-secondary-hover); +} + +.hero__github_button { + border: none; +} + +.hero__buttons .button--primary { + background-color: var(--color-button-primary); + border-color: var(--color-button-primary); +} + +.hero__buttons .button--primary:hover { + background-color: var(--color-button-primary-hover); + border-color: var(--color-button-primary-hover); +} + +html[data-theme="dark"] .hero__buttons .button--primary { + background-color: var(--color-button-primary); + border-color: var(--color-button-primary); +} + +html[data-theme="dark"] .hero__buttons .button--primary:hover { + background-color: var(--color-button-primary-hover); + border-color: var(--color-button-primary-hover); +} + +#features .col { + padding-left: 2rem; + padding-right: 2rem; +} + +.hero__img { + float: right; + max-width: 200px; + padding-left: 40px; + padding-right: 6rem; + padding-top: 1.1rem; +} + +.hero__buttons { + padding-top: 2rem; +} + +.hero__buttons .button { + margin-right: 3rem; +} + +.banner_src-theme-Hero- { + text-align: left; +} + +.hero__buttons .button--lg { + --ifm-button-size-multiplier: 1.5; +} + +.docusaurus-highlight-code-line { + display: block; + margin: 0 calc(-1 * var(--ifm-pre-padding)); + padding: 0 var(--ifm-pre-padding); + background-color: rgb(72, 77, 91); +} + +html[data-theme="dark"] .docusaurus-highlight-code-line { + background-color: rgba(0, 0, 0, 0.3); +} + +/*.navbar__items { + vertical-align: middle; +}*/ + +a { + transition: none; +} + +.navbar__item.navbar__link[href*="users"] { + display: flex; + padding: 0px 12px; + padding-right: 12px; +} +.navbar__item.navbar__link[href*="users"]:before { + mask: url('/img/user-astronaut-solid.svg'); + mask-size: cover; + display: inline-block; + content: ''; + background-color: black; + width: 0.875rem; + height: 1rem; + margin-top: 0.3rem; + margin-right: 0.5rem; +} + +.navbar__title:hover { + /*color: red;*/ + text-align: center; + -webkit-animation: glow 1s ease-in-out infinite alternate; + -moz-animation: glow 1s ease-in-out infinite alternate; + animation: glow 1s ease-in-out infinite alternate; +} + +@keyframes glow { + from { + text-shadow: 0 0 10px #fff, 0 0 20px #fff, 0 0 30px #e60073, 0 0 40px #e60073, 0 0 50px #e60073, 0 0 60px #e60073, 0 0 70px #e60073; + } + to { + text-shadow: 0 0 20px #fff, 0 0 30px #ff4da6, 0 0 40px #ff4da6, 0 0 50px #ff4da6, 0 0 60px #ff4da6, 0 0 70px #ff4da6, 0 0 80px #ff4da6; + } +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="users"]:before { + background-color: white; +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="users"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.navbar__item.navbar__link[href*="users"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.navbar__item.navbar__link[href*="dev"] { + display: flex; + padding: 0px 12px; +} +.navbar__item.navbar__link[href*="dev"]:before { + mask: url("/img/hat-wizard-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: black; + width: 1rem; + height: 1rem; + margin-top: 0.3rem; + margin-right: 0.5rem; +} + +.navbar__item.navbar__link[href*="dev"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="dev"]:before { + background-color: white; +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="dev"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.navbar__item.navbar__link[href*="api"] { + display: flex; + padding: 0px 12px; +} +.navbar__item.navbar__link[href*="api"]:before { + mask: url("/img/book-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: black; + width: 0.875rem; + height: 1rem; + margin-top: 0.3rem; + margin-right: 0.5rem; +} + +.navbar__item.navbar__link[href*="api"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="api"]:before { + background-color: white; +} + +html[data-theme="dark"] .navbar__item.navbar__link[href*="api"]:hover:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.footer__link-item:before { + opacity:0; +} + +.footer__link-item:hover:before { + opacity:1; +} + +.footer__link-item:hover:before { + animation: fadeIn ease 0.25s; + -webkit-animation: fadeIn ease 0.25s; + -moz-animation: fadeIn ease 0.25s; + -o-animation: fadeIn ease 0.25s; + -ms-animation: fadeIn ease 0.25s; +} + +@keyframes fadeIn { + 0% {opacity:0;} + 100% {opacity:1;} +} + +.footer__link-item { + margin-left: -1.75rem; +} + +.footer__link-item[href*="api"] { + display: flex; +} + +.footer__link-item[href*="api"]:before { + mask: url("/img/book-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 0.875rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.6875rem; + margin-left: 0.1875rem; +} + + +.footer__link-item[href*="dev"] { + display: flex; +} + +.footer__link-item[href*="dev"]:before { + mask: url("/img/hat-wizard-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1rem; + height: 1rem; + margin-top: 0.5rem; + margin-left: 0.125rem; + margin-right: 0.625rem; +} + +.footer__link-item[href*="users"] { + display: flex; +} + +.footer__link-item[href*="users"]:before { + mask: url("/img/user-astronaut-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 0.875rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.6875rem; + margin-left: 0.1875rem; +} + +.footer__link-item[href*="projects"] { + display: flex; +} + +.footer__link-item[href*="projects"]:before { + mask: url("/img/tasks-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 0.875rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.6875rem; + margin-left: 0.1875rem; +} + +.footer__link-item[href*="fork"] { + display: flex; +} + +.footer__link-item[href*="fork"]:before { + mask: url("/img/code-branch-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 0.75rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.875rem; + margin-left: 0.1875rem; +} + +.footer__link-item[href*="choose"] { + display: flex; +} + +.footer__link-item[href*="choose"]:before { + mask: url("/img/hand-sparkles-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.25rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; +} + +.footer__link-item[href*="discussions"] { + display: flex; +} + +.footer__link-item[href*="discussions"]:before { + mask: url("/img/comments-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.125rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5625rem; + margin-left: 0.0625rem; +} + +.footer__link-item[href*="feedback"] { + display: flex; +} + +.footer__link-item[href*="feedback"]:before { + mask: url("/img/smile-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 0.99rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.63rem; + margin-left: 0.13rem; +} + +.footer__link-item[href*="LICENSE.txt"] { + display: flex; +} + +.footer__link-item[href*="LICENSE.txt"]:before { + mask: url("/img/scroll-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.25rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; + /*margin-left: 0.1875rem;*/ +} + +.footer__link-item[href*="code-of-conduct/"] { + display: flex; +} + +.footer__link-item[href*="code-of-conduct/"]:before { + mask: url("/img/handshake-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.25rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; + /*margin-left: 0.1875rem;*/ +} + +.footer__link-item[href*="legal/terms"]:before { + mask: url("/img/scroll-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.25rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; + /*margin-left: 0.1875rem;*/ +} + +.footer__link-item[href*="legal/privacy"]:before { + mask: url("/img/handshake-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.25rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; + /*margin-left: 0.1875rem;*/ +} + +.button[href*="user"] { + display: flex; +} + +.button[href*="user"]:before { + mask: url("/img/play-circle-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.0rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; +} + +.button[href*="dev"] { + display: flex; +} + +.button[href*="dev"]:before { + mask: url("/img/tools-solid.svg"); + mask-size: cover; + display: inline-block; + content: ''; + background-color: white; + width: 1.0rem; + height: 1rem; + margin-top: 0.5rem; + margin-right: 0.5rem; +} + +html[data-theme="dark"] .button[href*="user"]:before { + background-color: black; +} + +html[data-theme="dark"] .button[href*="dev"]:before { + background-color: black; +} + +.footer__link-item:hover:before { + background-color: var(--ifm-footer-link-hover-color); +} + +.navbar__item.navbar__link--active[href*="user"]:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.navbar__item.navbar__link--active[href*="dev"]:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +.navbar__item.navbar__link--active[href*="api"]:before { + background-color: var(--ifm-navbar-link-hover-color); +} + +/*.tooltip { + position: relative; + display: inline-block; +} + +.tooltip .tooltiptext::after { + content: " "; + position: absolute; + bottom: 100%; + left: 50%; + margin-left: -5px; + border-width: 5px; + border-style: solid; + border-color: transparent transparent black transparent; +}*/ diff --git a/website/src/pages/index.js b/website/src/pages/index.js new file mode 100644 index 00000000..d482e301 --- /dev/null +++ b/website/src/pages/index.js @@ -0,0 +1,35 @@ +import React from 'react'; +import clsx from 'clsx'; +import Layout from '@theme/Layout'; +import Link from '@docusaurus/Link'; +import useDocusaurusContext from '@docusaurus/useDocusaurusContext'; +import useBaseUrl from '@docusaurus/useBaseUrl'; +import styles from './styles.module.css'; + +// Prism (Rust) +import Prism from "prism-react-renderer/prism"; +(typeof global !== "undefined" ? global : window).Prism = Prism; +require("prismjs/components/prism-rust"); + +// Our theme +import Examples from "@theme/Examples"; +import Features from "@theme/Features"; +import Hero from "@theme/Hero"; + +function Home() { + const context = useDocusaurusContext(); + const {siteConfig = {}} = context; + return ( + + +

+ + +
+ + ); +} + +export default Home; diff --git a/website/src/pages/styles.module.css b/website/src/pages/styles.module.css new file mode 100644 index 00000000..53ddcd63 --- /dev/null +++ b/website/src/pages/styles.module.css @@ -0,0 +1,42 @@ +/* stylelint-disable docusaurus/copyright-header */ + +/** + * CSS files with the .module.css suffix will be treated as CSS modules + * and scoped locally. + */ + +.heroBanner { + padding: 4rem 0; + text-align: center; + position: relative; + overflow: hidden; +} + +@media screen and (max-width: 966px) { + .heroBanner { + padding: 2rem; + } +} + +.heroImg { + height: 200px; + margin: 10px 0; +} + +.buttons { + display: flex; + align-items: center; + justify-content: center; +} + +.features { + display: flex; + align-items: center; + padding: 2rem 0; + width: 100%; +} + +.featureImage { + height: 200px; + width: 200px; +} diff --git a/website/src/theme/CodeSnippet/index.js b/website/src/theme/CodeSnippet/index.js new file mode 100644 index 00000000..a5081c7d --- /dev/null +++ b/website/src/theme/CodeSnippet/index.js @@ -0,0 +1,57 @@ +import React, { useEffect, useState } from "react"; +import Highlight, { defaultProps } from "prism-react-renderer"; +import useDocusaurusContext from "@docusaurus/useDocusaurusContext"; +import useThemeContext from "@theme/hooks/useThemeContext"; + +import styles from "./styles.module.css"; + +function CodeSnippet(props) { + const { + siteConfig: { + themeConfig: { prism = {} }, + }, + } = useDocusaurusContext(); + + const [mounted, setMounted] = useState(false); + // The Prism theme on SSR is always the default theme but the site theme + // can be in a different mode. React hydration doesn't update DOM styles + // that come from SSR. Hence force a re-render after mounting to apply the + // current relevant styles. There will be a flash seen of the original + // styles seen using this current approach but that's probably ok. Fixing + // the flash will require changing the theming approach and is not worth it + // at this point. + useEffect(() => { + setMounted(true); + }, []); + + const { isDarkTheme } = useThemeContext(); + const lightModeTheme = prism.theme; + const darkModeTheme = prism.darkTheme || lightModeTheme; + const prismTheme = isDarkTheme ? darkModeTheme : lightModeTheme; + + const { language = "python", code } = props; + + return ( + + {({ className, style, tokens, getLineProps, getTokenProps }) => ( +
+          {tokens.map((line, i) => (
+            
+ {line.map((token, key) => ( + + ))} +
+ ))} +
+ )} +
+ ); +} + +export default CodeSnippet; diff --git a/website/src/theme/CodeSnippet/styles.module.css b/website/src/theme/CodeSnippet/styles.module.css new file mode 100644 index 00000000..e88b5dbb --- /dev/null +++ b/website/src/theme/CodeSnippet/styles.module.css @@ -0,0 +1,3 @@ +.code { + font-size: 10pt; +} diff --git a/website/src/theme/Examples/index.js b/website/src/theme/Examples/index.js new file mode 100644 index 00000000..ce533ad1 --- /dev/null +++ b/website/src/theme/Examples/index.js @@ -0,0 +1,63 @@ +import React from "react"; +import CodeSnippet from "@theme/CodeSnippet"; +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + +import Headline from "@theme/Headline"; +import snippets from "./snippets"; +import styles from "./styles.module.css"; + +function renderTabs() { + return ( + { + return { label: props.label, value: props.label }; + })} + className={styles.tabs} + > + {snippets.map((props, idx) => ( + + + + ))} + + ); +} + +function Examples() { + return ( + <> + {snippets && snippets.length && ( +
+
+
+
+ + {renderTabs()} +
+
+
+ + + + + + + + + + +
+
+
+
+
+ )} + + ); +} + +export default Examples; diff --git a/website/src/theme/Examples/snippets.js b/website/src/theme/Examples/snippets.js new file mode 100644 index 00000000..2083f6f9 --- /dev/null +++ b/website/src/theme/Examples/snippets.js @@ -0,0 +1,43 @@ +const snippets = [ + { + label: "Bivariate Normal", + code: +`import torch +import flowtorch.bijectors as bij +import flowtorch.distributions as dist +import flowtorch.parameters as params + +# Lazily instantiated flow plus base and target distributions +params = params.DenseAutoregressive(hidden_dims=(32,)) +bijectors = bij.AffineAutoregressive(params=params) +base_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2), torch.ones(2)), + 1 +) +target_dist = torch.distributions.Independent( + torch.distributions.Normal(torch.zeros(2)+5, torch.ones(2)*0.5), + 1 +) + +# Instantiate transformed distribution and parameters +flow = dist.Flow(base_dist, bijectors) + +# Training loop +opt = torch.optim.Adam(flow.parameters(), lr=5e-3) +frame = 0 +for idx in range(3001): + opt.zero_grad() + + # Minimize KL(p || q) + y = target_dist.sample((1000,)) + loss = -flow.log_prob(y).mean() + + if idx % 500 == 0: + print('epoch', idx, 'loss', loss) + + loss.backward() + opt.step()`, + }, +]; + +export default snippets; diff --git a/website/src/theme/Examples/styles.module.css b/website/src/theme/Examples/styles.module.css new file mode 100644 index 00000000..5229caf5 --- /dev/null +++ b/website/src/theme/Examples/styles.module.css @@ -0,0 +1,105 @@ +/*.examples { + border-top: 1px solid var(--ifm-color-emphasis-300); +}*/ + +.examples { + margin-bottom: 3.5rem; +} + +.animation_svg { + margin: auto; + padding-top: 8rem; + width: 30rem; + height: 38rem; +} + +.animation > * { + opacity: 0; + animation-duration: 7s; + animation-iteration-count: infinite; + animation-timing-function: steps(1); +} + +.example_container { display: flex; justify-content: center; height: 721px;} + +html[data-theme="dark"] .example_container svg { + filter: invert(100%); +} + +@keyframes animation-1 { + 0% { + opacity: 1; + } + 16.66666% { + opacity: 0; + } +} + +.animation > :nth-child(1) { + animation-name: animation-1; +} + +@keyframes animation-2 { + 16.66666% { + opacity: 1; + } + 33.33333% { + opacity: 0; + } +} + +.animation > :nth-child(2) { + animation-name: animation-2; +} + +@keyframes animation-3 { + 33.33333% { + opacity: 1; + } + 50% { + opacity: 0; + } +} + +.animation > :nth-child(3) { + animation-name: animation-3; +} + +@keyframes animation-4 { + 50% { + opacity: 1; + } + 66.66667% { + opacity: 0; + } +} + +.animation > :nth-child(4) { + animation-name: animation-4; +} + +@keyframes animation-5 { + 66.66667% { + opacity: 1; + } + 83.33333% { + opacity: 0; + } +} + +.animation > :nth-child(5) { + animation-name: animation-5; +} + +@keyframes animation-6 { + 83.33333% { + opacity: 1; + } + 100% { + opacity: 1; + } +} + +.animation > :nth-child(6) { + animation-name: animation-6; +} diff --git a/website/src/theme/Features/index.js b/website/src/theme/Features/index.js new file mode 100644 index 00000000..98ee6519 --- /dev/null +++ b/website/src/theme/Features/index.js @@ -0,0 +1,85 @@ +import React from "react"; +import clsx from "clsx"; +import { FaMeteor, FaDumbbell, FaHandsHelping, FaCubes, FaIndustry } from "react-icons/fa"; + +import styles from "./styles.module.css"; + +const size = 24; +const data = [ + { + icon: , + title: <>Simple but powerful, + description: ( + <> + Design, train, and sample from complex probability distributions using only a few lines of code. Advanced features such as conditionality, caching, and structured representations are planned for future released. + + ), + }, + { + icon: , + title: <>Community focused, + description: ( + <> + We help you be a successful user or contributor through detailed user, developer, and API guides. Educational tutorials and research benchmarks are planned for the future. We welcome your feedback! + + ), + }, + { + icon: , + title: <>Modular and extendable, + description: ( + <> + Combine multiple bijections to form complex normalizing flows, and mix-and-match conditioning networks with bijections. + FlowTorch has a well-defined interface so you easily create your own components! + + ), + }, + { + icon: , + title: <>Production ready, + description: ( + <> + Proven code migrated from Pyro, with improved unit testing and continuous integration. + And it is easy to add standard unit tests to components you write yourself! + + ), + }, +]; + +function Feature({ icon, title, description }) { + return ( +
+
+
+ {icon &&
{icon}
} +

{title}

+
+

{description}

+
+
+ ); +} + +function Features() { + return ( + <> + {data && data.length && ( +
+
+
+
+
+ {data.map((props, idx) => ( + + ))} +
+
+
+
+
+ )} + + ); +} + +export default Features; diff --git a/website/src/theme/Features/styles.module.css b/website/src/theme/Features/styles.module.css new file mode 100644 index 00000000..78aab439 --- /dev/null +++ b/website/src/theme/Features/styles.module.css @@ -0,0 +1,40 @@ +.features { + display: flex; + align-items: center; + width: 100%; + padding-bottom: 1rem !important; + padding-top: 6rem; + background-color: --var(--ifm-background-color); +} + +.features .feature p { + margin-bottom: 0; +} + +.features .feature:not(:last-child) { + margin-bottom: 3rem; +} + +/* @media screen and (min-width: 576px) { + margin-bottom: 3rem; + padding-right: 5rem; + }*/ + +.features .feature .header { + display: flex; + flex-direction: row; + align-items: center; + margin-bottom: 1rem; +} + +.features .feature .header .icon { + display: flex; + align-items: center; + margin-right: 1rem; + color: var(--ifm-color-primary); +} + +.features .feature .header .title { + font-size: 1.25rem; + margin-bottom: 0; +} diff --git a/website/src/theme/Headline/index.js b/website/src/theme/Headline/index.js new file mode 100644 index 00000000..7887a90c --- /dev/null +++ b/website/src/theme/Headline/index.js @@ -0,0 +1,33 @@ +import React from "react"; +import { PropTypes } from "prop-types"; + +import styles from "./styles.module.css"; + +function Headline(props) { + const { category, title, subtitle, offset } = props; + + return ( +
+
+
+ {category && {category}} + {title &&

{title}

} + {subtitle &&

{subtitle}

} +
+
+
+ ); +} + +Headline.propTypes = { + category: PropTypes.string, + title: PropTypes.string, + subtitle: PropTypes.string, + offset: PropTypes.number, +}; + +Headline.defaultProps = { + offset: 0, +}; + +export default Headline; diff --git a/website/src/theme/Headline/styles.module.css b/website/src/theme/Headline/styles.module.css new file mode 100644 index 00000000..f2e3bddd --- /dev/null +++ b/website/src/theme/Headline/styles.module.css @@ -0,0 +1,28 @@ +.headline { + margin-top: 2rem; +} + +.category { + display: inline-flex; + align-items: center; + margin-bottom: 1rem; + font-weight: bold; + text-transform: uppercase; + color: var(--ifm-color-primary-light); +} + +.title { + max-width: 500px; + font-size: 2rem; + line-height: initial; + word-spacing: -0.25rem; + + @media screen and (min-width: 576px) { + font-size: 2.4rem; + } +} + +.subtitle { + margin-top: 2rem; + color: var(--ifm-color-emphasis-600); +} diff --git a/website/src/theme/Hero/index.js b/website/src/theme/Hero/index.js new file mode 100644 index 00000000..6e456743 --- /dev/null +++ b/website/src/theme/Hero/index.js @@ -0,0 +1,62 @@ +import React from "react"; +import clsx from "clsx"; +import Link from "@docusaurus/Link"; +import useBaseUrl from "@docusaurus/useBaseUrl"; +import useDocusaurusContext from "@docusaurus/useDocusaurusContext"; + +import styles from "./styles.module.css"; + +function Hero() { + const context = useDocusaurusContext(); + const { siteConfig = {} } = context; + + return ( +
+
+
+ +

+ + Easily learn and sample complex probability distributions with PyTorch +

+ +
+
+
+ + Get Started + +
+
+ + Contribute + + +