From 393ad44407d8658a3e5f984b9ce49a7dcd760d96 Mon Sep 17 00:00:00 2001
From: T-Dynamos <anshdadwal298@gmail.com>
Date: Sun, 25 Jun 2023 08:18:37 +0530
Subject: [PATCH 001/158] libzmq: fix build
---
pythonforandroid/recipes/libzmq/__init__.py | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/libzmq/__init__.py b/pythonforandroid/recipes/libzmq/__init__.py
index 243517bc96..5a05edf1f0 100644
--- a/pythonforandroid/recipes/libzmq/__init__.py
+++ b/pythonforandroid/recipes/libzmq/__init__.py
@@ -6,7 +6,7 @@
class LibZMQRecipe(Recipe):
- version = '4.3.2'
+ version = '4.3.4'
url = 'https://github.com/zeromq/libzmq/releases/download/v{version}/zeromq-{version}.zip'
depends = []
built_libraries = {'libzmq.so': 'src/.libs'}
@@ -34,6 +34,7 @@ def build_arch(self, arch):
'--prefix={}'.format(prefix),
'--with-libsodium=no',
'--disable-libunwind',
+ '--disable-Werror',
_env=env)
shprint(sh.make, _env=env)
shprint(sh.make, 'install', _env=env)
From 12ef2a23be5e10aea1a5de1dd8657e4145e569cf Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sun, 17 Sep 2023 16:15:27 +0200
Subject: [PATCH 002/158] Update CHANGELOG.md and bump version to 2023.09.16
---
CHANGELOG.md | 60 ++++++++++++++++++++++++++++++++++++
pythonforandroid/__init__.py | 2 +-
2 files changed, 61 insertions(+), 1 deletion(-)
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 98288f265d..2ce4711cbb 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,65 @@
# Changelog
+## [v2023.09.16](https://github.com/kivy/python-for-android/tree/v2023.09.16)
+
+[Full Changelog](https://github.com/kivy/python-for-android/compare/v2023.05.21...v2023.09.16)
+
+**Closed issues:**
+
+- Microphone And Audio permissions doesn't work [\#2889](https://github.com/kivy/python-for-android/issues/2889)
+- Error with Scipy [\#2883](https://github.com/kivy/python-for-android/issues/2883)
+- Download failed \( Downloading sqlite3 from https://www.sqlite.org/2021/sqlite-amalgamation-3350500.zip \) during buildozer -v andriod debug [\#2881](https://github.com/kivy/python-for-android/issues/2881)
+- ONNXruntime lib open failed due to 64-bit [\#2880](https://github.com/kivy/python-for-android/issues/2880)
+- Question: how to write a recipe to download and install\(coz build fail\) a wheel package directly? [\#2878](https://github.com/kivy/python-for-android/issues/2878)
+- Impossible to install python for android [\#2867](https://github.com/kivy/python-for-android/issues/2867)
+- scipy with kivy [\#2861](https://github.com/kivy/python-for-android/issues/2861)
+- False positve parser warning. [\#2856](https://github.com/kivy/python-for-android/issues/2856)
+- After successfully building app, its crashes with this error, using firebase-admin [\#2854](https://github.com/kivy/python-for-android/issues/2854)
+- Kivy [\#2837](https://github.com/kivy/python-for-android/issues/2837)
+- not installing on windows 10 [\#2836](https://github.com/kivy/python-for-android/issues/2836)
+- Could not find com.android.tools.build:gradle:7.2.0. in android studio [\#2834](https://github.com/kivy/python-for-android/issues/2834)
+- vlc recipe build fail [\#2822](https://github.com/kivy/python-for-android/issues/2822)
+- mysqldb recipe build fail [\#2813](https://github.com/kivy/python-for-android/issues/2813)
+- babel recipe build fail [\#2803](https://github.com/kivy/python-for-android/issues/2803)
+- Python 3.10 cffi build fails [\#2799](https://github.com/kivy/python-for-android/issues/2799)
+- \[Recipe request\] OpenColorIO & rawpy \(libraw\) [\#2789](https://github.com/kivy/python-for-android/issues/2789)
+- Longer app load time, bigger apk size and unnecessary logs [\#2704](https://github.com/kivy/python-for-android/issues/2704)
+- -fp-model argument not found, directory strict not found [\#2329](https://github.com/kivy/python-for-android/issues/2329)
+- Kivy crashes on Android: ImportError: dlopen failed: library "libpython3.7m.so" not found [\#2237](https://github.com/kivy/python-for-android/issues/2237)
+- pyconfig.h issue [\#2074](https://github.com/kivy/python-for-android/issues/2074)
+
+**Merged pull requests:**
+
+- Use Python's touch\(\) rather than shelling out. [\#2886](https://github.com/kivy/python-for-android/pull/2886) ([Julian-O](https://github.com/Julian-O))
+- Standardise on move [\#2884](https://github.com/kivy/python-for-android/pull/2884) ([Julian-O](https://github.com/Julian-O))
+- Remove deprecated FlatDir in Gradle template [\#2876](https://github.com/kivy/python-for-android/pull/2876) ([Julian-O](https://github.com/Julian-O))
+- :rotating\_light: linter fixes [\#2874](https://github.com/kivy/python-for-android/pull/2874) ([AndreMiras](https://github.com/AndreMiras))
+- Standardise ensure\_dir and rmdir [\#2871](https://github.com/kivy/python-for-android/pull/2871) ([Julian-O](https://github.com/Julian-O))
+- Correct check for --sdk option [\#2870](https://github.com/kivy/python-for-android/pull/2870) ([Julian-O](https://github.com/Julian-O))
+- Python versions: Update documentation & CI testing [\#2869](https://github.com/kivy/python-for-android/pull/2869) ([Julian-O](https://github.com/Julian-O))
+- Patching cleanup [\#2868](https://github.com/kivy/python-for-android/pull/2868) ([Julian-O](https://github.com/Julian-O))
+- Factor out dependency checking. Use modern version handling [\#2866](https://github.com/kivy/python-for-android/pull/2866) ([Julian-O](https://github.com/Julian-O))
+- `build_platform` should be all-lowercase [\#2864](https://github.com/kivy/python-for-android/pull/2864) ([misl6](https://github.com/misl6))
+- Fix simple typos in comments [\#2863](https://github.com/kivy/python-for-android/pull/2863) ([Julian-O](https://github.com/Julian-O))
+- Use a pinned version of `Cython` for now, as most of the recipes are incompatible with `Cython==3.x.x` [\#2862](https://github.com/kivy/python-for-android/pull/2862) ([misl6](https://github.com/misl6))
+- Docs: Fix typos and updated command to build apk - README [\#2860](https://github.com/kivy/python-for-android/pull/2860) ([kulothunganug](https://github.com/kulothunganug))
+- Docs: Fix code string - quickstart.rst [\#2859](https://github.com/kivy/python-for-android/pull/2859) ([kulothunganug](https://github.com/kulothunganug))
+- Automatically generate required pre-requisites [\#2858](https://github.com/kivy/python-for-android/pull/2858) ([Julian-O](https://github.com/Julian-O))
+- Use platform.uname instead of os.uname [\#2857](https://github.com/kivy/python-for-android/pull/2857) ([Julian-O](https://github.com/Julian-O))
+- Bump `kivy` version to `2.2.1` [\#2855](https://github.com/kivy/python-for-android/pull/2855) ([misl6](https://github.com/misl6))
+- Correct sys\_platform [\#2852](https://github.com/kivy/python-for-android/pull/2852) ([Julian-O](https://github.com/Julian-O))
+- Changed the url to use https as http fails [\#2846](https://github.com/kivy/python-for-android/pull/2846) ([kuzeyron](https://github.com/kuzeyron))
+- vlc: fix build [\#2841](https://github.com/kivy/python-for-android/pull/2841) ([T-Dynamos](https://github.com/T-Dynamos))
+- Optimize CI runs, by avoiding unnecessary rebuilds [\#2833](https://github.com/kivy/python-for-android/pull/2833) ([misl6](https://github.com/misl6))
+- Remove `pytz` recipe, as it's not needed anymore [\#2830](https://github.com/kivy/python-for-android/pull/2830) ([misl6](https://github.com/misl6))
+- Remove `dateutil` recipe, as it's not needed anymore [\#2829](https://github.com/kivy/python-for-android/pull/2829) ([misl6](https://github.com/misl6))
+- Removes `mysqldb` recipe as does not support Python 3 [\#2828](https://github.com/kivy/python-for-android/pull/2828) ([misl6](https://github.com/misl6))
+- Bump `actions/setup-python` and `actions/checkout` versions, as old ones are deprecated [\#2827](https://github.com/kivy/python-for-android/pull/2827) ([misl6](https://github.com/misl6))
+- Removes `Babel` recipe as it's not needed anymore. [\#2826](https://github.com/kivy/python-for-android/pull/2826) ([misl6](https://github.com/misl6))
+- Cffi update [\#2800](https://github.com/kivy/python-for-android/pull/2800) ([HyTurtle](https://github.com/HyTurtle))
+- Merge master into develop [\#2797](https://github.com/kivy/python-for-android/pull/2797) ([misl6](https://github.com/misl6))
+- Use build rather than pep517 for building [\#2784](https://github.com/kivy/python-for-android/pull/2784) ([s-t-e-v-e-n-k](https://github.com/s-t-e-v-e-n-k))
+
## [v2023.05.21](https://github.com/kivy/python-for-android/tree/v2023.05.21)
[Full Changelog](https://github.com/kivy/python-for-android/compare/v2023.02.10...v2023.05.21)
diff --git a/pythonforandroid/__init__.py b/pythonforandroid/__init__.py
index ec30da5902..bcabd44859 100644
--- a/pythonforandroid/__init__.py
+++ b/pythonforandroid/__init__.py
@@ -1 +1 @@
-__version__ = '2023.05.21'
+__version__ = '2023.09.16'
From 3c053660438c5a36133c5ba9bc7d7f9b5d82e718 Mon Sep 17 00:00:00 2001
From: Hugo <mail@hugodaniel.pt>
Date: Tue, 17 Oct 2023 21:24:28 +0100
Subject: [PATCH 003/158] Update Lottie player version
---
.../bootstraps/common/build/templates/build.tmpl.gradle | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
index ce105736d3..d8d4326115 100644
--- a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
+++ b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
@@ -125,7 +125,7 @@ dependencies {
{%- endfor %}
{%- endif %}
{% if args.presplash_lottie %}
- implementation 'com.airbnb.android:lottie:3.4.0'
+ implementation 'com.airbnb.android:lottie:6.1.0'
{%- endif %}
}
From 0be5572a43aed4c199e740d7d467fde19779736e Mon Sep 17 00:00:00 2001
From: Julian <github@somethinkodd.com>
Date: Sun, 5 Nov 2023 04:15:28 +1100
Subject: [PATCH 004/158] Improve docs, add docs tests and remove vendored
theme (#2890)
* Add doc tests, make them pass
* push.yml
* Added documentation step which installs [docs], checks links and builds documentation to find any errors.
* Need to install to get all the dependencies, because p4a code is imported during doc generation.
* Updated rest of checkout action versions to be consistent with new code.
* apis.rst
* http -> https
* Disambiguated two modules with the same name.
* buildoptions.rst
* Corrected indenting
* conf,py
* Added language
* Added two URLS to ignore that link-checker was failing even though they worked.
* ext/sphinix_rtd_theme/*
* Old theme did not work with Sphinx 7.0+.
* Removed it from source control. Install the latest each time.
* launcher.rst
* This described an old feature that hasn't worked for years. Couldn't fix the links because the whole feature is gone now. Removed it. If the attempts to bring the feature back ever come to fruition, this can be restored and edited.
* index.rst
* Removed no-longer-existing anchor,
* Removed launcher page
* quickstart.rst
* Removed launcher
* recipe.rst
* Fixed syntax error in rst.
* services,rst
* Fixed broken URL (and linked to project rather than bug-tracker.)
* troubleshooting.rst
* http->https
* Link to English rather than Chinese web-sites.
* setup.py
* Add docs as extra.
* Updated with review comments.
* Fix broken links in documentation.
One dead link, causing test to fail.
Several permanent redirects where it is better to use the final destination.
The rest
---
.github/workflows/push.yml | 24 +-
doc/source/apis.rst | 12 +-
doc/source/buildoptions.rst | 2 +-
doc/source/conf.py | 14 +-
doc/source/ext/sphinx_rtd_theme/.gitignore | 13 -
doc/source/ext/sphinx_rtd_theme/Gemfile | 4 -
doc/source/ext/sphinx_rtd_theme/Gemfile.lock | 16 -
doc/source/ext/sphinx_rtd_theme/Gruntfile.js | 105 ----
doc/source/ext/sphinx_rtd_theme/LICENSE | 20 -
doc/source/ext/sphinx_rtd_theme/MANIFEST.in | 10 -
doc/source/ext/sphinx_rtd_theme/README.rst | 176 ------
doc/source/ext/sphinx_rtd_theme/bower.json | 33 -
.../ext/sphinx_rtd_theme/demo_docs/Makefile | 153 -----
.../demo_docs/source/__init__.py | 0
.../sphinx_rtd_theme/demo_docs/source/conf.py | 254 --------
.../demo_docs/source/demo.rst | 577 ------------------
.../demo_docs/source/index.rst | 217 -------
.../demo_docs/source/list.rst | 69 ---
.../source/static/yi_jing_01_chien.jpg | Bin 9498 -> 0 bytes
.../source/test_py_module/__init__.py | 0
.../demo_docs/source/test_py_module/test.py | 103 ----
doc/source/ext/sphinx_rtd_theme/package.json | 18 -
.../ext/sphinx_rtd_theme/requirements.txt | 1 -
.../sphinx_rtd_theme/sass/_theme_badge.sass | 93 ---
.../sass/_theme_badge_fa.sass | 68 ---
.../sass/_theme_breadcrumbs.sass | 25 -
.../_theme_font_awesome_compatability.sass | 22 -
.../sphinx_rtd_theme/sass/_theme_layout.sass | 292 ---------
.../sphinx_rtd_theme/sass/_theme_mathjax.sass | 5 -
.../ext/sphinx_rtd_theme/sass/_theme_rst.sass | 282 ---------
.../sass/_theme_variables.sass | 12 -
.../ext/sphinx_rtd_theme/sass/badge_only.sass | 16 -
.../ext/sphinx_rtd_theme/sass/theme.sass | 53 --
.../ext/sphinx_rtd_theme/screen_desktop.png | Bin 156671 -> 0 bytes
.../ext/sphinx_rtd_theme/screen_mobile.png | Bin 96294 -> 0 bytes
doc/source/ext/sphinx_rtd_theme/setup.py | 43 --
.../sphinx_rtd_theme/__init__.py | 17 -
.../sphinx_rtd_theme/breadcrumbs.html | 19 -
.../sphinx_rtd_theme/footer.html | 32 -
.../sphinx_rtd_theme/layout.html | 162 -----
.../sphinx_rtd_theme/layout_old.html | 205 -------
.../sphinx_rtd_theme/search.html | 50 --
.../sphinx_rtd_theme/searchbox.html | 7 -
.../static/css/badge_only.css | 2 -
.../sphinx_rtd_theme/static/css/theme.css | 5 -
.../static/fonts/FontAwesome.otf | Bin 62856 -> 0 bytes
.../static/fonts/fontawesome-webfont.eot | Bin 38205 -> 0 bytes
.../static/fonts/fontawesome-webfont.svg | 414 -------------
.../static/fonts/fontawesome-webfont.ttf | Bin 80652 -> 0 bytes
.../static/fonts/fontawesome-webfont.woff | Bin 44432 -> 0 bytes
.../sphinx_rtd_theme/static/js/theme.js | 47 --
.../sphinx_rtd_theme/theme.conf | 8 -
.../sphinx_rtd_theme/versions.html | 37 --
doc/source/index.rst | 3 +-
doc/source/launcher.rst | 101 ---
doc/source/quickstart.rst | 1 -
doc/source/recipes.rst | 2 +-
doc/source/services.rst | 4 +-
doc/source/troubleshooting.rst | 6 +-
setup.py | 3 +
60 files changed, 46 insertions(+), 3811 deletions(-)
delete mode 100644 doc/source/ext/sphinx_rtd_theme/.gitignore
delete mode 100644 doc/source/ext/sphinx_rtd_theme/Gemfile
delete mode 100644 doc/source/ext/sphinx_rtd_theme/Gemfile.lock
delete mode 100644 doc/source/ext/sphinx_rtd_theme/Gruntfile.js
delete mode 100644 doc/source/ext/sphinx_rtd_theme/LICENSE
delete mode 100644 doc/source/ext/sphinx_rtd_theme/MANIFEST.in
delete mode 100644 doc/source/ext/sphinx_rtd_theme/README.rst
delete mode 100644 doc/source/ext/sphinx_rtd_theme/bower.json
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/Makefile
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/__init__.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/conf.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/demo.rst
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/index.rst
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/list.rst
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/static/yi_jing_01_chien.jpg
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/__init__.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/test.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/package.json
delete mode 100644 doc/source/ext/sphinx_rtd_theme/requirements.txt
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_badge.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_badge_fa.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_breadcrumbs.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_font_awesome_compatability.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_layout.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_mathjax.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_rst.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/_theme_variables.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/badge_only.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sass/theme.sass
delete mode 100644 doc/source/ext/sphinx_rtd_theme/screen_desktop.png
delete mode 100644 doc/source/ext/sphinx_rtd_theme/screen_mobile.png
delete mode 100644 doc/source/ext/sphinx_rtd_theme/setup.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/__init__.py
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/breadcrumbs.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/footer.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout_old.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/search.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/searchbox.html
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/badge_only.css
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/theme.css
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/FontAwesome.otf
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.eot
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.svg
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.ttf
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.woff
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/js/theme.js
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/theme.conf
delete mode 100644 doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/versions.html
delete mode 100644 doc/source/launcher.rst
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index c6c5f55c2f..7b79a30a63 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -15,7 +15,7 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- name: Set up Python 3.x
uses: actions/setup-python@v4
with:
@@ -36,7 +36,7 @@ jobs:
os: [ubuntu-latest, macOs-latest]
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
with:
@@ -72,7 +72,7 @@ jobs:
target: testapps-service_library-aar
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- name: Build python-for-android docker image
run: |
docker build --tag=kivy/python-for-android .
@@ -121,7 +121,7 @@ jobs:
ANDROID_NDK_HOME: ${HOME}/.android/android-ndk
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- name: Install python-for-android
run: |
source ci/osx_ci.sh
@@ -170,7 +170,7 @@ jobs:
REBUILD_UPDATED_RECIPES_EXTRA_ARGS: --arch=${{ matrix.android_arch }}
steps:
- name: Checkout python-for-android (all-history)
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
with:
fetch-depth: 0
# helps with GitHub runner getting out of space
@@ -206,7 +206,7 @@ jobs:
REBUILD_UPDATED_RECIPES_EXTRA_ARGS: --arch=${{ matrix.android_arch }}
steps:
- name: Checkout python-for-android (all-history)
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Install python-for-android
@@ -238,3 +238,15 @@ jobs:
uses: AndreMiras/coveralls-python-action@develop
with:
parallel-finished: true
+
+ documentation:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+ - name: Requirements
+ run: python3 -m pip install .[docs]
+ - name: Check links
+ run: sphinx-build -b linkcheck doc/source doc/build
+ - name: Generate documentation
+ run: sphinx-build doc/source doc/build
+
diff --git a/doc/source/apis.rst b/doc/source/apis.rst
index a9f66ec933..7d94b74460 100644
--- a/doc/source/apis.rst
+++ b/doc/source/apis.rst
@@ -48,7 +48,7 @@ Requires `Permission.WRITE_EXTERNAL_STORAGE` to read and write to.
Read more on all the different storage types and what to use them for
in the Android documentation:
-https://developer.android.com/training/data-storage/files
+https://developer.android.com/training/data-storage
A note on permissions
~~~~~~~~~~~~~~~~~~~~~
@@ -148,7 +148,7 @@ Observing Activity result
.. module:: android.activity
-The default PythonActivity has a observer pattern for `onActivityResult <http://developer.android.com/reference/android/app/Activity.html#onActivityResult(int, int, android.content.Intent)>`_ and `onNewIntent <http://developer.android.com/reference/android/app/Activity.html#onNewIntent(android.content.Intent)>`_.
+The default PythonActivity has a observer pattern for `onActivityResult <https://developer.android.com/reference/android/app/Activity#onActivityResult(int, int, android.content.Intent)>`_ and `onNewIntent <https://developer.android.com/reference/android/app/Activity.html#onNewIntent(android.content.Intent)>`_.
.. function:: bind(eventname=callback, ...)
@@ -193,8 +193,6 @@ Example::
Activity lifecycle handling
~~~~~~~~~~~~~~~~~~~~~~~~~~~
-.. module:: android.activity
-
The Android ``Application`` class provides the `ActivityLifecycleCallbacks
<https://developer.android.com/reference/android/app/Application.ActivityLifecycleCallbacks>`_
interface where callbacks can be registered corresponding to `activity
@@ -241,7 +239,7 @@ Receiving Broadcast message
.. module:: android.broadcast
Implementation of the android `BroadcastReceiver
-<http://developer.android.com/reference/android/content/BroadcastReceiver.html>`_.
+<https://developer.android.com/reference/android/content/BroadcastReceiver.html>`_.
You can specify the callback that will receive the broadcast event, and actions
or categories filters.
@@ -308,7 +306,7 @@ Runnable
.. module:: android.runnable
:class:`Runnable` is a wrapper around the Java `Runnable
-<http://developer.android.com/reference/java/lang/Runnable.html>`_ class. This
+<https://developer.android.com/reference/java/lang/Runnable.html>`_ class. This
class can be used to schedule a call of a Python function into the
`PythonActivity` thread.
@@ -443,5 +441,5 @@ sometimes a little more involved, particularly if making Java classes
from Python code), but it's not Pythonic and it's not short. These are
problems that Plyer, explained below, attempts to address.
-You can check the `Pyjnius documentation <https://pyjnius.readthedocs.io/en/stable/>`_ for further details.
+You can check the `Pyjnius documentation <https://pyjnius.readthedocs.io/en/latest/>`_ for further details.
diff --git a/doc/source/buildoptions.rst b/doc/source/buildoptions.rst
index 8ad067b205..44bae679c1 100644
--- a/doc/source/buildoptions.rst
+++ b/doc/source/buildoptions.rst
@@ -62,7 +62,7 @@ options (this list may not be exhaustive):
Since Android ignores ``android:screenOrientation`` when in multi-window mode
(Which is the default on Android 12+), this option will also set the window orientation hints
for the SDL bootstrap. If multiple orientations are given,
-``android:screenOrientation`` will be set to ``unspecified``.
+ ``android:screenOrientation`` will be set to ``unspecified``.
- ``--manifest-orientation``: The orientation that will be set for the ``android:screenOrientation``
attribute of the activity in the ``AndroidManifest.xml`` file. If not set, the value
will be synthesized from the ``--orientation`` option.
diff --git a/doc/source/conf.py b/doc/source/conf.py
index 32f7d3055f..e4ea304fde 100644
--- a/doc/source/conf.py
+++ b/doc/source/conf.py
@@ -72,7 +72,7 @@
#
# This is also used if you do content translation via gettext catalogs.
# Usually you set "language" from the command line for these cases.
-language = None
+language = 'en'
# There are two options for replacing |today|: either, you set today to some
# non-false value, then it is used:
@@ -293,4 +293,14 @@
# Example configuration for intersphinx: refer to the Python standard library.
-intersphinx_mapping = {'https://docs.python.org/': None}
+intersphinx_mapping = {'python': ('https://docs.python.org/3', None)}
+
+# Ignore some troublesome links that are actually fine.
+linkcheck_ignore = [
+ # Special characters in URL seems to confuse link-checker.
+ r"https://developer.android.com/reference/android/app/Activity#onActivity.*",
+
+ # GitHub parses anchor tags differently to pure HTML
+ r"https://github.com/kivy/python-for-android/blob.*",
+ ]
+
diff --git a/doc/source/ext/sphinx_rtd_theme/.gitignore b/doc/source/ext/sphinx_rtd_theme/.gitignore
deleted file mode 100644
index 3076d7d8e9..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/.gitignore
+++ /dev/null
@@ -1,13 +0,0 @@
-*.pyc
-*.egg-info
-*.egg
-*build/
-.tox
-.coverage
-*.DS_Store
-*.sass-cache
-.ruby-version
-dist/
-bower_components/
-node_modules/
-npm-debug.log
diff --git a/doc/source/ext/sphinx_rtd_theme/Gemfile b/doc/source/ext/sphinx_rtd_theme/Gemfile
deleted file mode 100644
index 9f2165c761..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/Gemfile
+++ /dev/null
@@ -1,4 +0,0 @@
-# A sample Gemfile
-source "https://rubygems.org"
-
-gem "compass"
diff --git a/doc/source/ext/sphinx_rtd_theme/Gemfile.lock b/doc/source/ext/sphinx_rtd_theme/Gemfile.lock
deleted file mode 100644
index 5e9db1660a..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/Gemfile.lock
+++ /dev/null
@@ -1,16 +0,0 @@
-GEM
- remote: https://rubygems.org/
- specs:
- chunky_png (1.2.9)
- compass (0.12.2)
- chunky_png (~> 1.2)
- fssm (>= 0.2.7)
- sass (~> 3.1)
- fssm (0.2.10)
- sass (3.2.12)
-
-PLATFORMS
- ruby
-
-DEPENDENCIES
- compass
diff --git a/doc/source/ext/sphinx_rtd_theme/Gruntfile.js b/doc/source/ext/sphinx_rtd_theme/Gruntfile.js
deleted file mode 100644
index 8b2a207134..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/Gruntfile.js
+++ /dev/null
@@ -1,105 +0,0 @@
-module.exports = function(grunt) {
-
- // load all grunt tasks
- require('matchdep').filterDev('grunt-*').forEach(grunt.loadNpmTasks);
-
- grunt.initConfig({
- open : {
- dev: {
- path: 'http://localhost:1919'
- }
- },
-
- connect: {
- server: {
- options: {
- port: 1919,
- base: 'demo_docs/build',
- livereload: true
- }
- }
- },
- copy: {
- fonts: {
- files: [
- // includes files within path
- {expand: true, flatten: true, src: ['bower_components/font-awesome/fonts/*'], dest: 'sphinx_rtd_theme/static/fonts/', filter: 'isFile'}
- ]
- }
- },
-
- sass: {
- dev: {
- options: {
- style: 'expanded',
- loadPath: ['bower_components/bourbon/dist', 'bower_components/neat/app/assets/stylesheets', 'bower_components/font-awesome/scss', 'bower_components/wyrm/sass']
- },
- files: [{
- expand: true,
- cwd: 'sass',
- src: ['*.sass'],
- dest: 'sphinx_rtd_theme/static/css',
- ext: '.css'
- }]
- },
- build: {
- options: {
- style: 'compressed',
- loadPath: ['bower_components/bourbon/dist', 'bower_components/neat/app/assets/stylesheets', 'bower_components/font-awesome/scss', 'bower_components/wyrm/sass']
- },
- files: [{
- expand: true,
- cwd: 'sass',
- src: ['*.sass'],
- dest: 'sphinx_rtd_theme/static/css',
- ext: '.css'
- }]
- }
- },
-
- exec: {
- bower_update: {
- cmd: 'bower update'
- },
- build_sphinx: {
- cmd: 'sphinx-build demo_docs/source demo_docs/build'
- }
- },
- clean: {
- build: ["demo_docs/build"],
- fonts: ["sphinx_rtd_theme/static/fonts"]
- },
-
- watch: {
- /* Compile sass changes into theme directory */
- sass: {
- files: ['sass/*.sass', 'bower_components/**/*.sass'],
- tasks: ['sass:dev']
- },
- /* Changes in theme dir rebuild sphinx */
- sphinx: {
- files: ['sphinx_rtd_theme/**/*', 'demo_docs/**/*.rst', 'demo_docs/**/*.py'],
- tasks: ['clean:build','exec:build_sphinx']
- },
- /* live-reload the demo_docs if sphinx re-builds */
- livereload: {
- files: ['demo_docs/build/**/*'],
- options: { livereload: true }
- }
- }
-
- });
-
- grunt.loadNpmTasks('grunt-exec');
- grunt.loadNpmTasks('grunt-contrib-connect');
- grunt.loadNpmTasks('grunt-contrib-watch');
- grunt.loadNpmTasks('grunt-contrib-sass');
- grunt.loadNpmTasks('grunt-contrib-clean');
- grunt.loadNpmTasks('grunt-contrib-copy');
- grunt.loadNpmTasks('grunt-open');
-
- grunt.registerTask('fonts', ['clean:fonts','copy:fonts']);
- grunt.registerTask('default', ['exec:bower_update','clean:build','sass:dev','exec:build_sphinx','connect','open','watch']);
- grunt.registerTask('build', ['exec:bower_update','clean:build','sass:build','exec:build_sphinx']);
-}
-
diff --git a/doc/source/ext/sphinx_rtd_theme/LICENSE b/doc/source/ext/sphinx_rtd_theme/LICENSE
deleted file mode 100644
index 921f073883..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/LICENSE
+++ /dev/null
@@ -1,20 +0,0 @@
-The MIT License (MIT)
-
-Copyright (c) 2013 Dave Snider
-
-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/doc/source/ext/sphinx_rtd_theme/MANIFEST.in b/doc/source/ext/sphinx_rtd_theme/MANIFEST.in
deleted file mode 100644
index eaca81d850..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/MANIFEST.in
+++ /dev/null
@@ -1,10 +0,0 @@
-include *.txt
-include LICENSE
-recursive-include sphinx_rtd_theme *.conf
-recursive-include sphinx_rtd_theme *.css
-recursive-include sphinx_rtd_theme *.eot
-recursive-include sphinx_rtd_theme *.html
-recursive-include sphinx_rtd_theme *.js
-recursive-include sphinx_rtd_theme *.svg
-recursive-include sphinx_rtd_theme *.ttf
-recursive-include sphinx_rtd_theme *.woff
diff --git a/doc/source/ext/sphinx_rtd_theme/README.rst b/doc/source/ext/sphinx_rtd_theme/README.rst
deleted file mode 100644
index d06078fdab..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/README.rst
+++ /dev/null
@@ -1,176 +0,0 @@
-.. _readthedocs.org: http://www.readthedocs.org
-.. _bower: http://www.bower.io
-.. _sphinx: http://www.sphinx-doc.org
-.. _compass: http://www.compass-style.org
-.. _sass: http://www.sass-lang.com
-.. _wyrm: http://www.github.com/snide/wyrm/
-.. _grunt: http://www.gruntjs.com
-.. _node: http://www.nodejs.com
-.. _demo: http://docs.readthedocs.org
-.. _hidden: http://sphinx-doc.org/markup/toctree.html
-
-**************************
-Read the Docs Sphinx Theme
-**************************
-
-View a working demo_ over on readthedocs.org_.
-
-This is a mobile-friendly sphinx_ theme I made for readthedocs.org_. It's
-currently in development there and includes some rtd variable checks that can be ignored
-if you're just trying to use it on your project outside of that site.
-
-**This repo also exists as a submodule within the readthedocs itself**, so please make your edits to
-the SASS files here, rather than the .css files on RTD.
-
-.. image:: screen_mobile.png
- :width: 100%
-Installation
-============
-
-Via package
------------
-
-Download the package or add it to your ``requirements.txt`` file:
-
-.. code:: bash
-
- $ pip install sphinx_rtd_theme
-
-In your ``conf.py`` file:
-
-.. code:: python
-
- import sphinx_rtd_theme
-
- html_theme = "sphinx_rtd_theme"
-
- html_theme_path = [sphinx_rtd_theme.get_html_theme_path()]
-
-Via git or download
--------------------
-
-Symlink or subtree the ``sphinx_rtd_theme/sphinx_rtd_theme`` repository into your documentation at
-``docs/_themes/sphinx_rtd_theme`` then add the following two settings to your Sphinx
-conf.py file:
-
-.. code:: python
-
- html_theme = "sphinx_rtd_theme"
- html_theme_path = ["_themes", ]
-
-How the Table of Contents builds
-================================
-
-Currently the left menu will build based upon any ``toctree(s)`` defined in your index.rst file.
-It outputs 2 levels of depth, which should give your visitors a high level of access to your
-docs. If no toctrees are set the theme reverts to sphinx's usual local toctree.
-
-It's important to note that if you don't follow the same styling for your rST headers across
-your documents, the toctree will misbuild, and the resulting menu might not show the correct
-depth when it renders.
-
-Also note that the table of contents is set with ``includehidden=true``. This allows you
-to set a hidden toc in your index file with the hidden_ property that will allow you
-to build a toc without it rendering in your index.
-
-By default, the navigation will "stick" to the screen as you scroll. However if your toc
-is vertically too large, it revert to static positioning. To disable the sticky nav
-alltogether change the setting in ``conf.py``.
-
-Contributing or modifying the theme
-===================================
-
-The sphinx_rtd_theme is primarily a sass_ project that requires a few other sass libraries. I'm
-using bower_ to manage these dependencies and sass_ to build the css. The good news is
-I have a very nice set of grunt_ operations that will not only load these dependecies, but watch
-for changes, rebuild the sphinx demo docs and build a distributable version of the theme.
-The bad news is this means you'll need to set up your environment similar to that
-of a front-end developer (vs. that of a python developer). That means installing node and ruby.
-
-Set up your environment
------------------------
-
-1. Install sphinx_ into a virtual environment.
-
-.. code::
-
- pip install sphinx
-
-2. Install sass
-
-.. code::
-
- gem install sass
-
-2. Install node, bower and grunt.
-
-.. code::
-
- // Install node
- brew install node
-
- // Install bower and grunt
- npm install -g bower grunt-cli
-
- // Now that everything is installed, let's install the theme dependecies.
- npm install
-
-Now that our environment is set up, make sure you're in your virtual environment, go to
-this repository in your terminal and run grunt:
-
-.. code::
-
- grunt
-
-This default task will do the following **very cool things that make it worth the trouble**.
-
-1. It'll install and update any bower dependencies.
-2. It'll run sphinx and build new docs.
-3. It'll watch for changes to the sass files and build css from the changes.
-4. It'll rebuild the sphinx docs anytime it notices a change to .rst, .html, .js
- or .css files.
-
-
-Before you create an issue
---------------------------
-
-I don't have a lot of time to maintain this project due to other responsibilities.
-I know there are a lot of Python engineers out there that can't code sass / css and
-are unable to submit pull requests. That said, submitting random style bugs without
-at least providing sample documentation that replicates your problem is a good
-way for me to ignore your request. RST unfortunately can spit out a lot of things
-in a lot of ways. I don't have time to research your problem for you, but I do
-have time to fix the actual styling issue if you can replicate the problem for me.
-
-
-Before you send a Pull Request
-------------------------------
-
-When you're done with your edits, you can run ``grunt build`` to clean out the old
-files and rebuild a new distribution, compressing the css and cleaning out
-extraneous files. Please do this before you send in a PR.
-
-Using this theme locally, then building on Read the Docs?
-==========================================================
-
-Currently if you import sphinx_rtd_theme in your local sphinx build, then pass
-that same config to Read the Docs, it will fail, since RTD gets confused. If
-you want to run this theme locally and then also have it build on RTD, then
-you can add something like this to your config. Thanks to Daniel Oaks for this.
-
-.. code:: python
-
- # on_rtd is whether we are on readthedocs.org, this line of code grabbed from docs.readthedocs.org
- on_rtd = os.environ.get('READTHEDOCS', None) == 'True'
-
- if not on_rtd: # only import and set the theme if we're building docs locally
- import sphinx_rtd_theme
- html_theme = 'sphinx_rtd_theme'
- html_theme_path = [sphinx_rtd_theme.get_html_theme_path()]
-
- # otherwise, readthedocs.org uses their theme by default, so no need to specify it
-
-TODO
-====
-* Separate some sass variables at the theme level so you can overwrite some basic colors.
-
diff --git a/doc/source/ext/sphinx_rtd_theme/bower.json b/doc/source/ext/sphinx_rtd_theme/bower.json
deleted file mode 100644
index c068a63e9b..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/bower.json
+++ /dev/null
@@ -1,33 +0,0 @@
-{
- "name": "sphinx-rtd-theme",
- "version": "1.0",
- "homepage": "https://github.com/snide/wyrm",
- "authors": [
- "Dave Snider <dave.snider@gmail.com>"
- ],
- "description": "Sphinx theme for readthedocs.org.",
- "license": "MIT",
- "main": [
- "dist/**"
- ],
- "ignore": [
- "docs",
- "demo_docs",
- ".gitignore",
- ".DS_Store",
- ".sass-cache*",
- ".bowerrc",
- "bower.json",
- "package.json",
- "Gruntfile.js",
- "node_modules",
- "bower_components",
- "test",
- "tests",
- "src"
- ],
- "dependencies": {
- "wyrm": "~0.0.x"
- }
-}
-
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/Makefile b/doc/source/ext/sphinx_rtd_theme/demo_docs/Makefile
deleted file mode 100644
index 8c907fd422..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/Makefile
+++ /dev/null
@@ -1,153 +0,0 @@
-# Makefile for Sphinx documentation
-#
-
-# You can set these variables from the command line.
-SPHINXOPTS =
-SPHINXBUILD = sphinx-build
-PAPER =
-BUILDDIR = build
-
-# Internal variables.
-PAPEROPT_a4 = -D latex_paper_size=a4
-PAPEROPT_letter = -D latex_paper_size=letter
-ALLSPHINXOPTS = -d $(BUILDDIR)/doctrees $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) source
-# the i18n builder cannot share the environment and doctrees with the others
-I18NSPHINXOPTS = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) source
-
-.PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext
-
-help:
- @echo "Please use \`make <target>' where <target> is one of"
- @echo " html to make standalone HTML files"
- @echo " dirhtml to make HTML files named index.html in directories"
- @echo " singlehtml to make a single large HTML file"
- @echo " pickle to make pickle files"
- @echo " json to make JSON files"
- @echo " htmlhelp to make HTML files and a HTML help project"
- @echo " qthelp to make HTML files and a qthelp project"
- @echo " devhelp to make HTML files and a Devhelp project"
- @echo " epub to make an epub"
- @echo " latex to make LaTeX files, you can set PAPER=a4 or PAPER=letter"
- @echo " latexpdf to make LaTeX files and run them through pdflatex"
- @echo " text to make text files"
- @echo " man to make manual pages"
- @echo " texinfo to make Texinfo files"
- @echo " info to make Texinfo files and run them through makeinfo"
- @echo " gettext to make PO message catalogs"
- @echo " changes to make an overview of all changed/added/deprecated items"
- @echo " linkcheck to check all external links for integrity"
- @echo " doctest to run all doctests embedded in the documentation (if enabled)"
-
-clean:
- -rm -rf $(BUILDDIR)/*
-
-html:
- $(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html
- @echo
- @echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
-
-dirhtml:
- $(SPHINXBUILD) -b dirhtml $(ALLSPHINXOPTS) $(BUILDDIR)/dirhtml
- @echo
- @echo "Build finished. The HTML pages are in $(BUILDDIR)/dirhtml."
-
-singlehtml:
- $(SPHINXBUILD) -b singlehtml $(ALLSPHINXOPTS) $(BUILDDIR)/singlehtml
- @echo
- @echo "Build finished. The HTML page is in $(BUILDDIR)/singlehtml."
-
-pickle:
- $(SPHINXBUILD) -b pickle $(ALLSPHINXOPTS) $(BUILDDIR)/pickle
- @echo
- @echo "Build finished; now you can process the pickle files."
-
-json:
- $(SPHINXBUILD) -b json $(ALLSPHINXOPTS) $(BUILDDIR)/json
- @echo
- @echo "Build finished; now you can process the JSON files."
-
-htmlhelp:
- $(SPHINXBUILD) -b htmlhelp $(ALLSPHINXOPTS) $(BUILDDIR)/htmlhelp
- @echo
- @echo "Build finished; now you can run HTML Help Workshop with the" \
- ".hhp project file in $(BUILDDIR)/htmlhelp."
-
-qthelp:
- $(SPHINXBUILD) -b qthelp $(ALLSPHINXOPTS) $(BUILDDIR)/qthelp
- @echo
- @echo "Build finished; now you can run "qcollectiongenerator" with the" \
- ".qhcp project file in $(BUILDDIR)/qthelp, like this:"
- @echo "# qcollectiongenerator $(BUILDDIR)/qthelp/SphinxRTDthemedemo.qhcp"
- @echo "To view the help file:"
- @echo "# assistant -collectionFile $(BUILDDIR)/qthelp/SphinxRTDthemedemo.qhc"
-
-devhelp:
- $(SPHINXBUILD) -b devhelp $(ALLSPHINXOPTS) $(BUILDDIR)/devhelp
- @echo
- @echo "Build finished."
- @echo "To view the help file:"
- @echo "# mkdir -p $$HOME/.local/share/devhelp/SphinxRTDthemedemo"
- @echo "# ln -s $(BUILDDIR)/devhelp $$HOME/.local/share/devhelp/SphinxRTDthemedemo"
- @echo "# devhelp"
-
-epub:
- $(SPHINXBUILD) -b epub $(ALLSPHINXOPTS) $(BUILDDIR)/epub
- @echo
- @echo "Build finished. The epub file is in $(BUILDDIR)/epub."
-
-latex:
- $(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex
- @echo
- @echo "Build finished; the LaTeX files are in $(BUILDDIR)/latex."
- @echo "Run \`make' in that directory to run these through (pdf)latex" \
- "(use \`make latexpdf' here to do that automatically)."
-
-latexpdf:
- $(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex
- @echo "Running LaTeX files through pdflatex..."
- $(MAKE) -C $(BUILDDIR)/latex all-pdf
- @echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex."
-
-text:
- $(SPHINXBUILD) -b text $(ALLSPHINXOPTS) $(BUILDDIR)/text
- @echo
- @echo "Build finished. The text files are in $(BUILDDIR)/text."
-
-man:
- $(SPHINXBUILD) -b man $(ALLSPHINXOPTS) $(BUILDDIR)/man
- @echo
- @echo "Build finished. The manual pages are in $(BUILDDIR)/man."
-
-texinfo:
- $(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo
- @echo
- @echo "Build finished. The Texinfo files are in $(BUILDDIR)/texinfo."
- @echo "Run \`make' in that directory to run these through makeinfo" \
- "(use \`make info' here to do that automatically)."
-
-info:
- $(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo
- @echo "Running Texinfo files through makeinfo..."
- make -C $(BUILDDIR)/texinfo info
- @echo "makeinfo finished; the Info files are in $(BUILDDIR)/texinfo."
-
-gettext:
- $(SPHINXBUILD) -b gettext $(I18NSPHINXOPTS) $(BUILDDIR)/locale
- @echo
- @echo "Build finished. The message catalogs are in $(BUILDDIR)/locale."
-
-changes:
- $(SPHINXBUILD) -b changes $(ALLSPHINXOPTS) $(BUILDDIR)/changes
- @echo
- @echo "The overview file is in $(BUILDDIR)/changes."
-
-linkcheck:
- $(SPHINXBUILD) -b linkcheck $(ALLSPHINXOPTS) $(BUILDDIR)/linkcheck
- @echo
- @echo "Link check complete; look for any errors in the above output " \
- "or in $(BUILDDIR)/linkcheck/output.txt."
-
-doctest:
- $(SPHINXBUILD) -b doctest $(ALLSPHINXOPTS) $(BUILDDIR)/doctest
- @echo "Testing of doctests in the sources finished, look at the " \
- "results in $(BUILDDIR)/doctest/output.txt."
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/__init__.py b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/__init__.py
deleted file mode 100644
index e69de29bb2..0000000000
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/conf.py b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/conf.py
deleted file mode 100644
index 56c44d0a35..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/conf.py
+++ /dev/null
@@ -1,254 +0,0 @@
-# -*- coding: utf-8 -*-
-#
-# Sphinx RTD theme demo documentation build configuration file, created by
-# sphinx-quickstart on Sun Nov 3 11:56:36 2013.
-#
-# This file is execfile()d with the current directory set to its containing dir.
-#
-# Note that not all possible configuration values are present in this
-# autogenerated file.
-#
-# All configuration values have a default; values that are commented out
-# serve to show the default.
-
-import sys, os
-
-sys.path.append(os.path.abspath('.'))
-sys.path.append(os.path.abspath('./test_py_module'))
-
-# If extensions (or modules to document with autodoc) are in another directory,
-# add these directories to sys.path here. If the directory is relative to the
-# documentation root, use os.path.abspath to make it absolute, like shown here.
-#sys.path.insert(0, os.path.abspath('.'))
-
-# -- General configuration -----------------------------------------------------
-
-# If your documentation needs a minimal Sphinx version, state it here.
-#needs_sphinx = '1.0'
-
-# Add any Sphinx extension module names here, as strings. They can be extensions
-# coming with Sphinx (named 'sphinx.ext.*') or your custom ones.
-extensions = [
- 'sphinx.ext.autodoc',
- 'sphinx.ext.mathjax',
- 'sphinx.ext.viewcode',
-]
-
-# Math
-mathjax_path = "http://cdn.mathjax.org/mathjax/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML"
-
-# Add any paths that contain templates here, relative to this directory.
-templates_path = ['_templates']
-
-# The suffix of source filenames.
-source_suffix = '.rst'
-
-# The encoding of source files.
-#source_encoding = 'utf-8-sig'
-
-# The master toctree document.
-master_doc = 'index'
-
-# General information about the project.
-project = u'Sphinx RTD theme demo'
-copyright = u'2013, Dave Snider'
-
-# The version info for the project you're documenting, acts as replacement for
-# |version| and |release|, also used in various other places throughout the
-# built documents.
-#
-# The short X.Y version.
-version = '1'
-# The full version, including alpha/beta/rc tags.
-release = '1'
-
-# The language for content autogenerated by Sphinx. Refer to documentation
-# for a list of supported languages.
-#language = None
-
-# There are two options for replacing |today|: either, you set today to some
-# non-false value, then it is used:
-#today = ''
-# Else, today_fmt is used as the format for a strftime call.
-#today_fmt = '%B %d, %Y'
-
-# List of patterns, relative to source directory, that match files and
-# directories to ignore when looking for source files.
-exclude_patterns = []
-
-# The reST default role (used for this markup: `text`) to use for all documents.
-#default_role = None
-
-# If true, '()' will be appended to :func: etc. cross-reference text.
-#add_function_parentheses = True
-
-# If true, the current module name will be prepended to all description
-# unit titles (such as .. function::).
-#add_module_names = True
-
-# If true, sectionauthor and moduleauthor directives will be shown in the
-# output. They are ignored by default.
-#show_authors = False
-
-# The name of the Pygments (syntax highlighting) style to use.
-pygments_style = 'sphinx'
-
-# A list of ignored prefixes for module index sorting.
-#modindex_common_prefix = []
-
-
-# -- Options for HTML output ---------------------------------------------------
-
-# The theme to use for HTML and HTML Help pages. See the documentation for
-# a list of builtin themes.
-html_theme = 'sphinx_rtd_theme'
-
-# Theme options are theme-specific and customize the look and feel of a theme
-# further. For a list of options available for each theme, see the
-# documentation.
-html_theme_options = {
- # 'sticky_navigation' : True # Set to False to disable the sticky nav while scrolling.
-}
-
-# Add any paths that contain custom themes here, relative to this directory.
-html_theme_path = ["../.."]
-
-# The name for this set of Sphinx documents. If None, it defaults to
-# "<project> v<release> documentation".
-#html_title = None
-
-# A shorter title for the navigation bar. Default is the same as html_title.
-#html_short_title = None
-
-# The name of an image file (relative to this directory) to place at the top
-# of the sidebar.
-#html_logo = None
-
-# The name of an image file (within the static path) to use as favicon of the
-# docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32
-# pixels large.
-#html_favicon = None
-
-# Add any paths that contain custom static files (such as style sheets) here,
-# relative to this directory. They are copied after the builtin static files,
-# so a file named "default.css" will overwrite the builtin "default.css".
-#html_static_path = ['_static']
-
-# If not '', a 'Last updated on:' timestamp is inserted at every page bottom,
-# using the given strftime format.
-#html_last_updated_fmt = '%b %d, %Y'
-
-# If true, SmartyPants will be used to convert quotes and dashes to
-# typographically correct entities.
-#html_use_smartypants = True
-
-# Custom sidebar templates, maps document names to template names.
-#html_sidebars = {}
-
-# Additional templates that should be rendered to pages, maps page names to
-# template names.
-#html_additional_pages = {}
-
-# If false, no module index is generated.
-#html_domain_indices = True
-
-# If false, no index is generated.
-#html_use_index = True
-
-# If true, the index is split into individual pages for each letter.
-#html_split_index = False
-
-# If true, links to the reST sources are added to the pages.
-html_show_sourcelink = True
-
-# If true, "Created using Sphinx" is shown in the HTML footer. Default is True.
-#html_show_sphinx = True
-
-# If true, "(C) Copyright ..." is shown in the HTML footer. Default is True.
-#html_show_copyright = True
-
-# If true, an OpenSearch description file will be output, and all pages will
-# contain a <link> tag referring to it. The value of this option must be the
-# base URL from which the finished HTML is served.
-#html_use_opensearch = ''
-
-# This is the file name suffix for HTML files (e.g. ".xhtml").
-#html_file_suffix = None
-
-# Output file base name for HTML help builder.
-htmlhelp_basename = 'SphinxRTDthemedemodoc'
-
-
-# -- Options for LaTeX output --------------------------------------------------
-
-latex_elements = {
-# The paper size ('letterpaper' or 'a4paper').
-#'papersize': 'letterpaper',
-
-# The font size ('10pt', '11pt' or '12pt').
-#'pointsize': '10pt',
-
-# Additional stuff for the LaTeX preamble.
-#'preamble': '',
-}
-
-# Grouping the document tree into LaTeX files. List of tuples
-# (source start file, target name, title, author, documentclass [howto/manual]).
-latex_documents = [
- ('index', 'SphinxRTDthemedemo.tex', u'Sphinx RTD theme demo Documentation',
- u'Dave Snider', 'manual'),
-]
-
-# The name of an image file (relative to this directory) to place at the top of
-# the title page.
-#latex_logo = None
-
-# For "manual" documents, if this is true, then toplevel headings are parts,
-# not chapters.
-#latex_use_parts = False
-
-# If true, show page references after internal links.
-#latex_show_pagerefs = False
-
-# If true, show URL addresses after external links.
-#latex_show_urls = False
-
-# Documents to append as an appendix to all manuals.
-#latex_appendices = []
-
-# If false, no module index is generated.
-#latex_domain_indices = True
-
-
-# -- Options for manual page output --------------------------------------------
-
-# One entry per manual page. List of tuples
-# (source start file, name, description, authors, manual section).
-man_pages = [
- ('index', 'sphinxrtdthemedemo', u'Sphinx RTD theme demo Documentation',
- [u'Dave Snider'], 1)
-]
-
-# If true, show URL addresses after external links.
-#man_show_urls = False
-
-
-# -- Options for Texinfo output ------------------------------------------------
-
-# Grouping the document tree into Texinfo files. List of tuples
-# (source start file, target name, title, author,
-# dir menu entry, description, category)
-texinfo_documents = [
- ('index', 'SphinxRTDthemedemo', u'Sphinx RTD theme demo Documentation',
- u'Dave Snider', 'SphinxRTDthemedemo', 'One line description of project.',
- 'Miscellaneous'),
-]
-
-# Documents to append as an appendix to all manuals.
-#texinfo_appendices = []
-
-# If false, no module index is generated.
-#texinfo_domain_indices = True
-
-# How to display URL addresses: 'footnote', 'no', or 'inline'.
-#texinfo_show_urls = 'footnote'
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/demo.rst b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/demo.rst
deleted file mode 100644
index 1af45c81de..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/demo.rst
+++ /dev/null
@@ -1,577 +0,0 @@
-.. This is a comment. Note how any initial comments are moved by
- transforms to after the document title, subtitle, and docinfo.
-
-================================
- reStructuredText Demonstration
-================================
-
-.. Above is the document title, and below is the subtitle.
- They are transformed from section titles after parsing.
-
---------------------------------
- Examples of Syntax Constructs
---------------------------------
-
-.. bibliographic fields (which also require a transform):
-
-:Author: David Goodger
-:Address: 123 Example Street
- Example, EX Canada
- A1B 2C3
-:Contact: docutils-develop@lists.sourceforge.net
-:Authors: Me; Myself; I
-:organization: humankind
-:date: $Date: 2012-01-03 19:23:53 +0000 (Tue, 03 Jan 2012) $
-:status: This is a "work in progress"
-:revision: $Revision: 7302 $
-:version: 1
-:copyright: This document has been placed in the public domain. You
- may do with it as you wish. You may copy, modify,
- redistribute, reattribute, sell, buy, rent, lease,
- destroy, or improve it, quote it at length, excerpt,
- incorporate, collate, fold, staple, or mutilate it, or do
- anything else to it that your or anyone else's heart
- desires.
-:field name: This is a generic bibliographic field.
-:field name 2:
- Generic bibliographic fields may contain multiple body elements.
-
- Like this.
-
-:Dedication:
-
- For Docutils users & co-developers.
-
-:abstract:
-
- This document is a demonstration of the reStructuredText markup
- language, containing examples of all basic reStructuredText
- constructs and many advanced constructs.
-
-.. meta::
- :keywords: reStructuredText, demonstration, demo, parser
- :description lang=en: A demonstration of the reStructuredText
- markup language, containing examples of all basic
- constructs and many advanced constructs.
-
-.. contents:: Table of Contents
-.. section-numbering::
-
-
-Structural Elements
-===================
-
-Section Title
--------------
-
-That's it, the text just above this line.
-
-Transitions
------------
-
-Here's a transition:
-
----------
-
-It divides the section.
-
-Body Elements
-=============
-
-Paragraphs
-----------
-
-A paragraph.
-
-Inline Markup
-`````````````
-
-Paragraphs contain text and may contain inline markup: *emphasis*,
-**strong emphasis**, ``inline literals``, standalone hyperlinks
-(http://www.python.org), external hyperlinks (Python_), internal
-cross-references (example_), external hyperlinks with embedded URIs
-(`Python web site <http://www.python.org>`__), footnote references
-(manually numbered [1]_, anonymous auto-numbered [#]_, labeled
-auto-numbered [#label]_, or symbolic [*]_), citation references
-([CIT2002]_), substitution references (|example|), and _`inline
-hyperlink targets` (see Targets_ below for a reference back to here).
-Character-level inline markup is also possible (although exceedingly
-ugly!) in *re*\ ``Structured``\ *Text*. Problems are indicated by
-|problematic| text (generated by processing errors; this one is
-intentional).
-
-The default role for interpreted text is `Title Reference`. Here are
-some explicit interpreted text roles: a PEP reference (:PEP:`287`); an
-RFC reference (:RFC:`2822`); a :sub:`subscript`; a :sup:`superscript`;
-and explicit roles for :emphasis:`standard` :strong:`inline`
-:literal:`markup`.
-
-.. DO NOT RE-WRAP THE FOLLOWING PARAGRAPH!
-
-Let's test wrapping and whitespace significance in inline literals:
-``This is an example of --inline-literal --text, --including some--
-strangely--hyphenated-words. Adjust-the-width-of-your-browser-window
-to see how the text is wrapped. -- ---- -------- Now note the
-spacing between the words of this sentence (words
-should be grouped in pairs).``
-
-If the ``--pep-references`` option was supplied, there should be a
-live link to PEP 258 here.
-
-Bullet Lists
-------------
-
-- A bullet list
-
- + Nested bullet list.
- + Nested item 2.
-
-- Item 2.
-
- Paragraph 2 of item 2.
-
- * Nested bullet list.
- * Nested item 2.
-
- - Third level.
- - Item 2.
-
- * Nested item 3.
-
-Enumerated Lists
-----------------
-
-1. Arabic numerals.
-
- a) lower alpha)
-
- (i) (lower roman)
-
- A. upper alpha.
-
- I) upper roman)
-
-2. Lists that don't start at 1:
-
- 3. Three
-
- 4. Four
-
- C. C
-
- D. D
-
- iii. iii
-
- iv. iv
-
-#. List items may also be auto-enumerated.
-
-Definition Lists
-----------------
-
-Term
- Definition
-Term : classifier
- Definition paragraph 1.
-
- Definition paragraph 2.
-Term
- Definition
-
-Field Lists
------------
-
-:what: Field lists map field names to field bodies, like database
- records. They are often part of an extension syntax. They are
- an unambiguous variant of RFC 2822 fields.
-
-:how arg1 arg2:
-
- The field marker is a colon, the field name, and a colon.
-
- The field body may contain one or more body elements, indented
- relative to the field marker.
-
-Option Lists
-------------
-
-For listing command-line options:
-
--a command-line option "a"
--b file options can have arguments
- and long descriptions
---long options can be long also
---input=file long options can also have
- arguments
-
---very-long-option
- The description can also start on the next line.
-
- The description may contain multiple body elements,
- regardless of where it starts.
-
--x, -y, -z Multiple options are an "option group".
--v, --verbose Commonly-seen: short & long options.
--1 file, --one=file, --two file
- Multiple options with arguments.
-/V DOS/VMS-style options too
-
-There must be at least two spaces between the option and the
-description.
-
-Literal Blocks
---------------
-
-Literal blocks are indicated with a double-colon ("::") at the end of
-the preceding paragraph (over there ``-->``). They can be indented::
-
- if literal_block:
- text = 'is left as-is'
- spaces_and_linebreaks = 'are preserved'
- markup_processing = None
-
-Or they can be quoted without indentation::
-
->> Great idea!
->
-> Why didn't I think of that?
-
-Line Blocks
------------
-
-| This is a line block. It ends with a blank line.
-| Each new line begins with a vertical bar ("|").
-| Line breaks and initial indents are preserved.
-| Continuation lines are wrapped portions of long lines;
- they begin with a space in place of the vertical bar.
-| The left edge of a continuation line need not be aligned with
- the left edge of the text above it.
-
-| This is a second line block.
-|
-| Blank lines are permitted internally, but they must begin with a "|".
-
-Take it away, Eric the Orchestra Leader!
-
- | A one, two, a one two three four
- |
- | Half a bee, philosophically,
- | must, *ipso facto*, half not be.
- | But half the bee has got to be,
- | *vis a vis* its entity. D'you see?
- |
- | But can a bee be said to be
- | or not to be an entire bee,
- | when half the bee is not a bee,
- | due to some ancient injury?
- |
- | Singing...
-
-Block Quotes
-------------
-
-Block quotes consist of indented body elements:
-
- My theory by A. Elk. Brackets Miss, brackets. This theory goes
- as follows and begins now. All brontosauruses are thin at one
- end, much much thicker in the middle and then thin again at the
- far end. That is my theory, it is mine, and belongs to me and I
- own it, and what it is too.
-
- -- Anne Elk (Miss)
-
-Doctest Blocks
---------------
-
->>> print 'Python-specific usage examples; begun with ">>>"'
-Python-specific usage examples; begun with ">>>"
->>> print '(cut and pasted from interactive Python sessions)'
-(cut and pasted from interactive Python sessions)
-
-Tables
-------
-
-Here's a grid table followed by a simple table:
-
-+------------------------+------------+----------+----------+
-| Header row, column 1 | Header 2 | Header 3 | Header 4 |
-| (header rows optional) | | | |
-+========================+============+==========+==========+
-| body row 1, column 1 | column 2 | column 3 | column 4 |
-+------------------------+------------+----------+----------+
-| body row 2 | Cells may span columns. |
-+------------------------+------------+---------------------+
-| body row 3 | Cells may | - Table cells |
-+------------------------+ span rows. | - contain |
-| body row 4 | | - body elements. |
-+------------------------+------------+----------+----------+
-| body row 5 | Cells may also be | |
-| | empty: ``-->`` | |
-+------------------------+-----------------------+----------+
-
-===== ===== ======
- Inputs Output
------------- ------
- A B A or B
-===== ===== ======
-False False False
-True False True
-False True True
-True True True
-===== ===== ======
-
-Footnotes
----------
-
-.. [1] A footnote contains body elements, consistently indented by at
- least 3 spaces.
-
- This is the footnote's second paragraph.
-
-.. [#label] Footnotes may be numbered, either manually (as in [1]_) or
- automatically using a "#"-prefixed label. This footnote has a
- label so it can be referred to from multiple places, both as a
- footnote reference ([#label]_) and as a hyperlink reference
- (label_).
-
-.. [#] This footnote is numbered automatically and anonymously using a
- label of "#" only.
-
-.. [*] Footnotes may also use symbols, specified with a "*" label.
- Here's a reference to the next footnote: [*]_.
-
-.. [*] This footnote shows the next symbol in the sequence.
-
-.. [4] Here's an unreferenced footnote, with a reference to a
- nonexistent footnote: [5]_.
-
-Citations
----------
-
-.. [CIT2002] Citations are text-labeled footnotes. They may be
- rendered separately and differently from footnotes.
-
-Here's a reference to the above, [CIT2002]_, and a [nonexistent]_
-citation.
-
-Targets
--------
-
-.. _example:
-
-This paragraph is pointed to by the explicit "example" target. A
-reference can be found under `Inline Markup`_, above. `Inline
-hyperlink targets`_ are also possible.
-
-Section headers are implicit targets, referred to by name. See
-Targets_, which is a subsection of `Body Elements`_.
-
-Explicit external targets are interpolated into references such as
-"Python_".
-
-.. _Python: http://www.python.org/
-
-Targets may be indirect and anonymous. Thus `this phrase`__ may also
-refer to the Targets_ section.
-
-__ Targets_
-
-Here's a `hyperlink reference without a target`_, which generates an
-error.
-
-Duplicate Target Names
-``````````````````````
-
-Duplicate names in section headers or other implicit targets will
-generate "info" (level-1) system messages. Duplicate names in
-explicit targets will generate "warning" (level-2) system messages.
-
-Duplicate Target Names
-``````````````````````
-
-Since there are two "Duplicate Target Names" section headers, we
-cannot uniquely refer to either of them by name. If we try to (like
-this: `Duplicate Target Names`_), an error is generated.
-
-Directives
-----------
-
-.. contents:: :local:
-
-These are just a sample of the many reStructuredText Directives. For
-others, please see
-http://docutils.sourceforge.net/docs/ref/rst/directives.html.
-
-Document Parts
-``````````````
-
-An example of the "contents" directive can be seen above this section
-(a local, untitled table of contents_) and at the beginning of the
-document (a document-wide `table of contents`_).
-
-Images
-``````
-
-An image directive (also clickable -- a hyperlink reference):
-
-.. image:: images/title.png
- :target: directives_
-
-A figure directive:
-
-.. figure:: images/title.png
- :alt: reStructuredText, the markup syntax
-
- A figure is an image with a caption and/or a legend:
-
- +------------+-----------------------------------------------+
- | re | Revised, revisited, based on 're' module. |
- +------------+-----------------------------------------------+
- | Structured | Structure-enhanced text, structuredtext. |
- +------------+-----------------------------------------------+
- | Text | Well it is, isn't it? |
- +------------+-----------------------------------------------+
-
- This paragraph is also part of the legend.
-
-A figure directive with center alignment
-
-.. figure:: images/title.png
- :align: center
- :width: 300
-
-Admonitions
-```````````
-
-.. Attention:: Directives at large.
-
-.. Caution::
-
- Don't take any wooden nickels.
-
-.. DANGER:: Mad scientist at work!
-
-.. Error:: Does not compute.
-
-.. Hint:: It's bigger than a bread box.
-
-.. Important::
- - Wash behind your ears.
- - Clean up your room.
- - Call your mother.
- - Back up your data.
-
-.. Note:: This is a note.
-
-.. Tip:: 15% if the service is good.
-
-.. WARNING:: Strong prose may provoke extreme mental exertion.
- Reader discretion is strongly advised.
-
-.. admonition:: And, by the way...
-
- You can make up your own admonition too.
-
-Topics, Sidebars, and Rubrics
-`````````````````````````````
-
-.. sidebar:: Sidebar Title
- :subtitle: Optional Subtitle
-
- This is a sidebar. It is for text outside the flow of the main
- text.
-
- .. rubric:: This is a rubric inside a sidebar
-
- Sidebars often appears beside the main text with a border and
- background color.
-
-.. topic:: Topic Title
-
- This is a topic.
-
-.. rubric:: This is a rubric
-
-Target Footnotes
-````````````````
-
-.. target-notes::
-
-Replacement Text
-````````````````
-
-I recommend you try |Python|_.
-
-.. |Python| replace:: Python, *the* best language around
-
-Compound Paragraph
-``````````````````
-
-.. compound::
-
- This paragraph contains a literal block::
-
- Connecting... OK
- Transmitting data... OK
- Disconnecting... OK
-
- and thus consists of a simple paragraph, a literal block, and
- another simple paragraph. Nonetheless it is semantically *one*
- paragraph.
-
-This construct is called a *compound paragraph* and can be produced
-with the "compound" directive.
-
-Substitution Definitions
-------------------------
-
-An inline image (|example|) example:
-
-.. |EXAMPLE| image:: images/biohazard.png
-
-(Substitution definitions are not visible in the HTML source.)
-
-Comments
---------
-
-Here's one:
-
-.. Comments begin with two dots and a space. Anything may
- follow, except for the syntax of footnotes, hyperlink
- targets, directives, or substitution definitions.
-
- Double-dashes -- "--" -- must be escaped somehow in HTML output.
-
-(View the HTML source to see the comment.)
-
-Field Lists
-===========
-
-:Field List:
- Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do
- eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
- minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip
- ex ea commodo consequat.
-
- Duis aute irure dolor in reprehenderit in voluptate velit esse cillum
- dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non
- proident, sunt in culpa qui officia deserunt mollit anim id est laborum.
-
-some text
-
-:Field List 2: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor
-
-Error Handling
-==============
-
-Any errors caught during processing will generate system messages.
-
-|*** Expect 6 errors (including this one). ***|
-
-There should be six messages in the following, auto-generated
-section, "Docutils System Messages":
-
-.. section should be added by Docutils automatically
-
-demo.rst from: http://docutils.sourceforge.net/docs/user/rst/demo.txt
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/index.rst b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/index.rst
deleted file mode 100644
index 23492420c7..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/index.rst
+++ /dev/null
@@ -1,217 +0,0 @@
-.. Sphinx RTD theme demo documentation master file, created by
- sphinx-quickstart on Sun Nov 3 11:56:36 2013.
- You can adapt this file completely to your liking, but it should at least
- contain the root `toctree` directive.
-
-=================================================
-Demo Docs
-=================================================
-
-:Page Status: Incomplete
-:Last Reviewed: 2013-10-29
-
-Contents:
-
-.. toctree::
- :maxdepth: 2
-
- demo
- list
-
-Maaaaath!
-=========
-
-This is a test. Here is an equation:
-:math:`X_{0:5} = (X_0, X_1, X_2, X_3, X_4)`.
-Here is another:
-
-.. math::
-
- \nabla^2 f =
- \frac{1}{r^2} \frac{\partial}{\partial r}
- \left( r^2 \frac{\partial f}{\partial r} \right) +
- \frac{1}{r^2 \sin \theta} \frac{\partial f}{\partial \theta}
- \left( \sin \theta \, \frac{\partial f}{\partial \theta} \right) +
- \frac{1}{r^2 \sin^2\theta} \frac{\partial^2 f}{\partial \phi^2}
-
-
-Giant tables
-============
-
-+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+
-| Header 1 | Header 2 | Header 3 | Header 1 | Header 2 | Header 3 | Header 1 | Header 2 | Header 3 | Header 1 | Header 2 | Header 3 |
-+============+============+===========+============+============+===========+============+============+===========+============+============+===========+
-| body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 |
-+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+
-| body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 |
-+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+
-| body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 |
-+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+
-| body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 | body row 1 | column 2 | column 3 |
-+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+------------+------------+-----------+
-
-API Test
-========
-
-.. automodule:: test_py_module.test
- :members:
- :private-members:
- :special-members:
-
-Optional parameter args
------------------------
-
-At this point optional parameters `cannot be generated from code`_.
-However, some projects will manually do it, like so:
-
-This example comes from `django-payments module docs`_.
-
-.. class:: payments.dotpay.DotpayProvider(seller_id, pin[, channel=0[, lock=False], lang='pl'])
-
- This backend implements payments using a popular Polish gateway, `Dotpay.pl <http://www.dotpay.pl>`_.
-
- Due to API limitations there is no support for transferring purchased items.
-
-
- :param seller_id: Seller ID assigned by Dotpay
- :param pin: PIN assigned by Dotpay
- :param channel: Default payment channel (consult reference guide)
- :param lang: UI language
- :param lock: Whether to disable channels other than the default selected above
-
-.. _cannot be generated from code: https://groups.google.com/forum/#!topic/sphinx-users/_qfsVT5Vxpw
-.. _django-payments module docs: http://django-payments.readthedocs.org/en/latest/modules.html#payments.authorizenet.AuthorizeNetProvider
-
-Code test
-=========
-
-.. parsed-literal::
-
- # parsed-literal test
- curl -O http://someurl/release-|version|.tar-gz
-
-
-.. code-block:: json
-
- {
- "windows": [
- {
- "panes": [
- {
- "shell_command": [
- "echo 'did you know'",
- "echo 'you can inline'"
- ]
- },
- {
- "shell_command": "echo 'single commands'"
- },
- "echo 'for panes'"
- ],
- "window_name": "long form"
- }
- ],
- "session_name": "shorthands"
- }
-
-Sidebar
-=======
-
-.. sidebar:: Ch'ien / The Creative
-
- .. image:: static/yi_jing_01_chien.jpg
-
- *Above* CH'IEN THE CREATIVE, HEAVEN
-
- *Below* CH'IEN THE CREATIVE, HEAVEN
-
-The first hexagram is made up of six unbroken lines. These unbroken lines stand for the primal power, which is light-giving, active, strong, and of the spirit. The hexagram is consistently strong in character, and since it is without weakness, its essence is power or energy. Its image is heaven. Its energy is represented as unrestricted by any fixed conditions in space and is therefore conceived of as motion. Time is regarded as the basis of this motion. Thus the hexagram includes also the power of time and the power of persisting in time, that is, duration.
-
-The power represented by the hexagram is to be interpreted in a dual sense in terms of its action on the universe and of its action on the world of men. In relation to the universe, the hexagram expresses the strong, creative action of the Deity. In relation to the human world, it denotes the creative action of the holy man or sage, of the ruler or leader of men, who through his power awakens and develops their higher nature.
-
-Code with Sidebar
-=================
-
-.. sidebar:: A code example
-
- With a sidebar on the right.
-
-.. literalinclude:: test_py_module/test.py
- :language: python
- :linenos:
- :lines: 1-40
-
-Boxes
-=====
-
-.. tip::
- Equations within a note
- :math:`G_{\mu\nu} = 8 \pi G (T_{\mu\nu} + \rho_\Lambda g_{\mu\nu})`.
-
-.. note::
- Equations within a note
- :math:`G_{\mu\nu} = 8 \pi G (T_{\mu\nu} + \rho_\Lambda g_{\mu\nu})`.
-
-.. danger::
- Equations within a note
- :math:`G_{\mu\nu} = 8 \pi G (T_{\mu\nu} + \rho_\Lambda g_{\mu\nu})`.
-
-.. warning::
- Equations within a note
- :math:`G_{\mu\nu} = 8 \pi G (T_{\mu\nu} + \rho_\Lambda g_{\mu\nu})`.
-
-
-Inline code and references
-==========================
-
-`reStructuredText`_ is a markup language. It can use roles and
-declarations to turn reST into HTML.
-
-In reST, ``*hello world*`` becomes ``<em>hello world</em>``. This is
-because a library called `Docutils`_ was able to parse the reST and use a
-``Writer`` to output it that way.
-
-If I type ````an inline literal```` it will wrap it in ``<tt>``. You can
-see more details on the `Inline Markup`_ on the Docutils homepage.
-
-Also with ``sphinx.ext.autodoc``, which I use in the demo, I can link to
-:class:`test_py_module.test.Foo`. It will link you right my code
-documentation for it.
-
-.. _reStructuredText: http://docutils.sourceforge.net/rst.html
-.. _Docutils: http://docutils.sourceforge.net/
-.. _Inline Markup: http://docutils.sourceforge.net/docs/ref/rst/restructuredtext.html#inline-markup
-
-.. note:: Every other line in this table will have white text on a white background.
- This is bad.
-
- +---------+
- | Example |
- +=========+
- | Thing1 |
- +---------+
- | Thing2 |
- +---------+
- | Thing3 |
- +---------+
-
-Emphasized lines with line numbers
-==================================
-
-.. code-block:: python
- :linenos:
- :emphasize-lines: 3,5
-
- def some_function():
- interesting = False
- print 'This line is highlighted.'
- print 'This one is not...'
- print '...but this one is.'
-
-
-Citation
-========
-
-Here I am making a citation [1]_
-
-.. [1] This is the citation I made, let's make this extremely long so that we can tell that it doesn't follow the normal responsive table stuff.
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/list.rst b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/list.rst
deleted file mode 100644
index 47affdd2cf..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/list.rst
+++ /dev/null
@@ -1,69 +0,0 @@
-.. important::
-
- wanna play a game?
-
- - inside
- - this
-
- - list
- - ``in the world``
-
- - hi
- - his
-
- hi
-
-
-
-A list
-======
-
-- here
- - is
- - some
- - list
- - items
- - `yahoo <http://www.yahoo.com>`_
- - ``huh``
-- how
-- ``inline literall``
-- ``inline literall``
-- ``inline literall``
-
-Second list level
------------------
-
-- here is a list in a second-level section.
-- `yahoo <http://www.yahoo.com>`_
-- `yahoo <http://www.yahoo.com>`_
-
- - `yahoo <http://www.yahoo.com>`_
- - here is an inner bullet ``oh``
-
- - one more ``with an inline literally``. `yahoo <http://www.yahoo.com>`_
-
- heh heh. child. try to beat this embed:
-
- .. literalinclude:: test_py_module/test.py
- :language: python
- :linenos:
- :lines: 1-10
- - and another. `yahoo <http://www.yahoo.com>`_
- - `yahoo <http://www.yahoo.com>`_
- - ``hi``
-- and hehe
-
-But deeper down the rabbit hole
-"""""""""""""""""""""""""""""""
-
-- I kept saying that, "deeper down the rabbit hole". `yahoo <http://www.yahoo.com>`_
-
- - I cackle at night `yahoo <http://www.yahoo.com>`_.
-- I'm so lonely here in GZ ``guangzhou``
-- A man of python destiny, hopes and dreams. `yahoo <http://www.yahoo.com>`_
-
- - `yahoo <http://www.yahoo.com>`_
-
- - `yahoo <http://www.yahoo.com>`_ ``hi``
- - ``destiny``
-
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/static/yi_jing_01_chien.jpg b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/static/yi_jing_01_chien.jpg
deleted file mode 100644
index 276df14df2228bdd676f5ef4256be66c20703aa4..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 9498
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
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/__init__.py b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/__init__.py
deleted file mode 100644
index e69de29bb2..0000000000
diff --git a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/test.py b/doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/test.py
deleted file mode 100644
index ca7a8bc92a..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/demo_docs/source/test_py_module/test.py
+++ /dev/null
@@ -1,103 +0,0 @@
-# -*- coding: utf-8 -*-
-"""Test Module for sphinx_rtd_theme."""
-
-
-class Foo:
-
- r"""Docstring for class Foo.
-
- This text tests for the formatting of docstrings generated from output
- ``sphinx.ext.autodoc``. Which contain reST, but sphinx nests it in the
- ``<dl>``, and ``<dt>`` tags. Also, ``<tt>`` is used for class, method names
- and etc, but those will *always* have the ``.descname`` or
- ``.descclassname`` class.
-
- Normal ``<tt>`` (like the <tt> I just wrote here) needs to be shown with
- the same style as anything else with ````this type of markup````.
-
- It's common for programmers to give a code example inside of their
- docstring::
-
- from test_py_module import Foo
-
- myclass = Foo()
- myclass.dothismethod('with this argument')
- myclass.flush()
-
- print(myclass)
-
- """
-
- #: Doc comment for class attribute Foo.bar.
- #: It can have multiple lines.
- bar = 1
-
- flox = 1.5 #: Doc comment for Foo.flox. One line only.
-
- baz = 2
- """Docstring for class attribute Foo.baz."""
-
- def __init__(self, qux, spam=False):
- """Start the Foo.
-
- :param qux: The first argument to initialize class.
- :type qux: string
- :param spam: Spam me yes or no...
- :type spam: bool
-
- """
- #: Doc comment for instance attribute qux.
- self.qux = 3
-
- self.spam = 4
- """Docstring for instance attribute spam."""
-
- def add(self, val1, val2):
- """Return the added values.
-
- :param val1: First number to add.
- :type val1: int
- :param val2: Second number to add.
- :type val2: int
- :rtype: int
-
- """
-
- return val1 + val2
-
- def capitalize(self, myvalue):
- """Return a string as uppercase.
-
- :param myvalue: String to change
- :type myvalue: string
- :rtype: string
-
- """
-
- return myvalue.upper()
-
- def another_function(self, a, b, **kwargs):
- """
- Here is another function.
-
- :param a: The number of green hats you own.
- :type a: int
-
- :param b: The number of non-green hats you own.
- :type b: int
-
- :param kwargs: Additional keyword arguments. Each keyword parameter
- should specify the name of your favorite cuisine.
- The values should be floats, specifying the mean price
- of your favorite dish in that cooking style.
- :type kwargs: float
-
- :returns: A 2-tuple. The first element is the mean price of all dishes
- across cuisines. The second element is the total number of
- hats you own: :math:`a + b`.
- :rtype: tuple
-
- :raises ValueError: When ``a`` is not an integer.
-
- """
- return sum(kwargs.values()) / len(kwargs), a + b
diff --git a/doc/source/ext/sphinx_rtd_theme/package.json b/doc/source/ext/sphinx_rtd_theme/package.json
deleted file mode 100644
index 190cd21ecf..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/package.json
+++ /dev/null
@@ -1,18 +0,0 @@
-{
- "name": "sphinx_rtd_theme",
- "version": "0.0.11",
- "private": true,
- "dependencies": {},
- "devDependencies": {
- "grunt": ">=1.3.0",
- "grunt-contrib-sass": "~0.7.2",
- "grunt-contrib-watch": "~0.4.3",
- "grunt-contrib-connect": "0.5.0",
- "grunt-contrib-clean": "0.5.0",
- "grunt-contrib-copy": "0.5.0",
- "connect-livereload": "~0.3.0",
- "grunt-exec": "~0.4.2",
- "grunt-open": "0.2.2",
- "matchdep": "~0.1.2"
- }
-}
diff --git a/doc/source/ext/sphinx_rtd_theme/requirements.txt b/doc/source/ext/sphinx_rtd_theme/requirements.txt
deleted file mode 100644
index b8f7aa674e..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/requirements.txt
+++ /dev/null
@@ -1 +0,0 @@
-sphinx>=1.1
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge.sass
deleted file mode 100644
index bde7092786..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge.sass
+++ /dev/null
@@ -1,93 +0,0 @@
-.rst-versions
- position: fixed
- bottom: 0
- left: 0
- width: $nav-desktop-width
- color: $section-background-color
- background: darken($menu-background-color, 8%)
- border-top: solid 10px $menu-background-color
- font-family: $base-font-family
- z-index: $z-index-tray
- a
- color: $link_color
- text-decoration: none
- .rst-badge-small
- display: none
- .rst-current-version
- padding: $base-line-height / 2
- background-color: darken($menu-background-color, 5%)
- display: block
- text-align: right
- font-size: 90%
- cursor: pointer
- color: $green
- +clearfix
- .fa
- color: $section-background-color
- .fa-book
- float: left
- .icon-book
- float: left
- &.rst-out-of-date
- background-color: $red
- color: $white
- &.rst-active-old-version
- background-color: $yellow
- color: $black
- &.shift-up .rst-other-versions
- display: block
- .rst-other-versions
- font-size: 90%
- padding: $base-line-height / 2
- color: $text-medium
- display: none
- hr
- display: block
- height: 1px
- border: 0
- margin: 20px 0
- padding: 0
- border-top: solid 1px lighten($menu-background-color, 5%)
- dd
- display: inline-block
- margin: 0
- a
- display: inline-block
- padding: $base-line-height / 4
- color: $section-background-color
- &.rst-badge
- width: auto
- bottom: 20px
- right: 20px
- left: auto
- border: none
- max-width: $nav-desktop-width
- .icon-book
- float: none
- .fa-book
- float: none
- &.shift-up .rst-current-version
- text-align: right
- .fa-book
- float: left
- .icon-book
- float: left
- .rst-current-version
- width: auto
- height: 30px
- line-height: 30px
- padding: 0 $base-line-height / 4
- display: block
- text-align: center
-
-+media($tablet)
- .rst-versions
- width: 85%
- display: none
- &.shift
- display: block
- img
- width: 100%
- height: auto
-
-
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge_fa.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge_fa.sass
deleted file mode 100644
index f265a80bdb..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_badge_fa.sass
+++ /dev/null
@@ -1,68 +0,0 @@
-// Slimmer version of FA for use on the badge_only.sass file.
-
-+font-face(FontAwesome, '#{$font-awesome-dir}fontawesome_webfont')
-
-.fa:before
- display: inline-block
- font-family: FontAwesome
- font-style: normal
- font-weight: normal
- line-height: 1
- text-decoration: inherit
- +font-smooth
-
-a .fa
- display: inline-block
- text-decoration: inherit
-
-
-li
- .fa
- display: inline-block
- .fa-large:before,
- .fa-large:before
- /* 1.5 increased font size for fa-large * 1.25 width
- width: 1.5 * 1.25em
-
-ul.fas
- list-style-type: none
- margin-left: 2em
- text-indent: -0.8em
- li
- .fa
- width: .8em
- .fa-large:before,
- .fa-large:before
- /* 1.5 increased font size for fa-large * 1.25 width
- vertical-align: baseline
- // width: 1.5*1.25em
-
-.fa-book:before
- content: "\f02d"
-
-.icon-book:before
- content: "\f02d"
-
-.fa-caret-down:before
- content: "\f0d7"
-
-.icon-caret-down:before
- content: "\f0d7"
-
-.fa-caret-up:before
- content: "\f0d8"
-
-.icon-caret-up:before
- content: "\f0d8"
-
-.fa-caret-left:before
- content: "\f0d9"
-
-.icon-caret-left:before
- content: "\f0d9"
-
-.fa-caret-right:before
- content: "\f0da"
-
-.icon-caret-right:before
- content: "\f0da"
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_breadcrumbs.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_breadcrumbs.sass
deleted file mode 100644
index a7c1536968..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_breadcrumbs.sass
+++ /dev/null
@@ -1,25 +0,0 @@
-.wy-breadcrumbs li
- display: inline-block
- &.wy-breadcrumbs-aside
- float: right
- a
- display: inline-block
- padding: 5px
- &:first-child
- padding-left: 0
-.wy-breadcrumbs-extra
- margin-bottom: 0
- color: $text-light
- font-size: 80%
- display: inline-block
-
-
-+media($mobile)
- .wy-breadcrumbs-extra
- display: none
- .wy-breadcrumbs li.wy-breadcrumbs-aside
- display: none
-
-@media print
- .wy-breadcrumbs li.wy-breadcrumbs-aside
- display: none
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_font_awesome_compatability.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_font_awesome_compatability.sass
deleted file mode 100644
index d97a55eb3f..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_font_awesome_compatability.sass
+++ /dev/null
@@ -1,22 +0,0 @@
-.icon
- @extend .fa
-.icon-home
- @extend .fa-home
-.icon-search
- @extend .fa-search
-.icon-book
- @extend .fa-book
-.icon-caret-down
- @extend .fa-caret-down
-.icon-github
- @extend .fa-github
-.icon-bitbucket
- @extend .fa-bitbucket
-.icon-fire
- @extend .fa-fire
-.icon-circle-arrow-right
- @extend .fa-arrow-circle-right
-.icon-circle-arrow-left
- @extend .fa-arrow-circle-left
-.icon-link
- @extend .fa-link
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_layout.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_layout.sass
deleted file mode 100644
index 1e9061247f..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_layout.sass
+++ /dev/null
@@ -1,292 +0,0 @@
-.wy-affix
- position: fixed
- top: $gutter
-
-.wy-menu
- a:hover
- text-decoration: none
-
-.wy-menu-horiz
- +clearfix
- ul, li
- display: inline-block
- li:hover
- background: rgba(255,255,255,.1)
- li
- &.divide-left
- border-left: solid 1px hsl(0, 0%, 25%)
- &.divide-right
- border-right: solid 1px hsl(0, 0%, 25%)
- a
- height: $base-font-size * 2
- display: inline-block
- line-height: $base-font-size * 2
- padding: 0 $base-font-size
-
-.wy-menu-vertical
- header
- height: $base-font-size * 2
- display: inline-block
- line-height: $base-font-size * 2
- padding: 0 $gutter
- display: block
- font-weight: bold
- text-transform: uppercase
- font-size: 80%
- color: $menu-logo-color
- white-space: nowrap
-
- ul
- margin-bottom: 0
- li
- &.divide-top
- border-top: solid 1px hsl(0, 0%, 25%)
- &.divide-bottom
- border-bottom: solid 1px hsl(0, 0%, 25%)
- &.current
- background: darken($section-background-color, 10%)
- a
- color: $text-medium
- border-right: solid 1px darken($section-background-color, 20%)
- padding: $gutter / 4 $gutter * 1.5
- &:hover
- background: darken($section-background-color, 15%)
- // On state for the first level
- li.on a, li.current > a
- color: $text-color
- padding: $gutter / 4 $gutter
- font-weight: bold
- position: relative
- background: $section-background-color
- border: none
- border-bottom: solid 1px darken($section-background-color, 20%)
- border-top: solid 1px darken($section-background-color, 20%)
- padding-left: $gutter -4px
- +font-smooth
- &:hover
- background: $section-background-color
- // This is the on state for pages beyond second level
- li.toctree-l2.current > a
- background: darken($section-background-color, 20%)
- padding: $gutter / 4 $gutter * 1.5
- li.current ul
- display: block
- li ul
- margin-bottom: 0
- display: none
- .local-toc
- li ul
- display: block
- li ul li a
- margin-bottom: 0
- color: $text-light
- font-weight: normal
- a
- display: inline-block
- line-height: 18px
- padding: $gutter / 4 $gutter
- display: block
- position: relative
- font-size: 90%
- color: $text-light
- &:hover
- background-color: lighten($menu-background-color, 10%)
- cursor: pointer
- &:active
- background-color: $menu-logo-color
- cursor: pointer
- color: $white
-
-.wy-side-nav-search
- z-index: $z-index-popover
- background-color: $link-color
- text-align: center
- padding: $gutter / 2
- display: block
- color: $section-background-color
- margin-bottom: $gutter / 2
- input[type=text]
- width: 100%
- border-radius: 50px
- padding: 6px 12px
- border-color: darken($link-color, 5%)
- img
- display: block
- margin: auto auto $gutter / 2 auto
- height: 45px
- width: 45px
- background-color: $menu-logo-color
- padding: 5px
- border-radius: 100%
- > a, .wy-dropdown > a
- color: $section-background-color
- font-size: 100%
- font-weight: bold
- display: inline-block
- padding: $base-line-height / 6 $base-line-height / 4
- margin-bottom: $gutter / 2
- +font-smooth
- &:hover
- background: rgba(255,255,255,.1)
-
-.wy-nav .wy-menu-vertical
- header
- color: $link-color
- a
- color: $text-light
- &:hover
- background-color: $link-color
- color: $white
-
-[data-menu-wrap]
- +transition(all .2s ease-in)
- position: absolute
- opacity: 1
- width: 100%
- opacity: 0
- &.move-center
- left: 0
- right: auto
- opacity: 1
- &.move-left
- right: auto
- left: -100%
- opacity: 0
- &.move-right
- right: -100%
- left: auto
- opacity: 0
-
-
-.wy-body-for-nav
- background: left repeat-y $section-background-color
- background-image: url()
- background-size: $nav-desktop-width 1px
-
-.wy-grid-for-nav
- position: absolute
- width: 100%
- height: 100%
-
-.wy-nav-side
- position: absolute
- top: 0
- left: 0
- width: $nav-desktop-width
- overflow: hidden
- min-height: 100%
- background: $menu-background-color
- z-index: $z-index-popover
-
-.wy-nav-top
- display: none
- background: $link-color
- color: $white
- padding: $gutter / 4 $gutter / 2
- position: relative
- line-height: 50px
- text-align: center
- font-size: 100%
- +clearfix
- a
- color: $white
- font-weight: bold
- +font-smooth
- img
- margin-right: $base-line-height / 2
- height: 45px
- width: 45px
- background-color: $menu-logo-color
- padding: 5px
- border-radius: 100%
- i
- font-size: 30px
- float: left
- cursor: pointer
-
-.wy-nav-content-wrap
- margin-left: $nav-desktop-width
- background: $section-background-color
- min-height: 100%
-
-.wy-nav-content
- padding: $gutter $gutter * 2
- height: 100%
- max-width: 800px
- margin: auto
-
-.wy-body-mask
- position: fixed
- width: 100%
- height: 100%
- background: rgba(0,0,0,.2)
- display: none
- z-index: $z-index-modal - 1
- &.on
- display: block
-footer
- color: $gray-light
- p
- margin-bottom: $base-line-height / 2
-
-.rst-footer-buttons
- +clearfix
-
-#search-results
- .search li
- margin-bottom: $base-line-height
- border-bottom: solid 1px $table_border_color
- padding-bottom: $base-line-height
- .search li:first-child
- border-top: solid 1px $table_border_color
- padding-top: $base-line-height
- .search li a
- font-size: 120%
- margin-bottom: $base-line-height / 2
- display: inline-block
- .context
- color: $text-medium
- font-size: 90%
-
-
-+media($tablet)
- .wy-body-for-nav
- background: $section-background-color
- .wy-nav-top
- display: block
- .wy-nav-side
- @if $nav-desktop-position == left
- left: -$nav-desktop-width
- @else
- right: -$nav-desktop-width
- &.shift
- width: 85%
- left: 0
- .wy-nav-content-wrap
- margin-left: 0
- .wy-nav-content
- padding: $gutter
- &.shift
- position: fixed
- min-width: 100%
- left: 85%
- top: 0
- height: 100%
- overflow: hidden
-
-+media($desktop-wider)
- .wy-nav-content-wrap
- background: rgba(0,0,0,.05)
- .wy-nav-content
- margin: 0
- background: $section-background-color
-
-@media print
- .rst-versions, footer, .wy-nav-side
- display: none
- .wy-nav-content-wrap
- margin-left: 0
-
-nav.stickynav
- position: fixed
- top: 0
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_mathjax.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_mathjax.sass
deleted file mode 100644
index 1004993657..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_mathjax.sass
+++ /dev/null
@@ -1,5 +0,0 @@
-span[id*='MathJax-Span']
- color: $mathjax-color
-
-.math
- text-align: center
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_rst.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_rst.sass
deleted file mode 100644
index 02581a46d6..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_rst.sass
+++ /dev/null
@@ -1,282 +0,0 @@
-// -------------------------------------------------------------------------------------------------------------------
-// CONTRIBUTORS, PLEASE READ THIS!
-// -------------------------------------------------------------------------------------------------------------------
-// Couple things...
-// 1. Lots of this @extends from wyrm_core/_type.sass (http://www.github.com/snide/wyrm/.
-// * Try not to replace any @extends code. It's pretty generic stuff meant to work together.
-// * That said, know that I'm very unlikely to accept PRs from wyrm just to change style here.
-// 2. I plan to remove the !importants in here. Part of it is due to lazyness, part to sphinx's fondness for nesting.
-// 3. Try to use variables from wyrm_core/wy_variables.sass. Notable are...
-// * $base-line-height // All margins, padding and line-height should use this in .25 increments.
-// * $text-color, $text-light, $text-dark...etc
-// * $base-font-family, $custom-font-family, $code-font-family
-// 4. If you have changes for mobile/tablet, put them at the bottom of the sass file.
-// --------------------------------------------------------------------------------------------------------------------
-
-.rst-content
- // Sphinx by default applies HxW style attributes to images. This fixes that oversite.
- img
- max-width: 100%
- height: auto !important
-
- div.figure
- margin-bottom: $base-line-height
-
- div.figure.align-center
- text-align: center
-
- // Usually it's a good idea to give images some space.
- .section > img
- margin-bottom: $base-line-height
- // Questionable whether this is nice or not. It styles eternal links, but comes with some baggage.
- // a.reference.external:after
- // font-family: FontAwesome
- // content: " \f08e "
- // color: $text-light
- // vertical-align: super
- // font-size: 60%
-
- // For the most part, its safe to assume that sphinx wants you to use a blockquote as an indent. It gets
- // used in many different ways, so don't assume you can apply some fancy style, just leave it be.
- blockquote
- margin-left: $base-line-height
- line-height: $base-line-height
- margin-bottom: $base-line-height
- .literal-block, pre.literal-block
- @extend .codeblock
- // These are the various note pullouts that sphinx applies
- .note, .attention, .caution, .danger, .error, .hint, .important, .tip, .warning, .seealso, .admonition-todo
- @extend .wy-alert
- .last
- margin-bottom: 0
- .admonition-title
- @extend .wy-alert-title
- @extend .fa
- @extend .fa-exclamation-circle
- &:before
- margin-right: 4px
- .note, .seealso
- @extend .wy-alert.wy-alert-info
- .hint, .tip, .important
- @extend .wy-alert.wy-alert-success
- .error, .danger
- @extend .wy-alert.wy-alert-danger
- .warning, .caution, .attention, .admonition-todo
- @extend .wy-alert.wy-alert-warning
- // Some people put tables in notes. Let's give them very basic support.
- .admonition table
- border-color: rgba(0,0,0,.1)
- td, th
- background: transparent !important
- border-color: rgba(0,0,0,.1) !important
- .section ul, .toctree-wrapper ul
- @extend .wy-plain-list-disc
- .section ol.loweralpha, .section ol.loweralpha li
- list-style: lower-alpha
- .section ol.upperalpha, .section ol.upperalpha li
- list-style: upper-alpha
- .section ol, ol.arabic
- @extend .wy-plain-list-decimal
- .section ol p, .section ul p
- margin-bottom: $base-line-height / 2
- .line-block
- margin-left: $base-line-height
-
- // Generics handling of headings and toc stuff.
- .topic-title
- font-weight: bold
- margin-bottom: $base-line-height / 2
- .toc-backref
- color: $text-color
- .align-right
- float: right
- margin: 0px 0px $base-line-height $base-line-height
- .align-left
- float: left
- margin: 0px $base-line-height $base-line-height 0px
- .align-center
- margin: auto
- display: block
-
- // This is the #href that shows up on hover. Sphinx's is terrible so I hack it away.
- h1, h2, h3, h4, h5, h6, dl dt
- .headerlink
- display: none
- visibility: hidden
- font-size: 14px
- @extend .fa
- &:after
- visibility: visible
- content: "\f0c1"
- font-family: FontAwesome
- display: inline-block
- &:hover .headerlink
- display: inline-block
-
- // Sidebar content. You'll see at the bottom of this file I change it in mobile.
- .sidebar
- float: right
- width: 40%
- display: block
- margin: 0 0 $base-line-height $base-line-height
- padding: $base-line-height
- background: $table-stripe-color
- border: solid 1px $table-border-color
- // Sidebar content is usually less relevant, so adjust the margins and sizes.
- p, ul, dl
- font-size: 90%
- .last
- margin-bottom: 0
- .sidebar-title
- display: block
- font-family: $custom-font-family
- font-weight: bold
- background: $table-border-color
- padding: $base-line-height / 4 $base-line-height / 2
- margin: -$base-line-height
- margin-bottom: $base-line-height
- font-size: 100%
- // Sphinx can highlight searched text with ?highlighted=searchterm
- .highlighted
- background: $yellow
- display: inline-block
- font-weight: bold
- padding: 0 $base-line-height / 4
-
- // These are the little citation links [1] that show up within paragraphs.
- .footnote-reference, .citation-reference
- vertical-align: super
- font-size: 90%
-
- // Tables! Sphinx LOVES TABLES. Most of wyrm assumes you're only going to use a table as a table
- // so I have to write a bunch of unique stuff for Sphinx to style them up differently like it's 2003.
- table.docutils.citation, table.docutils.footnote
- background: none
- border: none
- color: $gray-light
- td, tr
- border: none
- background-color: transparent !important
- white-space: normal
- td.label
- padding-left: 0
- padding-right: 0
- vertical-align: top
- table.docutils
- @extend .wy-table
- @extend .wy-table-bordered-all
- &:not(.field-list)
- @extend .wy-table-striped
- // This table is what gets spit out for auto-generated API stuff. I style it smaller bits of padding.
- table.field-list
- @extend .wy-table
- border: none
- td
- border: none
- padding-top: 5px
- td > strong
- display: inline-block
- margin-top: 3px
- .field-name
- padding-right: 10px
- text-align: left
- white-space: nowrap
- .field-body
- text-align: left
- padding-left: 0
-
- // These are the "literals" that get spit out when you mark stuff as ``code`` as your write.
- tt
- @extend code
- color: $black
- big, em
- font-size: 100% !important
- line-height: normal
-
- .xref, a &
- font-weight: bold
- // If the literal is inside an a tag, let's color it like a link
- a tt
- color: $link-color
- dl
- margin-bottom: $base-line-height
- dt
- font-weight: bold
- // Most of the content within these dls are one liners, so I halve the normal margins.
- p, table, ul, ol
- margin-bottom: $base-line-height / 2 !important
- // rST seems to want dds to be treated as the browser would, indented.
- dd
- margin: 0 0 $base-line-height / 2 $base-line-height
- // This is what Sphinx spits out for it's autodocs. Depending upon what language the person is referencing
- // these things usually have a class of "method" or "class" or something similar, but really who knows.
- // Sphinx doesn't give me a generic class on these, so unfortunately I have to apply it to the root dl.
- // This makes me terribly unhappy and makes this code very nesty. Unfortunately I've seen hand-written docs
- // that output similar, but not quite the same nesting so this is really the best we can do.
- dl:not(.docutils)
- margin-bottom: $base-line-height
- // This would be the equivilant of a .. class::
- dt
- display: inline-block
- margin: $base-line-height / 4 0
- font-size: 90%
- line-height: normal
- background: lighten($blue, 50%)
- color: $blue
- border-top: solid 3px lighten($blue, 20%)
- padding: $base-line-height / 4
- position: relative
- &:before
- color: lighten($blue, 20%)
- .headerlink
- color: $text-color
- font-size: 100% !important
- // And this would be the .. method::
- dl dt
- margin-bottom: $base-line-height / 4
- border: none
- border-left: solid 3px hsl(0,0%,80%)
- background: hsl(0,0%,94%)
- color: $text-medium
- .headerlink
- color: $text-color
- font-size: 100% !important
- dt:first-child
- margin-top: 0
- // Since dts get the callout style, we treat this less as callouts.
- tt
- font-weight: bold
- &.descname, &.descclassname
- background-color: transparent
- border: none
- padding: 0
- font-size: 100% !important
- &.descname
- font-weight: bold
- // This is for more advanced parameter control
- .optional
- display: inline-block
- padding: 0 4px
- color: $black
- font-weight: bold
- .property
- display: inline-block
- padding-right: 8px
- // Doc links to sourcecode
- .viewcode-link, .viewcode-back
- display: inline-block
- color: $green
- font-size: 80%
- padding-left: $base-line-height
- .viewcode-back
- display: block
- float: right
- p.rubric
- margin-bottom: 12px
- font-weight: bold
-
-// Mobile specific
-+media($mobile)
- .rst-content
- .sidebar
- width: 100%
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/_theme_variables.sass b/doc/source/ext/sphinx_rtd_theme/sass/_theme_variables.sass
deleted file mode 100644
index e1760bbfbe..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/_theme_variables.sass
+++ /dev/null
@@ -1,12 +0,0 @@
-// In here are varibles used for sphinx_rtd_theme, they either add to or overwrite the default ones
-// that are set in wyrm_core/wy_variables.sass. You'll find wyrm in bower_components if you're looking
-// for a reference.
-
-$font-awesome-dir: "../font/"
-$static-img: "../img/"
-$mathjax-color: $text-color
-
-$base-font-family: "Lato", "proxima-nova", "Helvetica Neue", Arial, sans-serif
-$custom-font-family: "Roboto Slab", "ff-tisa-web-pro", "Georgia", Arial, sans-serif
-$custom-font-family2: Georgia, serif
-$code-font-family: Consolas, "Andale Mono WT", "Andale Mono", "Lucida Console", "Lucida Sans Typewriter", "DejaVu Sans Mono", "Bitstream Vera Sans Mono", "Liberation Mono", "Nimbus Mono L", Monaco, "Courier New", Courier, monospace
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/badge_only.sass b/doc/source/ext/sphinx_rtd_theme/sass/badge_only.sass
deleted file mode 100644
index b8cdac3ac1..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/badge_only.sass
+++ /dev/null
@@ -1,16 +0,0 @@
-// ------------------------------------------------------------
-// CONTRIBUTORS, PLEASE READ THIS!
-// ------------------------------------------------------------
-// This generates the RTD sticky badge for non RTD themes. As
-// always, only files labeled "theme_*.sass should be edited".
-// ------------------------------------------------------------
-$border-box-sizing: false !default
-
-@import wyrm_core/wy_variables
-@import theme_variables
-@import bourbon
-@import neat
-@import wyrm_core/mixin
-@import wyrm_core/grid_settings
-@import _theme_badge_fa
-@import _theme_badge
diff --git a/doc/source/ext/sphinx_rtd_theme/sass/theme.sass b/doc/source/ext/sphinx_rtd_theme/sass/theme.sass
deleted file mode 100644
index efacd7a0e0..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sass/theme.sass
+++ /dev/null
@@ -1,53 +0,0 @@
-// ------------------------------------------------------------
-// CONTRIBUTORS, PLEASE READ THIS!
-// ------------------------------------------------------------
-// This theme pulls from other frontend projects. The only
-// things you should edit are the sass files that start with
-// "theme_*.sass". All other files are loaded through bower.
-// ------------------------------------------------------------
-
-// Variable defaults set by Wyrm
-@import wyrm_core/wy_variables
-
-// Variable overrides that change coloring and fonts for this theme.
-@import theme_variables
-
-// bourbon.io framework
-@import bourbon
-
-// Bourbon.io/neat framework, with some default media queries
-@import wyrm_core/grid_settings
-@import neat
-// Some corrections for neat
-@import wyrm_core/neat_extra
-
-// Custom reset
-@import wyrm_core/reset
-
-// Wyrm mixins
-@import wyrm_core/mixin
-
-// Font Awesome 4.0 with wyrm extras
-@import font-awesome
-@import wyrm_core/font_icon_defaults
-
-// Wyrm core styles used in this theme
-@import wyrm_core/alert
-@import wyrm_core/button
-@import wyrm_core/dropdown
-@import wyrm_core/form
-@import wyrm_core/generic
-@import wyrm_core/table
-@import wyrm_core/type
-
-// Pygments styling
-@import wyrm_addons/pygments/pygments
-@import wyrm_addons/pygments/pygments_light
-
-// Theme specific styles. These are likely the files you want to edit.
-@import theme_breadcrumbs
-@import theme_layout
-@import theme_badge
-@import theme_rst
-@import theme_mathjax
-@import theme_font_awesome_compatability
diff --git a/doc/source/ext/sphinx_rtd_theme/screen_desktop.png b/doc/source/ext/sphinx_rtd_theme/screen_desktop.png
deleted file mode 100644
index 4e0a5b7577d6e2699701fafc2e2a9f87164cbd0f..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 156671
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�GjsdU0=R;T^;oE=GDppLAZJY
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ͧi+>h2yeVB|D+
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
diff --git a/doc/source/ext/sphinx_rtd_theme/screen_mobile.png b/doc/source/ext/sphinx_rtd_theme/screen_mobile.png
deleted file mode 100644
index 2d27e31f09ea5a66813aab1a75aa7c2a23d51a87..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 96294
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	`TRP-^m6)epFyM&EGNE)Eiw^u;7Lx{}m#x
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
L>*Zg$`|N)JkWlU@
diff --git a/doc/source/ext/sphinx_rtd_theme/setup.py b/doc/source/ext/sphinx_rtd_theme/setup.py
deleted file mode 100644
index 47df30e3e3..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/setup.py
+++ /dev/null
@@ -1,43 +0,0 @@
-# -*- coding: utf-8 -*-
-"""`sphinx_rtd_theme` lives on `Github`_.
-
-.. _github: https://www.github.com/snide/sphinx_rtd_theme
-
-"""
-from setuptools import setup
-from sphinx_rtd_theme import __version__
-
-
-setup(
- name='sphinx_rtd_theme',
- version=__version__,
- url='https://github.com/snide/sphinx_rtd_theme/',
- license='MIT',
- author='Dave Snider',
- author_email='dave.snider@gmail.com',
- description='ReadTheDocs.org theme for Sphinx, 2013 version.',
- long_description=open('README.rst').read(),
- zip_safe=False,
- packages=['sphinx_rtd_theme'],
- package_data={'sphinx_rtd_theme': [
- 'theme.conf',
- '*.html',
- 'static/css/*.css',
- 'static/js/*.js',
- 'static/font/*.*'
- ]},
- include_package_data=True,
- install_requires=open('requirements.txt').read().splitlines(),
- classifiers=[
- 'Development Status :: 3 - Alpha',
- 'License :: OSI Approved :: BSD License',
- 'Environment :: Console',
- 'Environment :: Web Environment',
- 'Intended Audience :: Developers',
- 'Programming Language :: Python :: 2.7',
- 'Programming Language :: Python :: 3',
- 'Operating System :: OS Independent',
- 'Topic :: Documentation',
- 'Topic :: Software Development :: Documentation',
- ],
-)
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/__init__.py b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/__init__.py
deleted file mode 100644
index 1440863d68..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/__init__.py
+++ /dev/null
@@ -1,17 +0,0 @@
-"""Sphinx ReadTheDocs theme.
-
-From https://github.com/ryan-roemer/sphinx-bootstrap-theme.
-
-"""
-import os
-
-VERSION = (0, 1, 5)
-
-__version__ = ".".join(str(v) for v in VERSION)
-__version_full__ = __version__
-
-
-def get_html_theme_path():
- """Return list of HTML theme paths."""
- cur_dir = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
- return cur_dir
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/breadcrumbs.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/breadcrumbs.html
deleted file mode 100644
index ff0938e5c8..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/breadcrumbs.html
+++ /dev/null
@@ -1,19 +0,0 @@
-<div role="navigation" aria-label="breadcrumbs navigation">
- <ul class="wy-breadcrumbs">
- <li><a href="{{ pathto(master_doc) }}">Docs</a> »</li>
- {% for doc in parents %}
- <li><a href="{{ doc.link|e }}">{{ doc.title }}</a> »</li>
- {% endfor %}
- <li>{{ title }}</li>
- <li class="wy-breadcrumbs-aside">
- {% if display_github %}
- <a href="https://github.com/{{ github_user }}/{{ github_repo }}/blob/{{ github_version }}{{ conf_py_path }}{{ pagename }}{{ source_suffix }}" class="fa fa-github"> Edit on GitHub</a>
- {% elif display_bitbucket %}
- <a href="https://bitbucket.org/{{ bitbucket_user }}/{{ bitbucket_repo }}/src/{{ bitbucket_version}}{{ conf_py_path }}{{ pagename }}{{ source_suffix }}" class="fa fa-bitbucket"> Edit on Bitbucket</a>
- {% elif show_source and has_source and sourcename %}
- <a href="{{ pathto('_sources/' + sourcename, true)|e }}" rel="nofollow"> View page source</a>
- {% endif %}
- </li>
- </ul>
- <hr/>
-</div>
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/footer.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/footer.html
deleted file mode 100644
index 3c7afed33b..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/footer.html
+++ /dev/null
@@ -1,32 +0,0 @@
-<footer>
- {% if next or prev %}
- <div class="rst-footer-buttons" role="navigation" aria-label="footer navigation">
- {% if next %}
- <a href="{{ next.link|e }}" class="btn btn-neutral float-right" title="{{ next.title|striptags|e }}">Next <span class="fa fa-arrow-circle-right"></span></a>
- {% endif %}
- {% if prev %}
- <a href="{{ prev.link|e }}" class="btn btn-neutral" title="{{ prev.title|striptags|e }}"><span class="fa fa-arrow-circle-left"></span> Previous</a>
- {% endif %}
- </div>
- {% endif %}
-
- <hr/>
-
- <div role="contentinfo">
- <p>
- {%- if show_copyright %}
- {%- if hasdoc('copyright') %}
- {% trans path=pathto('copyright'), copyright=copyright|e %}© <a href="{{ path }}">Copyright</a> {{ copyright }}.{% endtrans %}
- {%- else %}
- {% trans copyright=copyright|e %}© Copyright {{ copyright }}.{% endtrans %}
- {%- endif %}
- {%- endif %}
-
- {%- if last_updated %}
- {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %}
- {%- endif %}
- </p>
- </div>
-
- {% trans %}<a href="https://github.com/snide/sphinx_rtd_theme">Sphinx theme</a> provided by <a href="https://readthedocs.org">Read the Docs</a>{% endtrans %}
-</footer>
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout.html
deleted file mode 100644
index 20a13eef84..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout.html
+++ /dev/null
@@ -1,162 +0,0 @@
-{# TEMPLATE VAR SETTINGS #}
-{%- set url_root = pathto('', 1) %}
-{%- if url_root == '#' %}{% set url_root = '' %}{% endif %}
-{%- if not embedded and docstitle %}
- {%- set titlesuffix = " — "|safe + docstitle|e %}
-{%- else %}
- {%- set titlesuffix = "" %}
-{%- endif %}
-
-<!DOCTYPE html>
-<!--[if IE 8]><html class="no-js lt-ie9" lang="en" > <![endif]-->
-<!--[if gt IE 8]><!--> <html class="no-js" lang="en" > <!--<![endif]-->
-<head>
- <meta charset="utf-8">
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
- {% block htmltitle %}
- <title>{{ title|striptags|e }}{{ titlesuffix }}</title>
- {% endblock %}
-
- {# FAVICON #}
- {% if favicon %}
- <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/>
- {% endif %}
-
- {# CSS #}
- <link href='https://fonts.googleapis.com/css?family=Lato:400,700,400italic,700italic|Roboto+Slab:400,700|Inconsolata:400,700' rel='stylesheet' type='text/css'>
-
- {# OPENSEARCH #}
- {% if not embedded %}
- {% if use_opensearch %}
- <link rel="search" type="application/opensearchdescription+xml" title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}" href="{{ pathto('_static/opensearch.xml', 1) }}"/>
- {% endif %}
-
- {% endif %}
-
- {# RTD hosts this file, so just load on non RTD builds #}
- {% if not READTHEDOCS %}
- <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" />
- {% endif %}
-
- {% for cssfile in css_files %}
- <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" />
- {% endfor %}
-
- {%- block linktags %}
- {%- if hasdoc('about') %}
- <link rel="author" title="{{ _('About these documents') }}"
- href="{{ pathto('about') }}"/>
- {%- endif %}
- {%- if hasdoc('genindex') %}
- <link rel="index" title="{{ _('Index') }}"
- href="{{ pathto('genindex') }}"/>
- {%- endif %}
- {%- if hasdoc('search') %}
- <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}"/>
- {%- endif %}
- {%- if hasdoc('copyright') %}
- <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}"/>
- {%- endif %}
- <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}"/>
- {%- if parents %}
- <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}"/>
- {%- endif %}
- {%- if next %}
- <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}"/>
- {%- endif %}
- {%- if prev %}
- <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}"/>
- {%- endif %}
- {%- endblock %}
- {%- block extrahead %} {% endblock %}
-
- {# Keep modernizr in head - http://modernizr.com/docs/#installing #}
- <script src="https://cdnjs.cloudflare.com/ajax/libs/modernizr/2.6.2/modernizr.min.js"></script>
-
-</head>
-
-<body class="wy-body-for-nav" role="document">
-
- <div class="wy-grid-for-nav">
-
- {# SIDE NAV, TOGGLES ON MOBILE #}
- <nav data-toggle="wy-nav-shift" class="wy-nav-side">
- <div class="wy-side-nav-search">
- {% block sidebartitle %}
- <a href="{{ pathto(master_doc) }}" class="fa fa-home"> {{ project }}</a>
- {% endblock %}
- {% include "searchbox.html" %}
- </div>
-
- <div class="wy-menu wy-menu-vertical" data-spy="affix" role="navigation" aria-label="main navigation">
- {% set toctree = toctree(maxdepth=2, collapse=False, includehidden=True) %}
- {% if toctree %}
- {{ toctree }}
- {% else %}
- <!-- Local TOC -->
- <div class="local-toc">{{ toc }}</div>
- {% endif %}
- </div>
-
- </nav>
-
- <section data-toggle="wy-nav-shift" class="wy-nav-content-wrap">
-
- {# MOBILE NAV, TRIGGLES SIDE NAV ON TOGGLE #}
- <nav class="wy-nav-top" role="navigation" aria-label="top navigation">
- <i data-toggle="wy-nav-top" class="fa fa-bars"></i>
- <a href="{{ pathto(master_doc) }}">{{ project }}</a>
- </nav>
-
-
- {# PAGE CONTENT #}
- <div class="wy-nav-content">
- <div class="rst-content">
- {% include "breadcrumbs.html" %}
- <div role="main" class="document">
- {% block body %}{% endblock %}
- </div>
- {% include "footer.html" %}
- </div>
- </div>
-
- </section>
-
- </div>
- {% include "versions.html" %}
-
- {% if not embedded %}
-
- <script type="text/javascript">
- var DOCUMENTATION_OPTIONS = {
- URL_ROOT:'{{ url_root }}',
- VERSION:'{{ release|e }}',
- COLLAPSE_INDEX:false,
- FILE_SUFFIX:'{{ '' if no_search_suffix else file_suffix }}',
- HAS_SOURCE: {{ has_source|lower }}
- };
- </script>
- {%- for scriptfile in script_files %}
- <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script>
- {%- endfor %}
-
- {% endif %}
-
- {# RTD hosts this file, so just load on non RTD builds #}
- {% if not READTHEDOCS %}
- <script type="text/javascript" src="{{ pathto('_static/js/theme.js', 1) }}"></script>
- {% endif %}
-
- {# STICKY NAVIGATION #}
- {% if theme_sticky_navigation %}
- <script type="text/javascript">
- jQuery(function () {
- SphinxRtdTheme.StickyNav.enable();
- });
- </script>
- {% endif %}
-
- {%- block footer %} {% endblock %}
-
-</body>
-</html>
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout_old.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout_old.html
deleted file mode 100644
index deb8df2a1a..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/layout_old.html
+++ /dev/null
@@ -1,205 +0,0 @@
-{#
- basic/layout.html
- ~~~~~~~~~~~~~~~~~
-
- Master layout template for Sphinx themes.
-
- :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS.
- :license: BSD, see LICENSE for details.
-#}
-{%- block doctype -%}
-<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
- "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
-{%- endblock %}
-{%- set reldelim1 = reldelim1 is not defined and ' »' or reldelim1 %}
-{%- set reldelim2 = reldelim2 is not defined and ' |' or reldelim2 %}
-{%- set render_sidebar = (not embedded) and (not theme_nosidebar|tobool) and
- (sidebars != []) %}
-{%- set url_root = pathto('', 1) %}
-{# XXX necessary? #}
-{%- if url_root == '#' %}{% set url_root = '' %}{% endif %}
-{%- if not embedded and docstitle %}
- {%- set titlesuffix = " — "|safe + docstitle|e %}
-{%- else %}
- {%- set titlesuffix = "" %}
-{%- endif %}
-
-{%- macro relbar() %}
- <div class="related">
- <h3>{{ _('Navigation') }}</h3>
- <ul>
- {%- for rellink in rellinks %}
- <li class="right" {% if loop.first %}style="margin-right: 10px"{% endif %}>
- <a href="{{ pathto(rellink[0]) }}" title="{{ rellink[1]|striptags|e }}"
- {{ accesskey(rellink[2]) }}>{{ rellink[3] }}</a>
- {%- if not loop.first %}{{ reldelim2 }}{% endif %}</li>
- {%- endfor %}
- {%- block rootrellink %}
- <li><a href="{{ pathto(master_doc) }}">{{ shorttitle|e }}</a>{{ reldelim1 }}</li>
- {%- endblock %}
- {%- for parent in parents %}
- <li><a href="{{ parent.link|e }}" {% if loop.last %}{{ accesskey("U") }}{% endif %}>{{ parent.title }}</a>{{ reldelim1 }}</li>
- {%- endfor %}
- {%- block relbaritems %} {% endblock %}
- </ul>
- </div>
-{%- endmacro %}
-
-{%- macro sidebar() %}
- {%- if render_sidebar %}
- <div class="sphinxsidebar">
- <div class="sphinxsidebarwrapper">
- {%- block sidebarlogo %}
- {%- if logo %}
- <p class="logo"><a href="{{ pathto(master_doc) }}">
- <img class="logo" src="{{ pathto('_static/' + logo, 1) }}" alt="Logo"/>
- </a></p>
- {%- endif %}
- {%- endblock %}
- {%- if sidebars != None %}
- {#- new style sidebar: explicitly include/exclude templates #}
- {%- for sidebartemplate in sidebars %}
- {%- include sidebartemplate %}
- {%- endfor %}
- {%- else %}
- {#- old style sidebars: using blocks -- should be deprecated #}
- {%- block sidebartoc %}
- {%- include "localtoc.html" %}
- {%- endblock %}
- {%- block sidebarrel %}
- {%- include "relations.html" %}
- {%- endblock %}
- {%- block sidebarsourcelink %}
- {%- include "sourcelink.html" %}
- {%- endblock %}
- {%- if customsidebar %}
- {%- include customsidebar %}
- {%- endif %}
- {%- block sidebarsearch %}
- {%- include "searchbox.html" %}
- {%- endblock %}
- {%- endif %}
- </div>
- </div>
- {%- endif %}
-{%- endmacro %}
-
-{%- macro script() %}
- <script type="text/javascript">
- var DOCUMENTATION_OPTIONS = {
- URL_ROOT: '{{ url_root }}',
- VERSION: '{{ release|e }}',
- COLLAPSE_INDEX: false,
- FILE_SUFFIX: '{{ '' if no_search_suffix else file_suffix }}',
- HAS_SOURCE: {{ has_source|lower }}
- };
- </script>
- {%- for scriptfile in script_files %}
- <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script>
- {%- endfor %}
-{%- endmacro %}
-
-{%- macro css() %}
- <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" />
- <link rel="stylesheet" href="{{ pathto('_static/pygments.css', 1) }}" type="text/css" />
- {%- for cssfile in css_files %}
- <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" />
- {%- endfor %}
-{%- endmacro %}
-
-<html xmlns="http://www.w3.org/1999/xhtml">
- <head>
- <meta http-equiv="Content-Type" content="text/html; charset={{ encoding }}" />
- {{ metatags }}
- {%- block htmltitle %}
- <title>{{ title|striptags|e }}{{ titlesuffix }}</title>
- {%- endblock %}
- {{ css() }}
- {%- if not embedded %}
- {{ script() }}
- {%- if use_opensearch %}
- <link rel="search" type="application/opensearchdescription+xml"
- title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}"
- href="{{ pathto('_static/opensearch.xml', 1) }}"/>
- {%- endif %}
- {%- if favicon %}
- <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/>
- {%- endif %}
- {%- endif %}
-{%- block linktags %}
- {%- if hasdoc('about') %}
- <link rel="author" title="{{ _('About these documents') }}" href="{{ pathto('about') }}" />
- {%- endif %}
- {%- if hasdoc('genindex') %}
- <link rel="index" title="{{ _('Index') }}" href="{{ pathto('genindex') }}" />
- {%- endif %}
- {%- if hasdoc('search') %}
- <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}" />
- {%- endif %}
- {%- if hasdoc('copyright') %}
- <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}" />
- {%- endif %}
- <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}" />
- {%- if parents %}
- <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}" />
- {%- endif %}
- {%- if next %}
- <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}" />
- {%- endif %}
- {%- if prev %}
- <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}" />
- {%- endif %}
-{%- endblock %}
-{%- block extrahead %} {% endblock %}
- </head>
- <body>
-{%- block header %}{% endblock %}
-
-{%- block relbar1 %}{{ relbar() }}{% endblock %}
-
-{%- block content %}
- {%- block sidebar1 %} {# possible location for sidebar #} {% endblock %}
-
- <div class="document">
- {%- block document %}
- <div class="documentwrapper">
- {%- if render_sidebar %}
- <div class="bodywrapper">
- {%- endif %}
- <div class="body">
- {% block body %} {% endblock %}
- </div>
- {%- if render_sidebar %}
- </div>
- {%- endif %}
- </div>
- {%- endblock %}
-
- {%- block sidebar2 %}{{ sidebar() }}{% endblock %}
- <div class="clearer"></div>
- </div>
-{%- endblock %}
-
-{%- block relbar2 %}{{ relbar() }}{% endblock %}
-
-{%- block footer %}
- <div class="footer">
- {%- if show_copyright %}
- {%- if hasdoc('copyright') %}
- {% trans path=pathto('copyright'), copyright=copyright|e %}© <a href="{{ path }}">Copyright</a> {{ copyright }}.{% endtrans %}
- {%- else %}
- {% trans copyright=copyright|e %}© Copyright {{ copyright }}.{% endtrans %}
- {%- endif %}
- {%- endif %}
- {%- if last_updated %}
- {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %}
- {%- endif %}
- {%- if show_sphinx %}
- {% trans sphinx_version=sphinx_version|e %}Created using <a href="http://sphinx-doc.org/">Sphinx</a> {{ sphinx_version }}.{% endtrans %}
- {%- endif %}
- </div>
- <p>asdf asdf asdf asdf 22</p>
-{%- endblock %}
- </body>
-</html>
-
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/search.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/search.html
deleted file mode 100644
index e3aa9b5c6e..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/search.html
+++ /dev/null
@@ -1,50 +0,0 @@
-{#
- basic/search.html
- ~~~~~~~~~~~~~~~~~
-
- Template for the search page.
-
- :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS.
- :license: BSD, see LICENSE for details.
-#}
-{%- extends "layout.html" %}
-{% set title = _('Search') %}
-{% set script_files = script_files + ['_static/searchtools.js'] %}
-{% block footer %}
- <script type="text/javascript">
- jQuery(function() { Search.loadIndex("{{ pathto('searchindex.js', 1) }}"); });
- </script>
- {# this is used when loading the search index using $.ajax fails,
- such as on Chrome for documents on localhost #}
- <script type="text/javascript" id="searchindexloader"></script>
- {{ super() }}
-{% endblock %}
-{% block body %}
- <noscript>
- <div id="fallback" class="admonition warning">
- <p class="last">
- {% trans %}Please activate JavaScript to enable the search
- functionality.{% endtrans %}
- </p>
- </div>
- </noscript>
-
- {% if search_performed %}
- <h2>{{ _('Search Results') }}</h2>
- {% if not search_results %}
- <p>{{ _('Your search did not match any documents. Please make sure that all words are spelled correctly and that you\'ve selected enough categories.') }}</p>
- {% endif %}
- {% endif %}
- <div id="search-results">
- {% if search_results %}
- <ul>
- {% for href, caption, context in search_results %}
- <li>
- <a href="{{ pathto(item.href) }}">{{ caption }}</a>
- <p class="context">{{ context|e }}</p>
- </li>
- {% endfor %}
- </ul>
- {% endif %}
- </div>
-{% endblock %}
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/searchbox.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/searchbox.html
deleted file mode 100644
index 24418d32bc..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/searchbox.html
+++ /dev/null
@@ -1,7 +0,0 @@
-<div role="search">
- <form id ="rtd-search-form" class="wy-form" action="{{ pathto('search') }}" method="get">
- <input type="text" name="q" placeholder="Search docs" />
- <input type="hidden" name="check_keywords" value="yes" />
- <input type="hidden" name="area" value="default" />
- </form>
-</div>
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/badge_only.css b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/badge_only.css
deleted file mode 100644
index 7e17fb148c..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/badge_only.css
+++ /dev/null
@@ -1,2 +0,0 @@
-.fa:before{-webkit-font-smoothing:antialiased}.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;content:""}.clearfix:after{clear:both}@font-face{font-family:FontAwesome;font-weight:normal;font-style:normal;src:url("../font/fontawesome_webfont.eot");src:url("../font/fontawesome_webfont.eot?#iefix") format("embedded-opentype"),url("../font/fontawesome_webfont.woff") format("woff"),url("../font/fontawesome_webfont.ttf") format("truetype"),url("../font/fontawesome_webfont.svg#FontAwesome") format("svg")}.fa:before{display:inline-block;font-family:FontAwesome;font-style:normal;font-weight:normal;line-height:1;text-decoration:inherit}a .fa{display:inline-block;text-decoration:inherit}li .fa{display:inline-block}li .fa-large:before,li .fa-large:before{width:1.875em}ul.fas{list-style-type:none;margin-left:2em;text-indent:-0.8em}ul.fas li .fa{width:0.8em}ul.fas li .fa-large:before,ul.fas li .fa-large:before{vertical-align:baseline}.fa-book:before{content:""}.icon-book:before{content:""}.fa-caret-down:before{content:""}.icon-caret-down:before{content:""}.fa-caret-up:before{content:""}.icon-caret-up:before{content:""}.fa-caret-left:before{content:""}.icon-caret-left:before{content:""}.fa-caret-right:before{content:""}.icon-caret-right:before{content:""}.rst-versions{position:fixed;bottom:0;left:0;width:300px;color:#fcfcfc;background:#1f1d1d;border-top:solid 10px #343131;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;z-index:400}.rst-versions a{color:#2980B9;text-decoration:none}.rst-versions .rst-badge-small{display:none}.rst-versions .rst-current-version{padding:12px;background-color:#272525;display:block;text-align:right;font-size:90%;cursor:pointer;color:#27AE60;*zoom:1}.rst-versions .rst-current-version:before,.rst-versions .rst-current-version:after{display:table;content:""}.rst-versions .rst-current-version:after{clear:both}.rst-versions .rst-current-version .fa{color:#fcfcfc}.rst-versions .rst-current-version .fa-book{float:left}.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version.rst-out-of-date{background-color:#E74C3C;color:#fff}.rst-versions .rst-current-version.rst-active-old-version{background-color:#F1C40F;color:#000}.rst-versions.shift-up .rst-other-versions{display:block}.rst-versions .rst-other-versions{font-size:90%;padding:12px;color:gray;display:none}.rst-versions .rst-other-versions hr{display:block;height:1px;border:0;margin:20px 0;padding:0;border-top:solid 1px #413d3d}.rst-versions .rst-other-versions dd{display:inline-block;margin:0}.rst-versions .rst-other-versions dd a{display:inline-block;padding:6px;color:#fcfcfc}.rst-versions.rst-badge{width:auto;bottom:20px;right:20px;left:auto;border:none;max-width:300px}.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge .fa-book{float:none}.rst-versions.rst-badge.shift-up .rst-current-version{text-align:right}.rst-versions.rst-badge.shift-up .rst-current-version .fa-book{float:left}.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge .rst-current-version{width:auto;height:30px;line-height:30px;padding:0 6px;display:block;text-align:center}@media screen and (max-width: 768px){.rst-versions{width:85%;display:none}.rst-versions.shift{display:block}img{width:100%;height:auto}}
-/*# sourceMappingURL=badge_only.css.map */
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/theme.css b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/theme.css
deleted file mode 100644
index 509536e6bd..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/css/theme.css
+++ /dev/null
@@ -1,5 +0,0 @@
-*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}article,aside,details,figcaption,figure,footer,header,hgroup,nav,section{display:block}audio,canvas,video{display:inline-block;*display:inline;*zoom:1}audio:not([controls]){display:none}[hidden]{display:none}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:100%;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}a:hover,a:active{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:bold}blockquote{margin:0}dfn{font-style:italic}ins{background:#ff9;color:#000;text-decoration:none}mark{background:#ff0;color:#000;font-style:italic;font-weight:bold}pre,code,.rst-content tt,kbd,samp{font-family:monospace,serif;_font-family:"courier new",monospace;font-size:1em}pre{white-space:pre}q{quotes:none}q:before,q:after{content:"";content:none}small{font-size:85%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-0.5em}sub{bottom:-0.25em}ul,ol,dl{margin:0;padding:0;list-style:none;list-style-image:none}li{list-style:none}dd{margin:0}img{border:0;-ms-interpolation-mode:bicubic;vertical-align:middle;max-width:100%}svg:not(:root){overflow:hidden}figure{margin:0}form{margin:0}fieldset{border:0;margin:0;padding:0}label{cursor:pointer}legend{border:0;*margin-left:-7px;padding:0;white-space:normal}button,input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle}button,input{line-height:normal}button,input[type="button"],input[type="reset"],input[type="submit"]{cursor:pointer;-webkit-appearance:button;*overflow:visible}button[disabled],input[disabled]{cursor:default}input[type="checkbox"],input[type="radio"]{box-sizing:border-box;padding:0;*width:13px;*height:13px}input[type="search"]{-webkit-appearance:textfield;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box}input[type="search"]::-webkit-search-decoration,input[type="search"]::-webkit-search-cancel-button{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top;resize:vertical}table{border-collapse:collapse;border-spacing:0}td{vertical-align:top}.chromeframe{margin:0.2em 0;background:#ccc;color:#000;padding:0.2em 0}.ir{display:block;border:0;text-indent:-999em;overflow:hidden;background-color:transparent;background-repeat:no-repeat;text-align:left;direction:ltr;*line-height:0}.ir br{display:none}.hidden{display:none !important;visibility:hidden}.visuallyhidden{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px}.visuallyhidden.focusable:active,.visuallyhidden.focusable:focus{clip:auto;height:auto;margin:0;overflow:visible;position:static;width:auto}.invisible{visibility:hidden}.relative{position:relative}big,small{font-size:100%}@media print{html,body,section{background:none !important}*{box-shadow:none !important;text-shadow:none !important;filter:none !important;-ms-filter:none !important}a,a:visited{text-decoration:underline}.ir a:after,a[href^="javascript:"]:after,a[href^="#"]:after{content:""}pre,blockquote{page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100% !important}@page{margin:0.5cm}p,h2,h3{orphans:3;widows:3}h2,h3{page-break-after:avoid}}.fa:before,.rst-content .admonition-title:before,.rst-content h1 .headerlink:before,.rst-content h2 .headerlink:before,.rst-content h3 .headerlink:before,.rst-content h4 .headerlink:before,.rst-content h5 .headerlink:before,.rst-content h6 .headerlink:before,.rst-content dl dt .headerlink:before,.icon:before,.wy-dropdown .caret:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before,.wy-alert,.rst-content .note,.rst-content .attention,.rst-content .caution,.rst-content .danger,.rst-content .error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .warning,.rst-content .seealso,.rst-content .admonition-todo,.btn,input[type="text"],input[type="password"],input[type="email"],input[type="url"],input[type="date"],input[type="month"],input[type="time"],input[type="datetime"],input[type="datetime-local"],input[type="week"],input[type="number"],input[type="search"],input[type="tel"],input[type="color"],select,textarea,.wy-menu-vertical li.on a,.wy-menu-vertical li.current>a,.wy-side-nav-search>a,.wy-side-nav-search .wy-dropdown>a,.wy-nav-top a{-webkit-font-smoothing:antialiased}.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;content:""}.clearfix:after{clear:both}/*!
- * Font Awesome 4.1.0 by @davegandy - http://fontawesome.io - @fontawesome
- * License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)
- */@font-face{font-family:'FontAwesome';src:url("../fonts/fontawesome-webfont.eot?v=4.1.0");src:url("../fonts/fontawesome-webfont.eot?#iefix&v=4.1.0") format("embedded-opentype"),url("../fonts/fontawesome-webfont.woff?v=4.1.0") format("woff"),url("../fonts/fontawesome-webfont.ttf?v=4.1.0") format("truetype"),url("../fonts/fontawesome-webfont.svg?v=4.1.0#fontawesomeregular") format("svg");font-weight:normal;font-style:normal}.fa,.rst-content .admonition-title,.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink,.icon{display:inline-block;font-family:FontAwesome;font-style:normal;font-weight:normal;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.fa-lg{font-size:1.33333em;line-height:0.75em;vertical-align:-15%}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-fw{width:1.28571em;text-align:center}.fa-ul{padding-left:0;margin-left:2.14286em;list-style-type:none}.fa-ul>li{position:relative}.fa-li{position:absolute;left:-2.14286em;width:2.14286em;top:0.14286em;text-align:center}.fa-li.fa-lg{left:-1.85714em}.fa-border{padding:.2em .25em .15em;border:solid 0.08em #eee;border-radius:.1em}.pull-right{float:right}.pull-left{float:left}.fa.pull-left,.rst-content .pull-left.admonition-title,.rst-content h1 .pull-left.headerlink,.rst-content h2 .pull-left.headerlink,.rst-content h3 .pull-left.headerlink,.rst-content h4 .pull-left.headerlink,.rst-content h5 .pull-left.headerlink,.rst-content h6 .pull-left.headerlink,.rst-content dl dt .pull-left.headerlink,.pull-left.icon{margin-right:.3em}.fa.pull-right,.rst-content .pull-right.admonition-title,.rst-content h1 .pull-right.headerlink,.rst-content h2 .pull-right.headerlink,.rst-content h3 .pull-right.headerlink,.rst-content h4 .pull-right.headerlink,.rst-content h5 .pull-right.headerlink,.rst-content h6 .pull-right.headerlink,.rst-content dl dt .pull-right.headerlink,.pull-right.icon{margin-left:.3em}.fa-spin{-webkit-animation:spin 2s infinite linear;-moz-animation:spin 2s infinite linear;-o-animation:spin 2s infinite linear;animation:spin 2s infinite linear}@-moz-keyframes spin{0%{-moz-transform:rotate(0deg)}100%{-moz-transform:rotate(359deg)}}@-webkit-keyframes spin{0%{-webkit-transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg)}}@-o-keyframes spin{0%{-o-transform:rotate(0deg)}100%{-o-transform:rotate(359deg)}}@keyframes spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}.fa-rotate-90{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=1);-webkit-transform:rotate(90deg);-moz-transform:rotate(90deg);-ms-transform:rotate(90deg);-o-transform:rotate(90deg);transform:rotate(90deg)}.fa-rotate-180{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2);-webkit-transform:rotate(180deg);-moz-transform:rotate(180deg);-ms-transform:rotate(180deg);-o-transform:rotate(180deg);transform:rotate(180deg)}.fa-rotate-270{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=3);-webkit-transform:rotate(270deg);-moz-transform:rotate(270deg);-ms-transform:rotate(270deg);-o-transform:rotate(270deg);transform:rotate(270deg)}.fa-flip-horizontal{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=0);-webkit-transform:scale(-1, 1);-moz-transform:scale(-1, 1);-ms-transform:scale(-1, 1);-o-transform:scale(-1, 1);transform:scale(-1, 1)}.fa-flip-vertical{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2);-webkit-transform:scale(1, -1);-moz-transform:scale(1, -1);-ms-transform:scale(1, -1);-o-transform:scale(1, -1);transform:scale(1, -1)}.fa-stack{position:relative;display:inline-block;width:2em;height:2em;line-height:2em;vertical-align:middle}.fa-stack-1x,.fa-stack-2x{position:absolute;left:0;width:100%;text-align:center}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa-glass:before{content:""}.fa-music:before{content:""}.fa-search:before,.icon-search:before{content:""}.fa-envelope-o:before{content:""}.fa-heart:before{content:""}.fa-star:before{content:""}.fa-star-o:before{content:""}.fa-user:before{content:""}.fa-film:before{content:""}.fa-th-large:before{content:""}.fa-th:before{content:""}.fa-th-list:before{content:""}.fa-check:before{content:""}.fa-times:before{content:""}.fa-search-plus:before{content:""}.fa-search-minus:before{content:""}.fa-power-off:before{content:""}.fa-signal:before{content:""}.fa-gear:before,.fa-cog:before{content:""}.fa-trash-o:before{content:""}.fa-home:before,.icon-home:before{content:""}.fa-file-o:before{content:""}.fa-clock-o:before{content:""}.fa-road:before{content:""}.fa-download:before{content:""}.fa-arrow-circle-o-down:before{content:""}.fa-arrow-circle-o-up:before{content:""}.fa-inbox:before{content:""}.fa-play-circle-o:before{content:""}.fa-rotate-right:before,.fa-repeat:before{content:""}.fa-refresh:before{content:""}.fa-list-alt:before{content:""}.fa-lock:before{content:""}.fa-flag:before{content:""}.fa-headphones:before{content:""}.fa-volume-off:before{content:""}.fa-volume-down:before{content:""}.fa-volume-up:before{content:""}.fa-qrcode:before{content:""}.fa-barcode:before{content:""}.fa-tag:before{content:""}.fa-tags:before{content:""}.fa-book:before,.icon-book:before{content:""}.fa-bookmark:before{content:""}.fa-print:before{content:""}.fa-camera:before{content:""}.fa-font:before{content:""}.fa-bold:before{content:""}.fa-italic:before{content:""}.fa-text-height:before{content:""}.fa-text-width:before{content:""}.fa-align-left:before{content:""}.fa-align-center:before{content:""}.fa-align-right:before{content:""}.fa-align-justify:before{content:""}.fa-list:before{content:""}.fa-dedent:before,.fa-outdent:before{content:""}.fa-indent:before{content:""}.fa-video-camera:before{content:""}.fa-photo:before,.fa-image:before,.fa-picture-o:before{content:""}.fa-pencil:before{content:""}.fa-map-marker:before{content:""}.fa-adjust:before{content:""}.fa-tint:before{content:""}.fa-edit:before,.fa-pencil-square-o:before{content:""}.fa-share-square-o:before{content:""}.fa-check-square-o:before{content:""}.fa-arrows:before{content:""}.fa-step-backward:before{content:""}.fa-fast-backward:before{content:""}.fa-backward:before{content:""}.fa-play:before{content:""}.fa-pause:before{content:""}.fa-stop:before{content:""}.fa-forward:before{content:""}.fa-fast-forward:before{content:""}.fa-step-forward:before{content:""}.fa-eject:before{content:""}.fa-chevron-left:before{content:""}.fa-chevron-right:before{content:""}.fa-plus-circle:before{content:""}.fa-minus-circle:before{content:""}.fa-times-circle:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before{content:""}.fa-check-circle:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before{content:""}.fa-question-circle:before{content:""}.fa-info-circle:before{content:""}.fa-crosshairs:before{content:""}.fa-times-circle-o:before{content:""}.fa-check-circle-o:before{content:""}.fa-ban:before{content:""}.fa-arrow-left:before{content:""}.fa-arrow-right:before{content:""}.fa-arrow-up:before{content:""}.fa-arrow-down:before{content:""}.fa-mail-forward:before,.fa-share:before{content:""}.fa-expand:before{content:""}.fa-compress:before{content:""}.fa-plus:before{content:""}.fa-minus:before{content:""}.fa-asterisk:before{content:""}.fa-exclamation-circle:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before,.rst-content .admonition-title:before{content:""}.fa-gift:before{content:""}.fa-leaf:before{content:""}.fa-fire:before,.icon-fire:before{content:""}.fa-eye:before{content:""}.fa-eye-slash:before{content:""}.fa-warning:before,.fa-exclamation-triangle:before{content:""}.fa-plane:before{content:""}.fa-calendar:before{content:""}.fa-random:before{content:""}.fa-comment:before{content:""}.fa-magnet:before{content:""}.fa-chevron-up:before{content:""}.fa-chevron-down:before{content:""}.fa-retweet:before{content:""}.fa-shopping-cart:before{content:""}.fa-folder:before{content:""}.fa-folder-open:before{content:""}.fa-arrows-v:before{content:""}.fa-arrows-h:before{content:""}.fa-bar-chart-o:before{content:""}.fa-twitter-square:before{content:""}.fa-facebook-square:before{content:""}.fa-camera-retro:before{content:""}.fa-key:before{content:""}.fa-gears:before,.fa-cogs:before{content:""}.fa-comments:before{content:""}.fa-thumbs-o-up:before{content:""}.fa-thumbs-o-down:before{content:""}.fa-star-half:before{content:""}.fa-heart-o:before{content:""}.fa-sign-out:before{content:""}.fa-linkedin-square:before{content:""}.fa-thumb-tack:before{content:""}.fa-external-link:before{content:""}.fa-sign-in:before{content:""}.fa-trophy:before{content:""}.fa-github-square:before{content:""}.fa-upload:before{content:""}.fa-lemon-o:before{content:""}.fa-phone:before{content:""}.fa-square-o:before{content:""}.fa-bookmark-o:before{content:""}.fa-phone-square:before{content:""}.fa-twitter:before{content:""}.fa-facebook:before{content:""}.fa-github:before,.icon-github:before{content:""}.fa-unlock:before{content:""}.fa-credit-card:before{content:""}.fa-rss:before{content:""}.fa-hdd-o:before{content:""}.fa-bullhorn:before{content:""}.fa-bell:before{content:""}.fa-certificate:before{content:""}.fa-hand-o-right:before{content:""}.fa-hand-o-left:before{content:""}.fa-hand-o-up:before{content:""}.fa-hand-o-down:before{content:""}.fa-arrow-circle-left:before,.icon-circle-arrow-left:before{content:""}.fa-arrow-circle-right:before,.icon-circle-arrow-right:before{content:""}.fa-arrow-circle-up:before{content:""}.fa-arrow-circle-down:before{content:""}.fa-globe:before{content:""}.fa-wrench:before{content:""}.fa-tasks:before{content:""}.fa-filter:before{content:""}.fa-briefcase:before{content:""}.fa-arrows-alt:before{content:""}.fa-group:before,.fa-users:before{content:""}.fa-chain:before,.fa-link:before,.icon-link:before{content:""}.fa-cloud:before{content:""}.fa-flask:before{content:""}.fa-cut:before,.fa-scissors:before{content:""}.fa-copy:before,.fa-files-o:before{content:""}.fa-paperclip:before{content:""}.fa-save:before,.fa-floppy-o:before{content:""}.fa-square:before{content:""}.fa-navicon:before,.fa-reorder:before,.fa-bars:before{content:""}.fa-list-ul:before{content:""}.fa-list-ol:before{content:""}.fa-strikethrough:before{content:""}.fa-underline:before{content:""}.fa-table:before{content:""}.fa-magic:before{content:""}.fa-truck:before{content:""}.fa-pinterest:before{content:""}.fa-pinterest-square:before{content:""}.fa-google-plus-square:before{content:""}.fa-google-plus:before{content:""}.fa-money:before{content:""}.fa-caret-down:before,.wy-dropdown .caret:before,.icon-caret-down:before{content:""}.fa-caret-up:before{content:""}.fa-caret-left:before{content:""}.fa-caret-right:before{content:""}.fa-columns:before{content:""}.fa-unsorted:before,.fa-sort:before{content:""}.fa-sort-down:before,.fa-sort-desc:before{content:""}.fa-sort-up:before,.fa-sort-asc:before{content:""}.fa-envelope:before{content:""}.fa-linkedin:before{content:""}.fa-rotate-left:before,.fa-undo:before{content:""}.fa-legal:before,.fa-gavel:before{content:""}.fa-dashboard:before,.fa-tachometer:before{content:""}.fa-comment-o:before{content:""}.fa-comments-o:before{content:""}.fa-flash:before,.fa-bolt:before{content:""}.fa-sitemap:before{content:""}.fa-umbrella:before{content:""}.fa-paste:before,.fa-clipboard:before{content:""}.fa-lightbulb-o:before{content:""}.fa-exchange:before{content:""}.fa-cloud-download:before{content:""}.fa-cloud-upload:before{content:""}.fa-user-md:before{content:""}.fa-stethoscope:before{content:""}.fa-suitcase:before{content:""}.fa-bell-o:before{content:""}.fa-coffee:before{content:""}.fa-cutlery:before{content:""}.fa-file-text-o:before{content:""}.fa-building-o:before{content:""}.fa-hospital-o:before{content:""}.fa-ambulance:before{content:""}.fa-medkit:before{content:""}.fa-fighter-jet:before{content:""}.fa-beer:before{content:""}.fa-h-square:before{content:""}.fa-plus-square:before{content:""}.fa-angle-double-left:before{content:""}.fa-angle-double-right:before{content:""}.fa-angle-double-up:before{content:""}.fa-angle-double-down:before{content:""}.fa-angle-left:before{content:""}.fa-angle-right:before{content:""}.fa-angle-up:before{content:""}.fa-angle-down:before{content:""}.fa-desktop:before{content:""}.fa-laptop:before{content:""}.fa-tablet:before{content:""}.fa-mobile-phone:before,.fa-mobile:before{content:""}.fa-circle-o:before{content:""}.fa-quote-left:before{content:""}.fa-quote-right:before{content:""}.fa-spinner:before{content:""}.fa-circle:before{content:""}.fa-mail-reply:before,.fa-reply:before{content:""}.fa-github-alt:before{content:""}.fa-folder-o:before{content:""}.fa-folder-open-o:before{content:""}.fa-smile-o:before{content:""}.fa-frown-o:before{content:""}.fa-meh-o:before{content:""}.fa-gamepad:before{content:""}.fa-keyboard-o:before{content:""}.fa-flag-o:before{content:""}.fa-flag-checkered:before{content:""}.fa-terminal:before{content:""}.fa-code:before{content:""}.fa-mail-reply-all:before,.fa-reply-all:before{content:""}.fa-star-half-empty:before,.fa-star-half-full:before,.fa-star-half-o:before{content:""}.fa-location-arrow:before{content:""}.fa-crop:before{content:""}.fa-code-fork:before{content:""}.fa-unlink:before,.fa-chain-broken:before{content:""}.fa-question:before{content:""}.fa-info:before{content:""}.fa-exclamation:before{content:""}.fa-superscript:before{content:""}.fa-subscript:before{content:""}.fa-eraser:before{content:""}.fa-puzzle-piece:before{content:""}.fa-microphone:before{content:""}.fa-microphone-slash:before{content:""}.fa-shield:before{content:""}.fa-calendar-o:before{content:""}.fa-fire-extinguisher:before{content:""}.fa-rocket:before{content:""}.fa-maxcdn:before{content:""}.fa-chevron-circle-left:before{content:""}.fa-chevron-circle-right:before{content:""}.fa-chevron-circle-up:before{content:""}.fa-chevron-circle-down:before{content:""}.fa-html5:before{content:""}.fa-css3:before{content:""}.fa-anchor:before{content:""}.fa-unlock-alt:before{content:""}.fa-bullseye:before{content:""}.fa-ellipsis-h:before{content:""}.fa-ellipsis-v:before{content:""}.fa-rss-square:before{content:""}.fa-play-circle:before{content:""}.fa-ticket:before{content:""}.fa-minus-square:before{content:""}.fa-minus-square-o:before{content:""}.fa-level-up:before{content:""}.fa-level-down:before{content:""}.fa-check-square:before{content:""}.fa-pencil-square:before{content:""}.fa-external-link-square:before{content:""}.fa-share-square:before{content:""}.fa-compass:before{content:""}.fa-toggle-down:before,.fa-caret-square-o-down:before{content:""}.fa-toggle-up:before,.fa-caret-square-o-up:before{content:""}.fa-toggle-right:before,.fa-caret-square-o-right:before{content:""}.fa-euro:before,.fa-eur:before{content:""}.fa-gbp:before{content:""}.fa-dollar:before,.fa-usd:before{content:""}.fa-rupee:before,.fa-inr:before{content:""}.fa-cny:before,.fa-rmb:before,.fa-yen:before,.fa-jpy:before{content:""}.fa-ruble:before,.fa-rouble:before,.fa-rub:before{content:""}.fa-won:before,.fa-krw:before{content:""}.fa-bitcoin:before,.fa-btc:before{content:""}.fa-file:before{content:""}.fa-file-text:before{content:""}.fa-sort-alpha-asc:before{content:""}.fa-sort-alpha-desc:before{content:""}.fa-sort-amount-asc:before{content:""}.fa-sort-amount-desc:before{content:""}.fa-sort-numeric-asc:before{content:""}.fa-sort-numeric-desc:before{content:""}.fa-thumbs-up:before{content:""}.fa-thumbs-down:before{content:""}.fa-youtube-square:before{content:""}.fa-youtube:before{content:""}.fa-xing:before{content:""}.fa-xing-square:before{content:""}.fa-youtube-play:before{content:""}.fa-dropbox:before{content:""}.fa-stack-overflow:before{content:""}.fa-instagram:before{content:""}.fa-flickr:before{content:""}.fa-adn:before{content:""}.fa-bitbucket:before,.icon-bitbucket:before{content:""}.fa-bitbucket-square:before{content:""}.fa-tumblr:before{content:""}.fa-tumblr-square:before{content:""}.fa-long-arrow-down:before{content:""}.fa-long-arrow-up:before{content:""}.fa-long-arrow-left:before{content:""}.fa-long-arrow-right:before{content:""}.fa-apple:before{content:""}.fa-windows:before{content:""}.fa-android:before{content:""}.fa-linux:before{content:""}.fa-dribbble:before{content:""}.fa-skype:before{content:""}.fa-foursquare:before{content:""}.fa-trello:before{content:""}.fa-female:before{content:""}.fa-male:before{content:""}.fa-gittip:before{content:""}.fa-sun-o:before{content:""}.fa-moon-o:before{content:""}.fa-archive:before{content:""}.fa-bug:before{content:""}.fa-vk:before{content:""}.fa-weibo:before{content:""}.fa-renren:before{content:""}.fa-pagelines:before{content:""}.fa-stack-exchange:before{content:""}.fa-arrow-circle-o-right:before{content:""}.fa-arrow-circle-o-left:before{content:""}.fa-toggle-left:before,.fa-caret-square-o-left:before{content:""}.fa-dot-circle-o:before{content:""}.fa-wheelchair:before{content:""}.fa-vimeo-square:before{content:""}.fa-turkish-lira:before,.fa-try:before{content:""}.fa-plus-square-o:before{content:""}.fa-space-shuttle:before{content:""}.fa-slack:before{content:""}.fa-envelope-square:before{content:""}.fa-wordpress:before{content:""}.fa-openid:before{content:""}.fa-institution:before,.fa-bank:before,.fa-university:before{content:""}.fa-mortar-board:before,.fa-graduation-cap:before{content:""}.fa-yahoo:before{content:""}.fa-google:before{content:""}.fa-reddit:before{content:""}.fa-reddit-square:before{content:""}.fa-stumbleupon-circle:before{content:""}.fa-stumbleupon:before{content:""}.fa-delicious:before{content:""}.fa-digg:before{content:""}.fa-pied-piper-square:before,.fa-pied-piper:before{content:""}.fa-pied-piper-alt:before{content:""}.fa-drupal:before{content:""}.fa-joomla:before{content:""}.fa-language:before{content:""}.fa-fax:before{content:""}.fa-building:before{content:""}.fa-child:before{content:""}.fa-paw:before{content:""}.fa-spoon:before{content:""}.fa-cube:before{content:""}.fa-cubes:before{content:""}.fa-behance:before{content:""}.fa-behance-square:before{content:""}.fa-steam:before{content:""}.fa-steam-square:before{content:""}.fa-recycle:before{content:""}.fa-automobile:before,.fa-car:before{content:""}.fa-cab:before,.fa-taxi:before{content:""}.fa-tree:before{content:""}.fa-spotify:before{content:""}.fa-deviantart:before{content:""}.fa-soundcloud:before{content:""}.fa-database:before{content:""}.fa-file-pdf-o:before{content:""}.fa-file-word-o:before{content:""}.fa-file-excel-o:before{content:""}.fa-file-powerpoint-o:before{content:""}.fa-file-photo-o:before,.fa-file-picture-o:before,.fa-file-image-o:before{content:""}.fa-file-zip-o:before,.fa-file-archive-o:before{content:""}.fa-file-sound-o:before,.fa-file-audio-o:before{content:""}.fa-file-movie-o:before,.fa-file-video-o:before{content:""}.fa-file-code-o:before{content:""}.fa-vine:before{content:""}.fa-codepen:before{content:""}.fa-jsfiddle:before{content:""}.fa-life-bouy:before,.fa-life-saver:before,.fa-support:before,.fa-life-ring:before{content:""}.fa-circle-o-notch:before{content:""}.fa-ra:before,.fa-rebel:before{content:""}.fa-ge:before,.fa-empire:before{content:""}.fa-git-square:before{content:""}.fa-git:before{content:""}.fa-hacker-news:before{content:""}.fa-tencent-weibo:before{content:""}.fa-qq:before{content:""}.fa-wechat:before,.fa-weixin:before{content:""}.fa-send:before,.fa-paper-plane:before{content:""}.fa-send-o:before,.fa-paper-plane-o:before{content:""}.fa-history:before{content:""}.fa-circle-thin:before{content:""}.fa-header:before{content:""}.fa-paragraph:before{content:""}.fa-sliders:before{content:""}.fa-share-alt:before{content:""}.fa-share-alt-square:before{content:""}.fa-bomb:before{content:""}.fa,.rst-content .admonition-title,.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink,.icon,.wy-dropdown .caret,.wy-inline-validate.wy-inline-validate-success .wy-input-context,.wy-inline-validate.wy-inline-validate-danger .wy-input-context,.wy-inline-validate.wy-inline-validate-warning .wy-input-context,.wy-inline-validate.wy-inline-validate-info .wy-input-context{font-family:inherit}.fa:before,.rst-content .admonition-title:before,.rst-content h1 .headerlink:before,.rst-content h2 .headerlink:before,.rst-content h3 .headerlink:before,.rst-content h4 .headerlink:before,.rst-content h5 .headerlink:before,.rst-content h6 .headerlink:before,.rst-content dl dt .headerlink:before,.icon:before,.wy-dropdown .caret:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before{font-family:"FontAwesome";display:inline-block;font-style:normal;font-weight:normal;line-height:1;text-decoration:inherit}a .fa,a .rst-content .admonition-title,.rst-content a .admonition-title,a .rst-content h1 .headerlink,.rst-content h1 a .headerlink,a .rst-content h2 .headerlink,.rst-content h2 a .headerlink,a .rst-content h3 .headerlink,.rst-content h3 a .headerlink,a .rst-content h4 .headerlink,.rst-content h4 a .headerlink,a .rst-content h5 .headerlink,.rst-content h5 a .headerlink,a .rst-content h6 .headerlink,.rst-content h6 a .headerlink,a .rst-content dl dt .headerlink,.rst-content dl dt a .headerlink,a .icon{display:inline-block;text-decoration:inherit}.btn .fa,.btn .rst-content .admonition-title,.rst-content .btn .admonition-title,.btn .rst-content h1 .headerlink,.rst-content h1 .btn .headerlink,.btn .rst-content h2 .headerlink,.rst-content h2 .btn .headerlink,.btn .rst-content h3 .headerlink,.rst-content h3 .btn .headerlink,.btn .rst-content h4 .headerlink,.rst-content h4 .btn .headerlink,.btn .rst-content h5 .headerlink,.rst-content h5 .btn .headerlink,.btn .rst-content h6 .headerlink,.rst-content h6 .btn .headerlink,.btn .rst-content dl dt .headerlink,.rst-content dl dt .btn .headerlink,.btn .icon,.nav .fa,.nav .rst-content .admonition-title,.rst-content .nav .admonition-title,.nav .rst-content h1 .headerlink,.rst-content h1 .nav .headerlink,.nav .rst-content h2 .headerlink,.rst-content h2 .nav .headerlink,.nav .rst-content h3 .headerlink,.rst-content h3 .nav .headerlink,.nav .rst-content h4 .headerlink,.rst-content h4 .nav .headerlink,.nav .rst-content h5 .headerlink,.rst-content h5 .nav .headerlink,.nav .rst-content h6 .headerlink,.rst-content h6 .nav .headerlink,.nav .rst-content dl dt .headerlink,.rst-content dl dt .nav .headerlink,.nav .icon{display:inline}.btn .fa.fa-large,.btn .rst-content .fa-large.admonition-title,.rst-content .btn .fa-large.admonition-title,.btn .rst-content h1 .fa-large.headerlink,.rst-content h1 .btn .fa-large.headerlink,.btn .rst-content h2 .fa-large.headerlink,.rst-content h2 .btn .fa-large.headerlink,.btn .rst-content h3 .fa-large.headerlink,.rst-content h3 .btn .fa-large.headerlink,.btn .rst-content h4 .fa-large.headerlink,.rst-content h4 .btn .fa-large.headerlink,.btn .rst-content h5 .fa-large.headerlink,.rst-content h5 .btn .fa-large.headerlink,.btn .rst-content h6 .fa-large.headerlink,.rst-content h6 .btn .fa-large.headerlink,.btn .rst-content dl dt .fa-large.headerlink,.rst-content dl dt .btn .fa-large.headerlink,.btn .fa-large.icon,.nav .fa.fa-large,.nav .rst-content .fa-large.admonition-title,.rst-content .nav .fa-large.admonition-title,.nav .rst-content h1 .fa-large.headerlink,.rst-content h1 .nav .fa-large.headerlink,.nav .rst-content h2 .fa-large.headerlink,.rst-content h2 .nav .fa-large.headerlink,.nav .rst-content h3 .fa-large.headerlink,.rst-content h3 .nav .fa-large.headerlink,.nav .rst-content h4 .fa-large.headerlink,.rst-content h4 .nav .fa-large.headerlink,.nav .rst-content h5 .fa-large.headerlink,.rst-content h5 .nav .fa-large.headerlink,.nav .rst-content h6 .fa-large.headerlink,.rst-content h6 .nav .fa-large.headerlink,.nav .rst-content dl dt .fa-large.headerlink,.rst-content dl dt .nav .fa-large.headerlink,.nav .fa-large.icon{line-height:0.9em}.btn .fa.fa-spin,.btn .rst-content .fa-spin.admonition-title,.rst-content .btn .fa-spin.admonition-title,.btn .rst-content h1 .fa-spin.headerlink,.rst-content h1 .btn .fa-spin.headerlink,.btn .rst-content h2 .fa-spin.headerlink,.rst-content h2 .btn .fa-spin.headerlink,.btn .rst-content h3 .fa-spin.headerlink,.rst-content h3 .btn .fa-spin.headerlink,.btn .rst-content h4 .fa-spin.headerlink,.rst-content h4 .btn .fa-spin.headerlink,.btn .rst-content h5 .fa-spin.headerlink,.rst-content h5 .btn .fa-spin.headerlink,.btn .rst-content h6 .fa-spin.headerlink,.rst-content h6 .btn .fa-spin.headerlink,.btn .rst-content dl dt .fa-spin.headerlink,.rst-content dl dt .btn .fa-spin.headerlink,.btn .fa-spin.icon,.nav .fa.fa-spin,.nav .rst-content .fa-spin.admonition-title,.rst-content .nav .fa-spin.admonition-title,.nav .rst-content h1 .fa-spin.headerlink,.rst-content h1 .nav .fa-spin.headerlink,.nav .rst-content h2 .fa-spin.headerlink,.rst-content h2 .nav .fa-spin.headerlink,.nav .rst-content h3 .fa-spin.headerlink,.rst-content h3 .nav .fa-spin.headerlink,.nav .rst-content h4 .fa-spin.headerlink,.rst-content h4 .nav .fa-spin.headerlink,.nav .rst-content h5 .fa-spin.headerlink,.rst-content h5 .nav .fa-spin.headerlink,.nav .rst-content h6 .fa-spin.headerlink,.rst-content h6 .nav .fa-spin.headerlink,.nav .rst-content dl dt .fa-spin.headerlink,.rst-content dl dt .nav .fa-spin.headerlink,.nav .fa-spin.icon{display:inline-block}.btn.fa:before,.rst-content .btn.admonition-title:before,.rst-content h1 .btn.headerlink:before,.rst-content h2 .btn.headerlink:before,.rst-content h3 .btn.headerlink:before,.rst-content h4 .btn.headerlink:before,.rst-content h5 .btn.headerlink:before,.rst-content h6 .btn.headerlink:before,.rst-content dl dt .btn.headerlink:before,.btn.icon:before{opacity:0.5;-webkit-transition:opacity 0.05s ease-in;-moz-transition:opacity 0.05s ease-in;transition:opacity 0.05s ease-in}.btn.fa:hover:before,.rst-content .btn.admonition-title:hover:before,.rst-content h1 .btn.headerlink:hover:before,.rst-content h2 .btn.headerlink:hover:before,.rst-content h3 .btn.headerlink:hover:before,.rst-content h4 .btn.headerlink:hover:before,.rst-content h5 .btn.headerlink:hover:before,.rst-content h6 .btn.headerlink:hover:before,.rst-content dl dt .btn.headerlink:hover:before,.btn.icon:hover:before{opacity:1}.btn-mini .fa:before,.btn-mini .rst-content .admonition-title:before,.rst-content .btn-mini .admonition-title:before,.btn-mini .rst-content h1 .headerlink:before,.rst-content h1 .btn-mini .headerlink:before,.btn-mini .rst-content h2 .headerlink:before,.rst-content h2 .btn-mini .headerlink:before,.btn-mini .rst-content h3 .headerlink:before,.rst-content h3 .btn-mini .headerlink:before,.btn-mini .rst-content h4 .headerlink:before,.rst-content h4 .btn-mini .headerlink:before,.btn-mini .rst-content h5 .headerlink:before,.rst-content h5 .btn-mini .headerlink:before,.btn-mini .rst-content h6 .headerlink:before,.rst-content h6 .btn-mini .headerlink:before,.btn-mini .rst-content dl dt .headerlink:before,.rst-content dl dt .btn-mini .headerlink:before,.btn-mini .icon:before{font-size:14px;vertical-align:-15%}.wy-alert,.rst-content .note,.rst-content .attention,.rst-content .caution,.rst-content .danger,.rst-content .error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .warning,.rst-content .seealso,.rst-content .admonition-todo{padding:12px;line-height:24px;margin-bottom:24px;background:#e7f2fa}.wy-alert-title,.rst-content .admonition-title{color:#fff;font-weight:bold;display:block;color:#fff;background:#6ab0de;margin:-12px;padding:6px 12px;margin-bottom:12px}.wy-alert.wy-alert-danger,.rst-content .wy-alert-danger.note,.rst-content .wy-alert-danger.attention,.rst-content .wy-alert-danger.caution,.rst-content .danger,.rst-content .error,.rst-content .wy-alert-danger.hint,.rst-content .wy-alert-danger.important,.rst-content .wy-alert-danger.tip,.rst-content .wy-alert-danger.warning,.rst-content .wy-alert-danger.seealso,.rst-content .wy-alert-danger.admonition-todo{background:#fdf3f2}.wy-alert.wy-alert-danger .wy-alert-title,.rst-content .wy-alert-danger.note .wy-alert-title,.rst-content .wy-alert-danger.attention .wy-alert-title,.rst-content .wy-alert-danger.caution .wy-alert-title,.rst-content .danger .wy-alert-title,.rst-content .error .wy-alert-title,.rst-content .wy-alert-danger.hint .wy-alert-title,.rst-content .wy-alert-danger.important .wy-alert-title,.rst-content .wy-alert-danger.tip .wy-alert-title,.rst-content .wy-alert-danger.warning .wy-alert-title,.rst-content .wy-alert-danger.seealso .wy-alert-title,.rst-content .wy-alert-danger.admonition-todo .wy-alert-title,.wy-alert.wy-alert-danger .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-danger .admonition-title,.rst-content .wy-alert-danger.note .admonition-title,.rst-content .wy-alert-danger.attention .admonition-title,.rst-content .wy-alert-danger.caution .admonition-title,.rst-content .danger .admonition-title,.rst-content .error .admonition-title,.rst-content .wy-alert-danger.hint .admonition-title,.rst-content .wy-alert-danger.important .admonition-title,.rst-content .wy-alert-danger.tip .admonition-title,.rst-content .wy-alert-danger.warning .admonition-title,.rst-content .wy-alert-danger.seealso .admonition-title,.rst-content .wy-alert-danger.admonition-todo .admonition-title{background:#f29f97}.wy-alert.wy-alert-warning,.rst-content .wy-alert-warning.note,.rst-content .attention,.rst-content .caution,.rst-content .wy-alert-warning.danger,.rst-content .wy-alert-warning.error,.rst-content .wy-alert-warning.hint,.rst-content .wy-alert-warning.important,.rst-content .wy-alert-warning.tip,.rst-content .warning,.rst-content .wy-alert-warning.seealso,.rst-content .admonition-todo{background:#ffedcc}.wy-alert.wy-alert-warning .wy-alert-title,.rst-content .wy-alert-warning.note .wy-alert-title,.rst-content .attention .wy-alert-title,.rst-content .caution .wy-alert-title,.rst-content .wy-alert-warning.danger .wy-alert-title,.rst-content .wy-alert-warning.error .wy-alert-title,.rst-content .wy-alert-warning.hint .wy-alert-title,.rst-content .wy-alert-warning.important .wy-alert-title,.rst-content .wy-alert-warning.tip .wy-alert-title,.rst-content .warning .wy-alert-title,.rst-content .wy-alert-warning.seealso .wy-alert-title,.rst-content .admonition-todo .wy-alert-title,.wy-alert.wy-alert-warning .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-warning .admonition-title,.rst-content .wy-alert-warning.note .admonition-title,.rst-content .attention .admonition-title,.rst-content .caution .admonition-title,.rst-content .wy-alert-warning.danger .admonition-title,.rst-content .wy-alert-warning.error .admonition-title,.rst-content .wy-alert-warning.hint .admonition-title,.rst-content .wy-alert-warning.important .admonition-title,.rst-content .wy-alert-warning.tip .admonition-title,.rst-content .warning .admonition-title,.rst-content .wy-alert-warning.seealso .admonition-title,.rst-content .admonition-todo .admonition-title{background:#f0b37e}.wy-alert.wy-alert-info,.rst-content .note,.rst-content .wy-alert-info.attention,.rst-content .wy-alert-info.caution,.rst-content .wy-alert-info.danger,.rst-content .wy-alert-info.error,.rst-content .wy-alert-info.hint,.rst-content .wy-alert-info.important,.rst-content .wy-alert-info.tip,.rst-content .wy-alert-info.warning,.rst-content .seealso,.rst-content .wy-alert-info.admonition-todo{background:#e7f2fa}.wy-alert.wy-alert-info .wy-alert-title,.rst-content .note .wy-alert-title,.rst-content .wy-alert-info.attention .wy-alert-title,.rst-content .wy-alert-info.caution .wy-alert-title,.rst-content .wy-alert-info.danger .wy-alert-title,.rst-content .wy-alert-info.error .wy-alert-title,.rst-content .wy-alert-info.hint .wy-alert-title,.rst-content .wy-alert-info.important .wy-alert-title,.rst-content .wy-alert-info.tip .wy-alert-title,.rst-content .wy-alert-info.warning .wy-alert-title,.rst-content .seealso .wy-alert-title,.rst-content .wy-alert-info.admonition-todo .wy-alert-title,.wy-alert.wy-alert-info .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-info .admonition-title,.rst-content .note .admonition-title,.rst-content .wy-alert-info.attention .admonition-title,.rst-content .wy-alert-info.caution .admonition-title,.rst-content .wy-alert-info.danger .admonition-title,.rst-content .wy-alert-info.error .admonition-title,.rst-content .wy-alert-info.hint .admonition-title,.rst-content .wy-alert-info.important .admonition-title,.rst-content .wy-alert-info.tip .admonition-title,.rst-content .wy-alert-info.warning .admonition-title,.rst-content .seealso .admonition-title,.rst-content .wy-alert-info.admonition-todo .admonition-title{background:#6ab0de}.wy-alert.wy-alert-success,.rst-content .wy-alert-success.note,.rst-content .wy-alert-success.attention,.rst-content .wy-alert-success.caution,.rst-content .wy-alert-success.danger,.rst-content .wy-alert-success.error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .wy-alert-success.warning,.rst-content .wy-alert-success.seealso,.rst-content .wy-alert-success.admonition-todo{background:#dbfaf4}.wy-alert.wy-alert-success .wy-alert-title,.rst-content .wy-alert-success.note .wy-alert-title,.rst-content .wy-alert-success.attention .wy-alert-title,.rst-content .wy-alert-success.caution .wy-alert-title,.rst-content .wy-alert-success.danger .wy-alert-title,.rst-content .wy-alert-success.error .wy-alert-title,.rst-content .hint .wy-alert-title,.rst-content .important .wy-alert-title,.rst-content .tip .wy-alert-title,.rst-content .wy-alert-success.warning .wy-alert-title,.rst-content .wy-alert-success.seealso .wy-alert-title,.rst-content .wy-alert-success.admonition-todo .wy-alert-title,.wy-alert.wy-alert-success .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-success .admonition-title,.rst-content .wy-alert-success.note .admonition-title,.rst-content .wy-alert-success.attention .admonition-title,.rst-content .wy-alert-success.caution .admonition-title,.rst-content .wy-alert-success.danger .admonition-title,.rst-content .wy-alert-success.error .admonition-title,.rst-content .hint .admonition-title,.rst-content .important .admonition-title,.rst-content .tip .admonition-title,.rst-content .wy-alert-success.warning .admonition-title,.rst-content .wy-alert-success.seealso .admonition-title,.rst-content .wy-alert-success.admonition-todo .admonition-title{background:#1abc9c}.wy-alert.wy-alert-neutral,.rst-content .wy-alert-neutral.note,.rst-content .wy-alert-neutral.attention,.rst-content .wy-alert-neutral.caution,.rst-content .wy-alert-neutral.danger,.rst-content .wy-alert-neutral.error,.rst-content .wy-alert-neutral.hint,.rst-content .wy-alert-neutral.important,.rst-content .wy-alert-neutral.tip,.rst-content .wy-alert-neutral.warning,.rst-content .wy-alert-neutral.seealso,.rst-content .wy-alert-neutral.admonition-todo{background:#f3f6f6}.wy-alert.wy-alert-neutral .wy-alert-title,.rst-content .wy-alert-neutral.note .wy-alert-title,.rst-content .wy-alert-neutral.attention .wy-alert-title,.rst-content .wy-alert-neutral.caution .wy-alert-title,.rst-content .wy-alert-neutral.danger .wy-alert-title,.rst-content .wy-alert-neutral.error .wy-alert-title,.rst-content .wy-alert-neutral.hint .wy-alert-title,.rst-content .wy-alert-neutral.important .wy-alert-title,.rst-content .wy-alert-neutral.tip .wy-alert-title,.rst-content .wy-alert-neutral.warning .wy-alert-title,.rst-content .wy-alert-neutral.seealso .wy-alert-title,.rst-content .wy-alert-neutral.admonition-todo .wy-alert-title,.wy-alert.wy-alert-neutral .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-neutral .admonition-title,.rst-content .wy-alert-neutral.note .admonition-title,.rst-content .wy-alert-neutral.attention .admonition-title,.rst-content .wy-alert-neutral.caution .admonition-title,.rst-content .wy-alert-neutral.danger .admonition-title,.rst-content .wy-alert-neutral.error .admonition-title,.rst-content .wy-alert-neutral.hint .admonition-title,.rst-content .wy-alert-neutral.important .admonition-title,.rst-content .wy-alert-neutral.tip .admonition-title,.rst-content .wy-alert-neutral.warning .admonition-title,.rst-content .wy-alert-neutral.seealso .admonition-title,.rst-content .wy-alert-neutral.admonition-todo .admonition-title{color:#404040;background:#e1e4e5}.wy-alert.wy-alert-neutral a,.rst-content .wy-alert-neutral.note a,.rst-content .wy-alert-neutral.attention a,.rst-content .wy-alert-neutral.caution a,.rst-content .wy-alert-neutral.danger a,.rst-content .wy-alert-neutral.error a,.rst-content .wy-alert-neutral.hint a,.rst-content .wy-alert-neutral.important a,.rst-content .wy-alert-neutral.tip a,.rst-content .wy-alert-neutral.warning a,.rst-content .wy-alert-neutral.seealso a,.rst-content .wy-alert-neutral.admonition-todo a{color:#2980B9}.wy-alert p:last-child,.rst-content .note p:last-child,.rst-content .attention p:last-child,.rst-content .caution p:last-child,.rst-content .danger p:last-child,.rst-content .error p:last-child,.rst-content .hint p:last-child,.rst-content .important p:last-child,.rst-content .tip p:last-child,.rst-content .warning p:last-child,.rst-content .seealso p:last-child,.rst-content .admonition-todo p:last-child{margin-bottom:0}.wy-tray-container{position:fixed;bottom:0px;left:0;z-index:600}.wy-tray-container li{display:block;width:300px;background:transparent;color:#fff;text-align:center;box-shadow:0 5px 5px 0 rgba(0,0,0,0.1);padding:0 24px;min-width:20%;opacity:0;height:0;line-height:56px;overflow:hidden;-webkit-transition:all 0.3s ease-in;-moz-transition:all 0.3s ease-in;transition:all 0.3s ease-in}.wy-tray-container li.wy-tray-item-success{background:#27AE60}.wy-tray-container li.wy-tray-item-info{background:#2980B9}.wy-tray-container li.wy-tray-item-warning{background:#E67E22}.wy-tray-container li.wy-tray-item-danger{background:#E74C3C}.wy-tray-container li.on{opacity:1;height:56px}@media screen and (max-width: 768px){.wy-tray-container{bottom:auto;top:0;width:100%}.wy-tray-container li{width:100%}}button{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle;cursor:pointer;line-height:normal;-webkit-appearance:button;*overflow:visible}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}button[disabled]{cursor:default}.btn{display:inline-block;border-radius:2px;line-height:normal;white-space:nowrap;text-align:center;cursor:pointer;font-size:100%;padding:6px 12px 8px 12px;color:#fff;border:1px solid rgba(0,0,0,0.1);background-color:#27AE60;text-decoration:none;font-weight:normal;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;box-shadow:0px 1px 2px -1px rgba(255,255,255,0.5) inset,0px -2px 0px 0px rgba(0,0,0,0.1) inset;outline-none:false;vertical-align:middle;*display:inline;zoom:1;-webkit-user-drag:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;-webkit-transition:all 0.1s linear;-moz-transition:all 0.1s linear;transition:all 0.1s linear}.btn-hover{background:#2e8ece;color:#fff}.btn:hover{background:#2cc36b;color:#fff}.btn:focus{background:#2cc36b;outline:0}.btn:active{box-shadow:0px -1px 0px 0px rgba(0,0,0,0.05) inset,0px 2px 0px 0px rgba(0,0,0,0.1) inset;padding:8px 12px 6px 12px}.btn:visited{color:#fff}.btn:disabled{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn-disabled{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn-disabled:hover,.btn-disabled:focus,.btn-disabled:active{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn::-moz-focus-inner{padding:0;border:0}.btn-small{font-size:80%}.btn-info{background-color:#2980B9 !important}.btn-info:hover{background-color:#2e8ece !important}.btn-neutral{background-color:#f3f6f6 !important;color:#404040 !important}.btn-neutral:hover{background-color:#e5ebeb !important;color:#404040}.btn-neutral:visited{color:#404040 !important}.btn-success{background-color:#27AE60 !important}.btn-success:hover{background-color:#295 !important}.btn-danger{background-color:#E74C3C !important}.btn-danger:hover{background-color:#ea6153 !important}.btn-warning{background-color:#E67E22 !important}.btn-warning:hover{background-color:#e98b39 !important}.btn-invert{background-color:#222}.btn-invert:hover{background-color:#2f2f2f !important}.btn-link{background-color:transparent !important;color:#2980B9;box-shadow:none;border-color:transparent !important}.btn-link:hover{background-color:transparent !important;color:#409ad5 !important;box-shadow:none}.btn-link:active{background-color:transparent !important;color:#409ad5 !important;box-shadow:none}.btn-link:visited{color:#9B59B6}.wy-btn-group .btn,.wy-control .btn{vertical-align:middle}.wy-btn-group{margin-bottom:24px;*zoom:1}.wy-btn-group:before,.wy-btn-group:after{display:table;content:""}.wy-btn-group:after{clear:both}.wy-dropdown{position:relative;display:inline-block}.wy-dropdown-menu{position:absolute;left:0;display:none;float:left;top:100%;min-width:100%;background:#fcfcfc;z-index:100;border:solid 1px #cfd7dd;box-shadow:0 2px 2px 0 rgba(0,0,0,0.1);padding:12px}.wy-dropdown-menu>dd>a{display:block;clear:both;color:#404040;white-space:nowrap;font-size:90%;padding:0 12px;cursor:pointer}.wy-dropdown-menu>dd>a:hover{background:#2980B9;color:#fff}.wy-dropdown-menu>dd.divider{border-top:solid 1px #cfd7dd;margin:6px 0}.wy-dropdown-menu>dd.search{padding-bottom:12px}.wy-dropdown-menu>dd.search input[type="search"]{width:100%}.wy-dropdown-menu>dd.call-to-action{background:#e3e3e3;text-transform:uppercase;font-weight:500;font-size:80%}.wy-dropdown-menu>dd.call-to-action:hover{background:#e3e3e3}.wy-dropdown-menu>dd.call-to-action .btn{color:#fff}.wy-dropdown.wy-dropdown-up .wy-dropdown-menu{bottom:100%;top:auto;left:auto;right:0}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu{background:#fcfcfc;margin-top:2px}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu a{padding:6px 12px}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu a:hover{background:#2980B9;color:#fff}.wy-dropdown.wy-dropdown-left .wy-dropdown-menu{right:0;text-align:right}.wy-dropdown-arrow:before{content:" ";border-bottom:5px solid #f5f5f5;border-left:5px solid transparent;border-right:5px solid transparent;position:absolute;display:block;top:-4px;left:50%;margin-left:-3px}.wy-dropdown-arrow.wy-dropdown-arrow-left:before{left:11px}.wy-form-stacked select{display:block}.wy-form-aligned input,.wy-form-aligned textarea,.wy-form-aligned select,.wy-form-aligned .wy-help-inline,.wy-form-aligned label{display:inline-block;*display:inline;*zoom:1;vertical-align:middle}.wy-form-aligned .wy-control-group>label{display:inline-block;vertical-align:middle;width:10em;margin:6px 12px 0 0;float:left}.wy-form-aligned .wy-control{float:left}.wy-form-aligned .wy-control label{display:block}.wy-form-aligned .wy-control select{margin-top:6px}fieldset{border:0;margin:0;padding:0}legend{display:block;width:100%;border:0;padding:0;white-space:normal;margin-bottom:24px;font-size:150%;*margin-left:-7px}label{display:block;margin:0 0 0.3125em 0;color:#999;font-size:90%}input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle}.wy-control-group{margin-bottom:24px;*zoom:1;max-width:68em;margin-left:auto;margin-right:auto;*zoom:1}.wy-control-group:before,.wy-control-group:after{display:table;content:""}.wy-control-group:after{clear:both}.wy-control-group:before,.wy-control-group:after{display:table;content:""}.wy-control-group:after{clear:both}.wy-control-group.wy-control-group-required>label:after{content:" *";color:#E74C3C}.wy-control-group .wy-form-full,.wy-control-group .wy-form-halves,.wy-control-group .wy-form-thirds{padding-bottom:12px}.wy-control-group .wy-form-full select,.wy-control-group .wy-form-halves select,.wy-control-group .wy-form-thirds select{width:100%}.wy-control-group .wy-form-full input[type="text"],.wy-control-group .wy-form-full input[type="password"],.wy-control-group .wy-form-full input[type="email"],.wy-control-group .wy-form-full input[type="url"],.wy-control-group .wy-form-full input[type="date"],.wy-control-group .wy-form-full input[type="month"],.wy-control-group .wy-form-full input[type="time"],.wy-control-group .wy-form-full input[type="datetime"],.wy-control-group .wy-form-full input[type="datetime-local"],.wy-control-group .wy-form-full input[type="week"],.wy-control-group .wy-form-full input[type="number"],.wy-control-group .wy-form-full input[type="search"],.wy-control-group .wy-form-full input[type="tel"],.wy-control-group .wy-form-full input[type="color"],.wy-control-group .wy-form-halves input[type="text"],.wy-control-group .wy-form-halves input[type="password"],.wy-control-group .wy-form-halves input[type="email"],.wy-control-group .wy-form-halves input[type="url"],.wy-control-group .wy-form-halves input[type="date"],.wy-control-group .wy-form-halves input[type="month"],.wy-control-group .wy-form-halves input[type="time"],.wy-control-group .wy-form-halves input[type="datetime"],.wy-control-group .wy-form-halves input[type="datetime-local"],.wy-control-group .wy-form-halves input[type="week"],.wy-control-group .wy-form-halves input[type="number"],.wy-control-group .wy-form-halves input[type="search"],.wy-control-group .wy-form-halves input[type="tel"],.wy-control-group .wy-form-halves input[type="color"],.wy-control-group .wy-form-thirds input[type="text"],.wy-control-group .wy-form-thirds input[type="password"],.wy-control-group .wy-form-thirds input[type="email"],.wy-control-group .wy-form-thirds input[type="url"],.wy-control-group .wy-form-thirds input[type="date"],.wy-control-group .wy-form-thirds input[type="month"],.wy-control-group .wy-form-thirds input[type="time"],.wy-control-group .wy-form-thirds input[type="datetime"],.wy-control-group .wy-form-thirds input[type="datetime-local"],.wy-control-group .wy-form-thirds input[type="week"],.wy-control-group .wy-form-thirds input[type="number"],.wy-control-group .wy-form-thirds input[type="search"],.wy-control-group .wy-form-thirds input[type="tel"],.wy-control-group .wy-form-thirds input[type="color"]{width:100%}.wy-control-group .wy-form-full{float:left;display:block;margin-right:2.35765%;width:100%;margin-right:0}.wy-control-group .wy-form-full:last-child{margin-right:0}.wy-control-group .wy-form-halves{float:left;display:block;margin-right:2.35765%;width:48.82117%}.wy-control-group .wy-form-halves:last-child{margin-right:0}.wy-control-group .wy-form-halves:nth-of-type(2n){margin-right:0}.wy-control-group .wy-form-halves:nth-of-type(2n+1){clear:left}.wy-control-group .wy-form-thirds{float:left;display:block;margin-right:2.35765%;width:31.76157%}.wy-control-group .wy-form-thirds:last-child{margin-right:0}.wy-control-group .wy-form-thirds:nth-of-type(3n){margin-right:0}.wy-control-group .wy-form-thirds:nth-of-type(3n+1){clear:left}.wy-control-group.wy-control-group-no-input .wy-control{margin:6px 0 0 0;font-size:90%}.wy-control-no-input{display:inline-block;margin:6px 0 0 0;font-size:90%}.wy-control-group.fluid-input input[type="text"],.wy-control-group.fluid-input input[type="password"],.wy-control-group.fluid-input input[type="email"],.wy-control-group.fluid-input input[type="url"],.wy-control-group.fluid-input input[type="date"],.wy-control-group.fluid-input input[type="month"],.wy-control-group.fluid-input input[type="time"],.wy-control-group.fluid-input input[type="datetime"],.wy-control-group.fluid-input input[type="datetime-local"],.wy-control-group.fluid-input input[type="week"],.wy-control-group.fluid-input input[type="number"],.wy-control-group.fluid-input input[type="search"],.wy-control-group.fluid-input input[type="tel"],.wy-control-group.fluid-input input[type="color"]{width:100%}.wy-form-message-inline{display:inline-block;padding-left:0.3em;color:#666;vertical-align:middle;font-size:90%}.wy-form-message{display:block;color:#999;font-size:70%;margin-top:0.3125em;font-style:italic}input{line-height:normal}input[type="button"],input[type="reset"],input[type="submit"]{-webkit-appearance:button;cursor:pointer;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;*overflow:visible}input[type="text"],input[type="password"],input[type="email"],input[type="url"],input[type="date"],input[type="month"],input[type="time"],input[type="datetime"],input[type="datetime-local"],input[type="week"],input[type="number"],input[type="search"],input[type="tel"],input[type="color"]{-webkit-appearance:none;padding:6px;display:inline-block;border:1px solid #ccc;font-size:80%;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;box-shadow:inset 0 1px 3px #ddd;border-radius:0;-webkit-transition:border 0.3s linear;-moz-transition:border 0.3s linear;transition:border 0.3s linear}input[type="datetime-local"]{padding:0.34375em 0.625em}input[disabled]{cursor:default}input[type="checkbox"],input[type="radio"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0;margin-right:0.3125em;*height:13px;*width:13px}input[type="search"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}input[type="search"]::-webkit-search-cancel-button,input[type="search"]::-webkit-search-decoration{-webkit-appearance:none}input[type="text"]:focus,input[type="password"]:focus,input[type="email"]:focus,input[type="url"]:focus,input[type="date"]:focus,input[type="month"]:focus,input[type="time"]:focus,input[type="datetime"]:focus,input[type="datetime-local"]:focus,input[type="week"]:focus,input[type="number"]:focus,input[type="search"]:focus,input[type="tel"]:focus,input[type="color"]:focus{outline:0;outline:thin dotted \9;border-color:#333}input.no-focus:focus{border-color:#ccc !important}input[type="file"]:focus,input[type="radio"]:focus,input[type="checkbox"]:focus{outline:thin dotted #333;outline:1px auto #129FEA}input[type="text"][disabled],input[type="password"][disabled],input[type="email"][disabled],input[type="url"][disabled],input[type="date"][disabled],input[type="month"][disabled],input[type="time"][disabled],input[type="datetime"][disabled],input[type="datetime-local"][disabled],input[type="week"][disabled],input[type="number"][disabled],input[type="search"][disabled],input[type="tel"][disabled],input[type="color"][disabled]{cursor:not-allowed;background-color:#f3f6f6;color:#cad2d3}input:focus:invalid,textarea:focus:invalid,select:focus:invalid{color:#E74C3C;border:1px solid #E74C3C}input:focus:invalid:focus,textarea:focus:invalid:focus,select:focus:invalid:focus{border-color:#E74C3C}input[type="file"]:focus:invalid:focus,input[type="radio"]:focus:invalid:focus,input[type="checkbox"]:focus:invalid:focus{outline-color:#E74C3C}input.wy-input-large{padding:12px;font-size:100%}textarea{overflow:auto;vertical-align:top;width:100%;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif}select,textarea{padding:0.5em 0.625em;display:inline-block;border:1px solid #ccc;font-size:80%;box-shadow:inset 0 1px 3px #ddd;-webkit-transition:border 0.3s linear;-moz-transition:border 0.3s linear;transition:border 0.3s linear}select{border:1px solid #ccc;background-color:#fff}select[multiple]{height:auto}select:focus,textarea:focus{outline:0}select[disabled],textarea[disabled],input[readonly],select[readonly],textarea[readonly]{cursor:not-allowed;background-color:#fff;color:#cad2d3;border-color:transparent}.wy-checkbox,.wy-radio{margin:6px 0;color:#404040;display:block}.wy-checkbox input,.wy-radio input{vertical-align:baseline}.wy-form-message-inline{display:inline-block;*display:inline;*zoom:1;vertical-align:middle}.wy-input-prefix,.wy-input-suffix{white-space:nowrap}.wy-input-prefix .wy-input-context,.wy-input-suffix .wy-input-context{padding:6px;display:inline-block;font-size:80%;background-color:#f3f6f6;border:solid 1px #ccc;color:#999}.wy-input-suffix .wy-input-context{border-left:0}.wy-input-prefix .wy-input-context{border-right:0}.wy-control-group.wy-control-group-error .wy-form-message,.wy-control-group.wy-control-group-error>label{color:#E74C3C}.wy-control-group.wy-control-group-error input[type="text"],.wy-control-group.wy-control-group-error input[type="password"],.wy-control-group.wy-control-group-error input[type="email"],.wy-control-group.wy-control-group-error input[type="url"],.wy-control-group.wy-control-group-error input[type="date"],.wy-control-group.wy-control-group-error input[type="month"],.wy-control-group.wy-control-group-error input[type="time"],.wy-control-group.wy-control-group-error input[type="datetime"],.wy-control-group.wy-control-group-error input[type="datetime-local"],.wy-control-group.wy-control-group-error input[type="week"],.wy-control-group.wy-control-group-error input[type="number"],.wy-control-group.wy-control-group-error input[type="search"],.wy-control-group.wy-control-group-error input[type="tel"],.wy-control-group.wy-control-group-error input[type="color"]{border:solid 1px #E74C3C}.wy-control-group.wy-control-group-error textarea{border:solid 1px #E74C3C}.wy-inline-validate{white-space:nowrap}.wy-inline-validate .wy-input-context{padding:0.5em 0.625em;display:inline-block;font-size:80%}.wy-inline-validate.wy-inline-validate-success .wy-input-context{color:#27AE60}.wy-inline-validate.wy-inline-validate-danger .wy-input-context{color:#E74C3C}.wy-inline-validate.wy-inline-validate-warning .wy-input-context{color:#E67E22}.wy-inline-validate.wy-inline-validate-info .wy-input-context{color:#2980B9}.rotate-90{-webkit-transform:rotate(90deg);-moz-transform:rotate(90deg);-ms-transform:rotate(90deg);-o-transform:rotate(90deg);transform:rotate(90deg)}.rotate-180{-webkit-transform:rotate(180deg);-moz-transform:rotate(180deg);-ms-transform:rotate(180deg);-o-transform:rotate(180deg);transform:rotate(180deg)}.rotate-270{-webkit-transform:rotate(270deg);-moz-transform:rotate(270deg);-ms-transform:rotate(270deg);-o-transform:rotate(270deg);transform:rotate(270deg)}.mirror{-webkit-transform:scaleX(-1);-moz-transform:scaleX(-1);-ms-transform:scaleX(-1);-o-transform:scaleX(-1);transform:scaleX(-1)}.mirror.rotate-90{-webkit-transform:scaleX(-1) rotate(90deg);-moz-transform:scaleX(-1) rotate(90deg);-ms-transform:scaleX(-1) rotate(90deg);-o-transform:scaleX(-1) rotate(90deg);transform:scaleX(-1) rotate(90deg)}.mirror.rotate-180{-webkit-transform:scaleX(-1) rotate(180deg);-moz-transform:scaleX(-1) rotate(180deg);-ms-transform:scaleX(-1) rotate(180deg);-o-transform:scaleX(-1) rotate(180deg);transform:scaleX(-1) rotate(180deg)}.mirror.rotate-270{-webkit-transform:scaleX(-1) rotate(270deg);-moz-transform:scaleX(-1) rotate(270deg);-ms-transform:scaleX(-1) rotate(270deg);-o-transform:scaleX(-1) rotate(270deg);transform:scaleX(-1) rotate(270deg)}@media only screen and (max-width: 480px){.wy-form button[type="submit"]{margin:0.7em 0 0}.wy-form input[type="text"],.wy-form input[type="password"],.wy-form input[type="email"],.wy-form input[type="url"],.wy-form input[type="date"],.wy-form input[type="month"],.wy-form input[type="time"],.wy-form input[type="datetime"],.wy-form input[type="datetime-local"],.wy-form input[type="week"],.wy-form input[type="number"],.wy-form input[type="search"],.wy-form input[type="tel"],.wy-form input[type="color"]{margin-bottom:0.3em;display:block}.wy-form label{margin-bottom:0.3em;display:block}.wy-form input[type="password"],.wy-form input[type="email"],.wy-form input[type="url"],.wy-form input[type="date"],.wy-form input[type="month"],.wy-form input[type="time"],.wy-form input[type="datetime"],.wy-form input[type="datetime-local"],.wy-form input[type="week"],.wy-form input[type="number"],.wy-form input[type="search"],.wy-form input[type="tel"],.wy-form input[type="color"]{margin-bottom:0}.wy-form-aligned .wy-control-group label{margin-bottom:0.3em;text-align:left;display:block;width:100%}.wy-form-aligned .wy-control{margin:1.5em 0 0 0}.wy-form .wy-help-inline,.wy-form-message-inline,.wy-form-message{display:block;font-size:80%;padding:6px 0}}@media screen and (max-width: 768px){.tablet-hide{display:none}}@media screen and (max-width: 480px){.mobile-hide{display:none}}.float-left{float:left}.float-right{float:right}.full-width{width:100%}.wy-table,.rst-content table.docutils,.rst-content table.field-list{border-collapse:collapse;border-spacing:0;empty-cells:show;margin-bottom:24px}.wy-table caption,.rst-content table.docutils caption,.rst-content table.field-list caption{color:#000;font:italic 85%/1 arial,sans-serif;padding:1em 0;text-align:center}.wy-table td,.rst-content table.docutils td,.rst-content table.field-list td,.wy-table th,.rst-content table.docutils th,.rst-content table.field-list th{font-size:90%;margin:0;overflow:visible;padding:8px 16px}.wy-table td:first-child,.rst-content table.docutils td:first-child,.rst-content table.field-list td:first-child,.wy-table th:first-child,.rst-content table.docutils th:first-child,.rst-content table.field-list th:first-child{border-left-width:0}.wy-table thead,.rst-content table.docutils thead,.rst-content table.field-list thead{color:#000;text-align:left;vertical-align:bottom;white-space:nowrap}.wy-table thead th,.rst-content table.docutils thead th,.rst-content table.field-list thead th{font-weight:bold;border-bottom:solid 2px #e1e4e5}.wy-table td,.rst-content table.docutils td,.rst-content table.field-list td{background-color:transparent;vertical-align:middle}.wy-table td p,.rst-content table.docutils td p,.rst-content table.field-list td p{line-height:18px}.wy-table td p:last-child,.rst-content table.docutils td p:last-child,.rst-content table.field-list td p:last-child{margin-bottom:0}.wy-table .wy-table-cell-min,.rst-content table.docutils .wy-table-cell-min,.rst-content table.field-list .wy-table-cell-min{width:1%;padding-right:0}.wy-table .wy-table-cell-min input[type=checkbox],.rst-content table.docutils .wy-table-cell-min input[type=checkbox],.rst-content table.field-list .wy-table-cell-min input[type=checkbox],.wy-table .wy-table-cell-min input[type=checkbox],.rst-content table.docutils .wy-table-cell-min input[type=checkbox],.rst-content table.field-list .wy-table-cell-min input[type=checkbox]{margin:0}.wy-table-secondary{color:gray;font-size:90%}.wy-table-tertiary{color:gray;font-size:80%}.wy-table-odd td,.wy-table-striped tr:nth-child(2n-1) td,.rst-content table.docutils:not(.field-list) tr:nth-child(2n-1) td{background-color:#f3f6f6}.wy-table-backed{background-color:#f3f6f6}.wy-table-bordered-all,.rst-content table.docutils{border:1px solid #e1e4e5}.wy-table-bordered-all td,.rst-content table.docutils td{border-bottom:1px solid #e1e4e5;border-left:1px solid #e1e4e5}.wy-table-bordered-all tbody>tr:last-child td,.rst-content table.docutils tbody>tr:last-child td{border-bottom-width:0}.wy-table-bordered{border:1px solid #e1e4e5}.wy-table-bordered-rows td{border-bottom:1px solid #e1e4e5}.wy-table-bordered-rows tbody>tr:last-child td{border-bottom-width:0}.wy-table-horizontal tbody>tr:last-child td{border-bottom-width:0}.wy-table-horizontal td,.wy-table-horizontal th{border-width:0 0 1px 0;border-bottom:1px solid #e1e4e5}.wy-table-horizontal tbody>tr:last-child td{border-bottom-width:0}.wy-table-responsive{margin-bottom:24px;max-width:100%;overflow:auto}.wy-table-responsive table{margin-bottom:0 !important}.wy-table-responsive table td,.wy-table-responsive table th{white-space:nowrap}a{color:#2980B9;text-decoration:none}a:hover{color:#3091d1}a:visited{color:#9B59B6}html{height:100%;overflow-x:hidden}body{font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;font-weight:normal;color:#404040;min-height:100%;overflow-x:hidden;background:#edf0f2}.wy-text-left{text-align:left}.wy-text-center{text-align:center}.wy-text-right{text-align:right}.wy-text-large{font-size:120%}.wy-text-normal{font-size:100%}.wy-text-small,small{font-size:80%}.wy-text-strike{text-decoration:line-through}.wy-text-warning{color:#E67E22 !important}a.wy-text-warning:hover{color:#eb9950 !important}.wy-text-info{color:#2980B9 !important}a.wy-text-info:hover{color:#409ad5 !important}.wy-text-success{color:#27AE60 !important}a.wy-text-success:hover{color:#36d278 !important}.wy-text-danger{color:#E74C3C !important}a.wy-text-danger:hover{color:#ed7669 !important}.wy-text-neutral{color:#404040 !important}a.wy-text-neutral:hover{color:#595959 !important}h1,h2,h3,h4,h5,h6,legend{margin-top:0;font-weight:700;font-family:"Roboto Slab","ff-tisa-web-pro","Georgia",Arial,sans-serif}p{line-height:24px;margin:0;font-size:16px;margin-bottom:24px}h1{font-size:175%}h2{font-size:150%}h3{font-size:125%}h4{font-size:115%}h5{font-size:110%}h6{font-size:100%}hr{display:block;height:1px;border:0;border-top:1px solid #e1e4e5;margin:24px 0;padding:0}code,.rst-content tt{white-space:nowrap;max-width:100%;background:#fff;border:solid 1px #e1e4e5;font-size:75%;padding:0 5px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;color:#E74C3C;overflow-x:auto}code.code-large,.rst-content tt.code-large{font-size:90%}.wy-plain-list-disc,.rst-content .section ul,.rst-content .toctree-wrapper ul,article ul{list-style:disc;line-height:24px;margin-bottom:24px}.wy-plain-list-disc li,.rst-content .section ul li,.rst-content .toctree-wrapper ul li,article ul li{list-style:disc;margin-left:24px}.wy-plain-list-disc li p:last-child,.rst-content .section ul li p:last-child,.rst-content .toctree-wrapper ul li p:last-child,article ul li p:last-child{margin-bottom:0}.wy-plain-list-disc li ul,.rst-content .section ul li ul,.rst-content .toctree-wrapper ul li ul,article ul li ul{margin-bottom:0}.wy-plain-list-disc li li,.rst-content .section ul li li,.rst-content .toctree-wrapper ul li li,article ul li li{list-style:circle}.wy-plain-list-disc li li li,.rst-content .section ul li li li,.rst-content .toctree-wrapper ul li li li,article ul li li li{list-style:square}.wy-plain-list-disc li ol li,.rst-content .section ul li ol li,.rst-content .toctree-wrapper ul li ol li,article ul li ol li{list-style:decimal}.wy-plain-list-decimal,.rst-content .section ol,.rst-content ol.arabic,article ol{list-style:decimal;line-height:24px;margin-bottom:24px}.wy-plain-list-decimal li,.rst-content .section ol li,.rst-content ol.arabic li,article ol li{list-style:decimal;margin-left:24px}.wy-plain-list-decimal li p:last-child,.rst-content .section ol li p:last-child,.rst-content ol.arabic li p:last-child,article ol li p:last-child{margin-bottom:0}.wy-plain-list-decimal li ul,.rst-content .section ol li ul,.rst-content ol.arabic li ul,article ol li ul{margin-bottom:0}.wy-plain-list-decimal li ul li,.rst-content .section ol li ul li,.rst-content ol.arabic li ul li,article ol li ul li{list-style:disc}.codeblock-example{border:1px solid #e1e4e5;border-bottom:none;padding:24px;padding-top:48px;font-weight:500;background:#fff;position:relative}.codeblock-example:after{content:"Example";position:absolute;top:0px;left:0px;background:#9B59B6;color:#fff;padding:6px 12px}.codeblock-example.prettyprint-example-only{border:1px solid #e1e4e5;margin-bottom:24px}.codeblock,pre.literal-block,.rst-content .literal-block,.rst-content pre.literal-block,div[class^='highlight']{border:1px solid #e1e4e5;padding:0px;overflow-x:auto;background:#fff;margin:1px 0 24px 0}.codeblock div[class^='highlight'],pre.literal-block div[class^='highlight'],.rst-content .literal-block div[class^='highlight'],div[class^='highlight'] div[class^='highlight']{border:none;background:none;margin:0}div[class^='highlight'] td.code{width:100%}.linenodiv pre{border-right:solid 1px #e6e9ea;margin:0;padding:12px 12px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;font-size:12px;line-height:1.5;color:#d9d9d9}div[class^='highlight'] pre{white-space:pre;margin:0;padding:12px 12px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;font-size:12px;line-height:1.5;display:block;overflow:auto;color:#404040}@media print{.codeblock,pre.literal-block,.rst-content .literal-block,.rst-content pre.literal-block,div[class^='highlight'],div[class^='highlight'] pre{white-space:pre-wrap}}.hll{background-color:#ffc;margin:0 -12px;padding:0 12px;display:block}.c{color:#998;font-style:italic}.err{color:#a61717;background-color:#e3d2d2}.k{font-weight:bold}.o{font-weight:bold}.cm{color:#998;font-style:italic}.cp{color:#999;font-weight:bold}.c1{color:#998;font-style:italic}.cs{color:#999;font-weight:bold;font-style:italic}.gd{color:#000;background-color:#fdd}.gd .x{color:#000;background-color:#faa}.ge{font-style:italic}.gr{color:#a00}.gh{color:#999}.gi{color:#000;background-color:#dfd}.gi .x{color:#000;background-color:#afa}.go{color:#888}.gp{color:#555}.gs{font-weight:bold}.gu{color:purple;font-weight:bold}.gt{color:#a00}.kc{font-weight:bold}.kd{font-weight:bold}.kn{font-weight:bold}.kp{font-weight:bold}.kr{font-weight:bold}.kt{color:#458;font-weight:bold}.m{color:#099}.s{color:#d14}.n{color:#333}.na{color:teal}.nb{color:#0086b3}.nc{color:#458;font-weight:bold}.no{color:teal}.ni{color:purple}.ne{color:#900;font-weight:bold}.nf{color:#900;font-weight:bold}.nn{color:#555}.nt{color:navy}.nv{color:teal}.ow{font-weight:bold}.w{color:#bbb}.mf{color:#099}.mh{color:#099}.mi{color:#099}.mo{color:#099}.sb{color:#d14}.sc{color:#d14}.sd{color:#d14}.s2{color:#d14}.se{color:#d14}.sh{color:#d14}.si{color:#d14}.sx{color:#d14}.sr{color:#009926}.s1{color:#d14}.ss{color:#990073}.bp{color:#999}.vc{color:teal}.vg{color:teal}.vi{color:teal}.il{color:#099}.gc{color:#999;background-color:#EAF2F5}.wy-breadcrumbs li{display:inline-block}.wy-breadcrumbs li.wy-breadcrumbs-aside{float:right}.wy-breadcrumbs li a{display:inline-block;padding:5px}.wy-breadcrumbs li a:first-child{padding-left:0}.wy-breadcrumbs-extra{margin-bottom:0;color:#b3b3b3;font-size:80%;display:inline-block}@media screen and (max-width: 480px){.wy-breadcrumbs-extra{display:none}.wy-breadcrumbs li.wy-breadcrumbs-aside{display:none}}@media print{.wy-breadcrumbs li.wy-breadcrumbs-aside{display:none}}.wy-affix{position:fixed;top:1.618em}.wy-menu a:hover{text-decoration:none}.wy-menu-horiz{*zoom:1}.wy-menu-horiz:before,.wy-menu-horiz:after{display:table;content:""}.wy-menu-horiz:after{clear:both}.wy-menu-horiz ul,.wy-menu-horiz li{display:inline-block}.wy-menu-horiz li:hover{background:rgba(255,255,255,0.1)}.wy-menu-horiz li.divide-left{border-left:solid 1px #404040}.wy-menu-horiz li.divide-right{border-right:solid 1px #404040}.wy-menu-horiz a{height:32px;display:inline-block;line-height:32px;padding:0 16px}.wy-menu-vertical header{height:32px;display:inline-block;line-height:32px;padding:0 1.618em;display:block;font-weight:bold;text-transform:uppercase;font-size:80%;color:#2980B9;white-space:nowrap}.wy-menu-vertical ul{margin-bottom:0}.wy-menu-vertical li.divide-top{border-top:solid 1px #404040}.wy-menu-vertical li.divide-bottom{border-bottom:solid 1px #404040}.wy-menu-vertical li.current{background:#e3e3e3}.wy-menu-vertical li.current a{color:gray;border-right:solid 1px #c9c9c9;padding:0.4045em 2.427em}.wy-menu-vertical li.current a:hover{background:#d6d6d6}.wy-menu-vertical li.on a,.wy-menu-vertical li.current>a{color:#404040;padding:0.4045em 1.618em;font-weight:bold;position:relative;background:#fcfcfc;border:none;border-bottom:solid 1px #c9c9c9;border-top:solid 1px #c9c9c9;padding-left:1.618em -4px}.wy-menu-vertical li.on a:hover,.wy-menu-vertical li.current>a:hover{background:#fcfcfc}.wy-menu-vertical li.toctree-l2.current>a{background:#c9c9c9;padding:0.4045em 2.427em}.wy-menu-vertical li.current ul{display:block}.wy-menu-vertical li ul{margin-bottom:0;display:none}.wy-menu-vertical .local-toc li ul{display:block}.wy-menu-vertical li ul li a{margin-bottom:0;color:#b3b3b3;font-weight:normal}.wy-menu-vertical a{display:inline-block;line-height:18px;padding:0.4045em 1.618em;display:block;position:relative;font-size:90%;color:#b3b3b3}.wy-menu-vertical a:hover{background-color:#4e4a4a;cursor:pointer}.wy-menu-vertical a:active{background-color:#2980B9;cursor:pointer;color:#fff}.wy-side-nav-search{z-index:200;background-color:#2980B9;text-align:center;padding:0.809em;display:block;color:#fcfcfc;margin-bottom:0.809em}.wy-side-nav-search input[type=text]{width:100%;border-radius:50px;padding:6px 12px;border-color:#2472a4}.wy-side-nav-search img{display:block;margin:auto auto 0.809em auto;height:45px;width:45px;background-color:#2980B9;padding:5px;border-radius:100%}.wy-side-nav-search>a,.wy-side-nav-search .wy-dropdown>a{color:#fcfcfc;font-size:100%;font-weight:bold;display:inline-block;padding:4px 6px;margin-bottom:0.809em}.wy-side-nav-search>a:hover,.wy-side-nav-search .wy-dropdown>a:hover{background:rgba(255,255,255,0.1)}.wy-nav .wy-menu-vertical header{color:#2980B9}.wy-nav .wy-menu-vertical a{color:#b3b3b3}.wy-nav .wy-menu-vertical a:hover{background-color:#2980B9;color:#fff}[data-menu-wrap]{-webkit-transition:all 0.2s ease-in;-moz-transition:all 0.2s ease-in;transition:all 0.2s ease-in;position:absolute;opacity:1;width:100%;opacity:0}[data-menu-wrap].move-center{left:0;right:auto;opacity:1}[data-menu-wrap].move-left{right:auto;left:-100%;opacity:0}[data-menu-wrap].move-right{right:-100%;left:auto;opacity:0}.wy-body-for-nav{background:left repeat-y #fcfcfc;background-image:url();background-size:300px 1px}.wy-grid-for-nav{position:absolute;width:100%;height:100%}.wy-nav-side{position:absolute;top:0;left:0;width:300px;overflow:hidden;min-height:100%;background:#343131;z-index:200}.wy-nav-top{display:none;background:#2980B9;color:#fff;padding:0.4045em 0.809em;position:relative;line-height:50px;text-align:center;font-size:100%;*zoom:1}.wy-nav-top:before,.wy-nav-top:after{display:table;content:""}.wy-nav-top:after{clear:both}.wy-nav-top a{color:#fff;font-weight:bold}.wy-nav-top img{margin-right:12px;height:45px;width:45px;background-color:#2980B9;padding:5px;border-radius:100%}.wy-nav-top i{font-size:30px;float:left;cursor:pointer}.wy-nav-content-wrap{margin-left:300px;background:#fcfcfc;min-height:100%}.wy-nav-content{padding:1.618em 3.236em;height:100%;max-width:800px;margin:auto}.wy-body-mask{position:fixed;width:100%;height:100%;background:rgba(0,0,0,0.2);display:none;z-index:499}.wy-body-mask.on{display:block}footer{color:#999}footer p{margin-bottom:12px}.rst-footer-buttons{*zoom:1}.rst-footer-buttons:before,.rst-footer-buttons:after{display:table;content:""}.rst-footer-buttons:after{clear:both}#search-results .search li{margin-bottom:24px;border-bottom:solid 1px #e1e4e5;padding-bottom:24px}#search-results .search li:first-child{border-top:solid 1px #e1e4e5;padding-top:24px}#search-results .search li a{font-size:120%;margin-bottom:12px;display:inline-block}#search-results .context{color:gray;font-size:90%}@media screen and (max-width: 768px){.wy-body-for-nav{background:#fcfcfc}.wy-nav-top{display:block}.wy-nav-side{left:-300px}.wy-nav-side.shift{width:85%;left:0}.wy-nav-content-wrap{margin-left:0}.wy-nav-content-wrap .wy-nav-content{padding:1.618em}.wy-nav-content-wrap.shift{position:fixed;min-width:100%;left:85%;top:0;height:100%;overflow:hidden}}@media screen and (min-width: 1400px){.wy-nav-content-wrap{background:rgba(0,0,0,0.05)}.wy-nav-content{margin:0;background:#fcfcfc}}@media print{.rst-versions,footer,.wy-nav-side{display:none}.wy-nav-content-wrap{margin-left:0}}nav.stickynav{position:fixed;top:0}.rst-versions{position:fixed;bottom:0;left:0;width:300px;color:#fcfcfc;background:#1f1d1d;border-top:solid 10px #343131;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;z-index:400}.rst-versions a{color:#2980B9;text-decoration:none}.rst-versions .rst-badge-small{display:none}.rst-versions .rst-current-version{padding:12px;background-color:#272525;display:block;text-align:right;font-size:90%;cursor:pointer;color:#27AE60;*zoom:1}.rst-versions .rst-current-version:before,.rst-versions .rst-current-version:after{display:table;content:""}.rst-versions .rst-current-version:after{clear:both}.rst-versions .rst-current-version .fa,.rst-versions .rst-current-version .rst-content .admonition-title,.rst-content .rst-versions .rst-current-version .admonition-title,.rst-versions .rst-current-version .rst-content h1 .headerlink,.rst-content h1 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h2 .headerlink,.rst-content h2 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h3 .headerlink,.rst-content h3 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h4 .headerlink,.rst-content h4 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h5 .headerlink,.rst-content h5 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h6 .headerlink,.rst-content h6 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content dl dt .headerlink,.rst-content dl dt .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .icon{color:#fcfcfc}.rst-versions .rst-current-version .fa-book,.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version.rst-out-of-date{background-color:#E74C3C;color:#fff}.rst-versions .rst-current-version.rst-active-old-version{background-color:#F1C40F;color:#000}.rst-versions.shift-up .rst-other-versions{display:block}.rst-versions .rst-other-versions{font-size:90%;padding:12px;color:gray;display:none}.rst-versions .rst-other-versions hr{display:block;height:1px;border:0;margin:20px 0;padding:0;border-top:solid 1px #413d3d}.rst-versions .rst-other-versions dd{display:inline-block;margin:0}.rst-versions .rst-other-versions dd a{display:inline-block;padding:6px;color:#fcfcfc}.rst-versions.rst-badge{width:auto;bottom:20px;right:20px;left:auto;border:none;max-width:300px}.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge .fa-book,.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge.shift-up .rst-current-version{text-align:right}.rst-versions.rst-badge.shift-up .rst-current-version .fa-book,.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge .rst-current-version{width:auto;height:30px;line-height:30px;padding:0 6px;display:block;text-align:center}@media screen and (max-width: 768px){.rst-versions{width:85%;display:none}.rst-versions.shift{display:block}img{width:100%;height:auto}}.rst-content img{max-width:100%;height:auto !important}.rst-content div.figure{margin-bottom:24px}.rst-content div.figure.align-center{text-align:center}.rst-content .section>img{margin-bottom:24px}.rst-content blockquote{margin-left:24px;line-height:24px;margin-bottom:24px}.rst-content .note .last,.rst-content .attention .last,.rst-content .caution .last,.rst-content .danger .last,.rst-content .error .last,.rst-content .hint .last,.rst-content .important .last,.rst-content .tip .last,.rst-content .warning .last,.rst-content .seealso .last,.rst-content .admonition-todo .last{margin-bottom:0}.rst-content .admonition-title:before{margin-right:4px}.rst-content .admonition table{border-color:rgba(0,0,0,0.1)}.rst-content .admonition table td,.rst-content .admonition table th{background:transparent !important;border-color:rgba(0,0,0,0.1) !important}.rst-content .section ol.loweralpha,.rst-content .section ol.loweralpha li{list-style:lower-alpha}.rst-content .section ol.upperalpha,.rst-content .section ol.upperalpha li{list-style:upper-alpha}.rst-content .section ol p,.rst-content .section ul p{margin-bottom:12px}.rst-content .line-block{margin-left:24px}.rst-content .topic-title{font-weight:bold;margin-bottom:12px}.rst-content .toc-backref{color:#404040}.rst-content .align-right{float:right;margin:0px 0px 24px 24px}.rst-content .align-left{float:left;margin:0px 24px 24px 0px}.rst-content .align-center{margin:auto;display:block}.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink{display:none;visibility:hidden;font-size:14px}.rst-content h1 .headerlink:after,.rst-content h2 .headerlink:after,.rst-content h3 .headerlink:after,.rst-content h4 .headerlink:after,.rst-content h5 .headerlink:after,.rst-content h6 .headerlink:after,.rst-content dl dt .headerlink:after{visibility:visible;content:"";font-family:FontAwesome;display:inline-block}.rst-content h1:hover .headerlink,.rst-content h2:hover .headerlink,.rst-content h3:hover .headerlink,.rst-content h4:hover .headerlink,.rst-content h5:hover .headerlink,.rst-content h6:hover .headerlink,.rst-content dl dt:hover .headerlink{display:inline-block}.rst-content .sidebar{float:right;width:40%;display:block;margin:0 0 24px 24px;padding:24px;background:#f3f6f6;border:solid 1px #e1e4e5}.rst-content .sidebar p,.rst-content .sidebar ul,.rst-content .sidebar dl{font-size:90%}.rst-content .sidebar .last{margin-bottom:0}.rst-content .sidebar .sidebar-title{display:block;font-family:"Roboto Slab","ff-tisa-web-pro","Georgia",Arial,sans-serif;font-weight:bold;background:#e1e4e5;padding:6px 12px;margin:-24px;margin-bottom:24px;font-size:100%}.rst-content .highlighted{background:#F1C40F;display:inline-block;font-weight:bold;padding:0 6px}.rst-content .footnote-reference,.rst-content .citation-reference{vertical-align:super;font-size:90%}.rst-content table.docutils.citation,.rst-content table.docutils.footnote{background:none;border:none;color:#999}.rst-content table.docutils.citation td,.rst-content table.docutils.citation tr,.rst-content table.docutils.footnote td,.rst-content table.docutils.footnote tr{border:none;background-color:transparent !important;white-space:normal}.rst-content table.docutils.citation td.label,.rst-content table.docutils.footnote td.label{padding-left:0;padding-right:0;vertical-align:top}.rst-content table.field-list{border:none}.rst-content table.field-list td{border:none;padding-top:5px}.rst-content table.field-list td>strong{display:inline-block;margin-top:3px}.rst-content table.field-list .field-name{padding-right:10px;text-align:left;white-space:nowrap}.rst-content table.field-list .field-body{text-align:left;padding-left:0}.rst-content tt{color:#000}.rst-content tt big,.rst-content tt em{font-size:100% !important;line-height:normal}.rst-content tt .xref,a .rst-content tt{font-weight:bold}.rst-content a tt{color:#2980B9}.rst-content dl{margin-bottom:24px}.rst-content dl dt{font-weight:bold}.rst-content dl p,.rst-content dl table,.rst-content dl ul,.rst-content dl ol{margin-bottom:12px !important}.rst-content dl dd{margin:0 0 12px 24px}.rst-content dl:not(.docutils){margin-bottom:24px}.rst-content dl:not(.docutils) dt{display:inline-block;margin:6px 0;font-size:90%;line-height:normal;background:#e7f2fa;color:#2980B9;border-top:solid 3px #6ab0de;padding:6px;position:relative}.rst-content dl:not(.docutils) dt:before{color:#6ab0de}.rst-content dl:not(.docutils) dt .headerlink{color:#404040;font-size:100% !important}.rst-content dl:not(.docutils) dl dt{margin-bottom:6px;border:none;border-left:solid 3px #ccc;background:#f0f0f0;color:gray}.rst-content dl:not(.docutils) dl dt .headerlink{color:#404040;font-size:100% !important}.rst-content dl:not(.docutils) dt:first-child{margin-top:0}.rst-content dl:not(.docutils) tt{font-weight:bold}.rst-content dl:not(.docutils) tt.descname,.rst-content dl:not(.docutils) tt.descclassname{background-color:transparent;border:none;padding:0;font-size:100% !important}.rst-content dl:not(.docutils) tt.descname{font-weight:bold}.rst-content dl:not(.docutils) .optional{display:inline-block;padding:0 4px;color:#000;font-weight:bold}.rst-content dl:not(.docutils) .property{display:inline-block;padding-right:8px}.rst-content .viewcode-link,.rst-content .viewcode-back{display:inline-block;color:#27AE60;font-size:80%;padding-left:24px}.rst-content .viewcode-back{display:block;float:right}.rst-content p.rubric{margin-bottom:12px;font-weight:bold}@media screen and (max-width: 480px){.rst-content .sidebar{width:100%}}span[id*='MathJax-Span']{color:#404040}.math{text-align:center}
-/*# sourceMappingURL=theme.css.map */
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/FontAwesome.otf b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/FontAwesome.otf
deleted file mode 100644
index 8b0f54e47e1d356dcf1496942a50e228e0f1ee14..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 62856
zcmcfp2Y3_5)&LBzEbU6(wGF`%u_do$I-wUs=poc3^xzP>t859|l91%ydy%{4ZewH9
zLNU#OK%5)jlp7M#adH#VlN(Y<F$u{?9vQwfE1Qt_{_pd=&;NYS=VdizXU@#doO9;1
zWm9nQCd!k_qQun5m21`nQkq8_D9U>~MSVYG)7F`Dsts8mQIv>+ztD)dFw+9OVG%`1
zdML`ns?&x=Qnp|IfM+dm&(}ePcdqmf<on<ruZy!LIb*$T6eXQSQ4A9omShs;Z-z&d
z18&9;7N>37+Ghm#p%f+FVKQ2*chjkzF#ZB~9w-bef!xGBr6D7h{6UGOP@t%*!8rhr
zqTX&D_txFJckW8F88SgJ<w0uU4@wMm7c$bUyYMH?cE5n|O!yIHP}F8hln(_lBNJ6p
zG$oOgO6Eejz@2(nsGUOjuTK9XXG%6(sO41PA46l&S)vMPA}p1Zf|&-wMg9~b4|gr(
z{oxi`z%K_ScV0|AK#d>DOYW<usZrFtp?c=NdMUMqT02xPqr9kwp}Z%>Qiq1}9HpST
zU`<34PZ)C!_3}_&M2)6kC53tq%16Wv<;B!kk^fL$a$g&o8ZTNrRL|U3FQqy}Aw%^t
z%FjbIl=r0M9>Z`rYKq77t>{++@-k0@oM~*1+}p2(7`Q4V*n=HYq=vsI?g5v}-nP
z3|{}}ibb1(*R0;YdDD}@+q7nj-e?F6nlWp}oWMD=X3yOms||yGW^I(#9B4HL0`>*2
zG{Pq6qjlCmi#Eba+D94TAv}p9V_D5%k=nR0b4*~E)oRv<#|upiMk~z0GGmR=Yz-V5
ze^pq5HgIj2Au?HKwVD>qoJsnJx#u=RZ=|+Tk5lVmJ2z1#N=q3aw}vu8YK7c-N>4=y
zwHEjdq-Iky;2wVdD3u7c7HAy@>636rQ}I+R6-Jq%%_eFi6$}s_rB+ajpcD*stEugP
zo136*FtrWZo1wQ}7%h+r0@$<Qo&)2|f!u6rF0_q>R$<VJjBbYCD4|y%%+3fkR!S#4
zSqUW*l?*NPFbAD5chV;Ua>MYWppE&yKBVk^ODoieQIXI-PMCWPv<icsq+U;j!#n*2
z#Q?oXiL8nRW=qz>3^jr9p7*cDDu9q6%xx{?3;;b@n3omixrmwx*YNmZf9p3xm@i;8
zp?TpJjUB@J0D^@;Vq@WEgcj}}s2gf=U*-SLs=qz||El20$!O-RlsfnS_J9)6lK^rf
z@F|+|fem;DctSVzuQ6lCs>g=*`}C{(m-TP#-`gM6ukSbXXY`l%AL#GuKiB_u|L6U`
z^xwJVb4z_|(yht2X53nKYvZlGw+y#3Zk69U@CS95u-8E9*x%q${UiIw^e^w<+#lK>
z-M_Ej)SuN~+27uOroXrU-Tp88`)^UVM&1epcn{s0b!+*p&9_2tnQmp>swD94ennAt
zcir7`_tDR9d~W}I%Sf-0+(^%nvXRn}u#+RjBRxinMp7g0j<_@8_K4p{{5Im&i2f13
zj`+pr(-A+9_-Vw=5kHRjVZ`?%z8i6aJ1^|@`u}w?=l`!y{<jB&`-*|PdWf?qlc*QY
zmn<rQ&VrMWqC&{2^gnO%Y^ZJzK}2XjTo&@AEVP02{%6o98R`wBPPi#|KLUfahdLz<
zc~X{UQFDgd2xAP@59i7Fk}RSBa?*|viv&`VAD-V39H_f{SAYx80Peju<oG3Ugmy#u
zq4&D?KKII`AG9HThEt(0sfU~7@4?HWQiS&zYJ2~4#^df?NftE^?sx*{|9CGK+Ce{n
zHBT7V-Pe1#KdBpjZ;~G_508DX&oGQE_q>JYkcahKF7zYy(4XAHaLAh7>kswf;WDJ8
zodnW*&mk}LA4<ubgqyloKaK3wa2o=9c&!8&PzL$tF3I16Io$q#rdgDNv>ATyzs;HS
z&jMIk)X1SUY8WQ8mk8qz!5gX{ac?|#KNXah-`{R{t;jx;+arrw4mTM?C=b`)g9B|K
zKbe$=Z!xqbc>xxr!#G3cIJ_43-sk>0XiMsaXE3e+56S@N-W&nebhy1GS=0t{!`!CB
zeXl$`20SDCO)=z#yl@A)%foXM<_FJ&aY(!S?qN9ajLc&>wDpF%>BD`=97%ujZX|^{
zkUJb;(Bvllh3Ak$Tkm1o9O@S+z@h#=rtsbr<wc%JvfR%|ooFZzaLHWfk(AVvv4!y=
zC3XMXJ=_k`>Eayd0}DguL&kx00m+ja=Bpt$)C)Jj(+GE#@N5{qN_YooPx`~Xe7HP3
z{%{$_+eqqQIN>I3Ngv^P)=&zdhx-v8M)G7X!|w&{r;s|*7v>g7Gy(!cXqP3lRov@8
zR1fWh=MwT9Zqok0{>Y@@?`{gwSN{7?L`gvE7m2*?lX6LUm1893w2Pdz9?n{^!(W2e
zdWpaFl9b@u0BLprBcj#q)KgjW@7iqlGG5Yvz*k2E1b+8G7f(?i1&<P?5`}^sNe_Su
zp{B-C(?AvsqLxt0KzdwDZ2-w}Hx&Ysl@6q}{UBAPQ`uA*RZBHf?bHeCEcG1KOT9u}
zq28xHrM{#3sXwVf5oiP=nP{X)AySDnqH&^!L_wluqQ^wRq7YH6$RtV?6^d#_Eus$5
zNzpmc1<_^Ei=tOVS41C(z83u``bG4+2#M}^upXm5R34K&f;^Ubto3-(V~0nShtVV1
zqrjuWqtT<&qs!xhN3X~09(^8Pdh~nT_HcN(#1gT$*k7y`PZUoR&lN8euM}?*Zxcs|
zjp7t>vA9XxDLyUk5nmBs6~80?xA;He-^DJ8RN^C1NybWMO6ExxOV&s>OP-SKlxQUu
zNxCEtRJdwMgQQb(MDmQ}tmIiqujCEMHOY0!H<H^lLo4Ws^i+B_J)d4euckNAPtzf^
zj*g@E(+PAMok{1?WpowYOn1<S=p*!bx`+NJ{V)0|-A8{-|3(io9*h?wXVlC?%q%8|
zS<0+pHZ$9q-Aov>kBMipnS7>{u``WKCv$?i#JtM9$^4u7g87d5nYqQ>kup*r>4Q>U
zI$1hRI!8KRx>mYFs*@&5bEW0dI%&J~sPvTdy!1usRp|%PFQwl}f0q6xb;-PBD%k|t
zY}tI-V%aj;YS{+aQ?dwIjLaxYk`>BoWsR~9*)iEk*+tn)va7OpWS_{smHjSrdP+V0
zJk_4#J?D9@_1xwe?HTK7@=Wl|@+|Uf_B`o%#`BWri=J<Je&qS3=Z~JZJqKA4%dw+a
zB|DLw!cJ#rv2)q^>_T=4`v|*&UBhl-L)Zv5p0%+J>@(~s_AL7X`wDx7eUJT&{SSMK
z9pETV%t<)~r{X4Z^SBk<7A}m7;^H_fm&|2x`CJ88%QbUt++pq*cal5LUErSMUf^El
zUgJLCKIVSme)FQdBwi!E<X&UECU`yMHQj5r*F3L<UQ4|m_1fvBjgAeoSmNR>`Us0Q
z%p9T98WOazMw1pS4`!>y8fGSUh&Ik-O^&x{%~AT;IIAusHq0EYwdzPtZ?PI<%-T3(
zf;Poyj0@2lgv1zcHAY2Q^wEZ}*a%}ZXpR=04ir-WpbZI&wOaLYTC*`MGSZl6h=r8Y
z4d>%cq(*NDHzt{4!;(WH^yY|Ityyc*hFL*fHES(8GA!v5YmA7AiVce<mJ^fCXwX^`
zjIr@?+7K9gC`X_UW#diz%@M{(Z8*FsB-XHK_-6?>8e_;!6kC&7Z?Hyy8O0n%G}drq
zY^2^A7ORi2YLl!XIxW$Sg>0fe(yD_8(T0#%Z4_w&Inczd&{N0@YP37MFWzF+M<tUv
z>kX06M(8q>71~9GMQF*2ge2%AwMG*R7f)W-5CO{_W(pxQ1Gtd{5P-01VNw=dm{|+^
z6%j+0-eT37Lc+r$ViLp5kx^l=IKzeEl&qvF4E7NA%LH2ey@o@10m4vTyAQN~fSq7A
zx?gWNFHF`H8*d3AI~%7r4CUPWFH{<1gk*m_3<L%Qp`bTJ8HJ1`!mI^rh0X~3NTxls
zwa~}C$KheHh{A4%na##T_tFYE_i_r^c$51f*;ru}2qFMd=u@;IQSq^{Ls?5)SZu5|
zDIzv3F6`b+qV-W$uzN&B>0u(tfF`iWB#nqQTC}hv2E8F#m?<omK^qYkt2IQzXkf@N
z#zh$8;$ZXQ!lDh@d#e*~8eSVR9kbS&sMW&W7)>SuDFTQn3UEkkc8@TWC!-F{GC^ww
z>q*$~q;*EKK82V{VgW}(B4CfL)<iv^oLQI^!4rb2LcxuuAuy)d7^6)FDzgkt(PGub
z$Aw!$;!OgnxeH`|q$m(Hpl*~v;%?F=y9pE<t2e{|Zn-zj6mQYNO90+$%|OIJ>4q56
z4)D)xH0hF~^)O1fFcUYy3iJruY7hufKutIFVd8R^gr`Ecp*I_TDL24)U<VN~fG2?C
zWc?z|7K<)2G8{G*7RL~-jg5^UYZe8oX4UFoXF>$r5ORbRg-pCjNXR?8@hRjlg!)^B
z(D!dOu%iM74)q`)qGOHW+C($Zqs|&;iLn3^gGC89>$Oo4U_&EF=f-R>g=zQ41JxU%
z^ai~(IaX`22o=$0BP<fEARt<NWVkjA=s6<Z8Ugk;Q<zCW5536(Ml-U)A>n|0z*CK8
zK%DqkW2^;?Z85-a0Z6ni9$1JOKmq#-j|FR7G;j-Zd_)ZF6-)}K?p{V%<Y=QY8d&Z<
zqP?#STLowXY{VKOER>Lg*B4T<l<u5+i3nRYSS(=UBe&2&$ixFKyN?WE|A#OLCTWb-
zMu){l0^bie!Zr?}CTTQ`r6dm=)@spP;XO=%vS!%bFcYvZE$~nsFtPa9a3EpW%V@2z
za>BUeba0p4h(`{lkhn<bWz`vh225H@ydEZR(GqNNj~ErD)kYbOk;3^SaBwnsydL<w
zAsTv_2%8U!g8{<~5!yIyWQ^V#MdkpY_i3%+TCG_ptY_4$C~c%M9+o}?W(B`wq5f7O
ztk>Ua;!S@mlEwb3uRAAna%X|R34lqnNUbFX_%$pF{0bXxjWdRmGt^CFZcG*MWq&*%
zpD-JDPJjsSWiSA$4WFQ~!(<C@e?Ma)i*z4zB1LTy%tJVrBaCEMEW){MiG#tTfHxcB
zw7O_GC{1CJz<h))CI_(qh@=Aj13pH=3c^7_qyWHMttmDcngB)vdm~K9%@cs+-8@>L
z(g@%$q;&`!M=`(;0H;FcJiPEeUTy)bGXu%#O;$^MxH}UvXTe-kd`b#g8@(3xP*30x
znc%M+5eqCjy*4&-n6<mdC&xhua5}(+c)dlZg>xnX2oC%!5s^Uj?t@SuO@S=#uW(bx
z{WX6b2|^FDjXG;w?7RqzWiB8Wa4|QJBTGftngtFZz*C@qy(Q$Y1K?iO@DUL*ch+1%
z9wK1j&>$1McLEb&Zk8+5#cF{jf&aTxfx3yPAYib-S%s<1oju2WfRYkWB~Tuak9)I+
z(-1(skh!xT*2bHo!{JN-dNJ<<FF|~Gj5#sf8bJu^9#IJ#A)M?m@ZWIYD|6>8yjM5m
zG60rH7zk-~uZGNixK`kLe=CruA#>*j!96b-j;Z)?t?(j4`6Spia^GJE{4Ojx680Zt
zNWe8%t069;H$XAk92OS^LR}2VREDV856=$Q!%mO|6<}C_6UCa{zd}W<5upDiblg`Y
z4Cvl7f*bc0-6U;-JxByu&zNWdaxxqBk$}(fNs-__0UlzBNj3priZ@%}*dQl4?7A@u
zxFO-}z(C>X2fT<kgv^~kpNQxB1Z{YHbV5v`Rv&JJ0}1|x%zb!{@QQ%IZcYg`LGIZO
zpJcFpWODFeexwnO)u989EbGGy5<u_-@l2RN$lL|9+((7GXoQM6aL<Js)_=h~xaY(D
zFx(5^Uig1;Mv2A|_1Y7BU6a5869!7Og{q>Os4u7+;J0*%HiJsMQxqoBiu59bC{I)*
zIwpEv)GK;ZbY1kl=qJ%1q5%)ugY$R_l;6D`VIDe<SWtj(di*XPC4OAIL0m09B0ete
z5}y;lEq-78kt9GePV%^9lVrPOH>j?~k_t(Uq#ab(*CcOB-jjSFxlRYtLG(g8nl{qO
zbOHT5{ZCLqIVOM^&rD@zGV_^TOav3dn3%)Nr_5K(_smbsZ;XR+Nxh{3(y`L%(je&q
z=^E)esaBdKO_%0LE2WLn1JX|EJJNqkKa+kfy&=6R{Z;m$EI>A1Hd!`RHd8iFwn+Af
zOe@pN;$&u7o$Qe8l<aNU2eOZ4pUJ+K-S8B7(w-AM*LZIB+~XPL83RgXj;GDD(epXa
ze|vr*sFF@r&Q4(Gu**Pc+zx7^l`UYaKy7ScJJ>VqKiD_fkJ-=Jui1W386V`Pb1S)E
zZZ{Xs={O@7&!utMTpf3Udy%`wead~q-Q@bYKfGjKDz6z{L0&7o9`}0EYlm03m(I)J
zmEe`?mG4#O)#laVb=0fN>w?#dUN3vS=Jl4>2VS3feeLyw*Uw(Rc{#l9deh#V_egJz
z_ayH*-iy4Kd2jIE?ESR2*4ylz<z4Ju?cM0z=6%e&%e&kAlJ^VVFL}S?{i*kN-nYH6
z_gx>xhxHlZ<LjgGnd&poXQ|H$pC^1a`2_py@Y(C5_p$n<`egaoeCm8!d^&ti`JD5)
z=+ooV`vyCf9|-;$^yqcWZhqYw{_19)yA#adxmW?$T+N-}@hsQD2GF5<`GA4;gKS1-
zZf1d|AhSHfo{jo&pI?4;O;L@irl!2AT7#~kYc$88xkv%mTn$&q<KQ5cTfySK2|Vgs
zz;g3hr`;7imvz{2@O&28uPXi;o8cTw@K5L`*Ah<g{*$k)YT#da^8I99?XZu&zlh|l
z0{?-}!}SMA(RuU-T=!B^uSW85>~0u+4bSNe2Avwqk&^$DHRv=KS#CD3;S~8SQm|;x
zN%uXOg<%H!6sOWpT07MECb~&~iaal%Kr~kA@W=0<cU=B1NU@M2NFDxYEDF2;t<>ly
z{t+$Uxdi~XHN7!e%}J9R(_7UXGlAu{@LgPTdU`T9mC4D=%h61g=2Yj|)i)V?b+ui?
zE#uW(1@DS-MfI`{o?I@T&abi;)~M_?7x@=n*uipt?Z;r>c-GlBp66Pcnp(J_b~W~k
zJU4;W8IE;z9Xr-_5FpZ3`8gH2s@$By{Co|!66RIRN3*C1^>ST?V>+@U!LTF2up`?-
zL$|?lw4^nqr~{nKnUu7&6b%lRrZlCsr~{Z@h76@~^htykcl!R`V4$yrCB3Hbq$w<b
zk75zewKg`ka+6sfGzLklcy24qXe&^cxQa!=ia9KIRcXWf_UuKx+NS%@jt;eL_46N=
zsBQnwe|ClXZOkIcsf%sZFSMOJ*e1h|9KdVeLYI|aJo9c(uR6DF|Hs{b$lh#2lP24g
zEmz*Mzo~ljt(=rA8XKDQ%hg2nvt#xz%}tc`Y<p(S!%F--e(VYSNRpac#sW%cdCqAM
zkhCl8JpKxQMdPMvyh}cXMNJHXawf}JQ{t0yG&tV1%T=<1HW$Wb#wn8<QtC6*4hsJ{
z&s9X1<dx*-Dfc&6jx{z^RyU{}PoXR>n746_@NOa-3Klzp2l^gn2VQjbAuo0?#JQLL
z$Mz}bSE*b<%<3&$R%={A(pBfD{9}jO88R43TRRf@j!umu(~;H5a&uR%M853YmDj$}
zIQyjET)Xy-no~>!4446Ue9XYDW$(ym^9NXsBiI!j&bBmH*VjYd5uCtsQXS7>`8HO>
zDbN}`0?ouLy46Rz8=vn%p8Uqm@ezB}D0m6pght^=)w6thX?kgz2G3qG5zoOZl-P#$
z;62Eu9_V9|U>i5{jy^LBsJUYYou6NrldH_F$f?R#6Z}L^@PMpQjwrgSs={8Q<o2)$
zHWmgiWnl%Z923fOiZmWGh*`|zFX0-G#%J-a;3=#lm_e&~Qbu2*F9Ekv5Ub%9vFm2>
zoOChE&E(fDVqJZ+_^S(9K%?|z4Qv@&$Gd<tVBid{B?wZ-5(lIR(#e3@NS@DgyICyf
z6k1rSgHZK7@~%9fLSB7E1<z^nFa5jlO_jaQR@$g3YbdR)QdSo?7adWjSEp1ZRfg4X
zX(=c*RX%+%yE6B1VO42wDVpS0gfPw36t%IK@U*HtczM+fAR3O41&0=~YL}e3z-nA_
z>6owP0l%>_y%&IxVx)<zdfg=UDucRE3S3<pZYuAPOQ!PNKlp9%YKluNr$}V+7!0KG
zB2b3H)y-@Rz0j+1bi2CI9N@FPLF@o9%c6lGMb7j4{~7lTeiOav7x&Clhj-a!t|~{B
z)Q1qehN~!S($p5&^J;P`^X=0PY(22GHMGUhR(7!JaILJov9zU4S^8?puO+ujua~^&
zUz(9)%U6Y!&^F7ym@>7#jOLcGPC4#d!g42=Yrv!#JYwQRKph}ax;`_tIz`20);H(1
zsJH++i<8d1wvyoE7px2R-tQK>V~5{WU|KHT4=~~?>;J-zTfD!37u?D8Q>s%Z8#$yy
z%h5wD_x>xdywB+ughWP$WMyPzRwT*3=TpiXGn-0FZKbMbDvnhisqR1g!-dcPCCh&K
zU-?&5z+T@$$>=nPF5$IkC4LdF#0#)`=@RwFOYj1u#w%4&w-#zI;XGu*dusADPKoOm
z8YZ0Itm0}4+W;2`1!=edNfwuq23(9Y^AiBwidZ$*g5<m!^`Sn%;c|k*Evg!OS$U17
zqqe!@u#%g_Um&PuM{@TKS-@>O$1LZ$6+E(!Uc|#A>nDKry|{>zcC#+K%kF13+aeB`
z9VD9p6UpVd$^V7B9CH{zE9`mIIchS3J(9JvNG|5m;2dy7E#^4~49g)Y8pA2@Lg!dK
zg2BOf!)Nnef3=~Zrna)izq+0-OJ%Z4GBT8|Rd_LG9C|4SxZ~=3jfW$p9$pYw$y_dg
z$>JhlV>uJMiW^X%#R@E9a470Q>roqx9zaWQErSDbk~yp(uQ0DT&%cNvuP5iE^LQ+u
z26PNWna=x2;dpDwYtF2PX<;eXb<CjS6zx~()~W-IEh@)trnNRbIbV~oj!w=`N>5R_
zZZpZ*jjdH0&h{xRQ82^3_v)+fai0dznTkb#fpNA>TZj!$wMBp(y(a5G+OcF=O-IX7
zI1yn7^P5|gEmh6+^=fi-zRxzcYPfTi=c-TFqDL>HS)ZW?kxW)_xu>W{<;ZnRK<Ch(
z8Xt@0Eq-4o#~65yv9cUF3^@gm9oFIHaOp@2sF=hPoFo;n_<al~#p$c)^n#qsbmc`h
zyE3~bUtL&LRa~tqFSnJJYdEQcN8WT}rM<pUiR9ASg4*0Fb#7HwSx%9Q!!kdvjaR#X
z1Dkm+Gcz?mMU|3TRiCNh@M)Y3*J<ug9uC8bDp(3Q|D19Kw8yaNS+Krb5U>UuRK|0&
z{yIfL1XJ`OLv>qeQ+d6Ac^h59<d^1`=Bn#=U&97^(2H~a2LopSVPeXX>pu}O!d{)1
zv*gVuu9H;FWrMuddxQ0v#UA3Pz#$I+SM%g3Mhc$GgAw6?7&+-zJQ9zbG>QEFIth(L
zBY*uBja2)zlewX3ESktV<fu=~CBe*kV_7)fmb2X8&cU5<8srz{`2)L0R!0!BJA!^^
z<zHsNs&8P`14N_%_C=E91I9{gw!bAi)w*9f#QcF-kFvRG%7#|(kcAq7!sL)W1k__T
zb3d?RE*&l98#!xzLR+S~pti24PF2Uz$;sJfv(j8^ZUq^OPHsJveMoiaP;D!q2VGy^
zT!ogf`I+?^pr5rYx9@YFL$lex=#Ntn;jEq)gtL{vc{^WIA*#Ko=zylGq@uW@sJy5Q
zR+>ZS|5(mkM&oHz$Xv$b>E&ZkH^c3ZkKeyP{@`J>81Zl|K725KKL~og7cTUw&+r2C
zUk9>oB)d(Z<z?(LAaaq-%8NO(Il^W>#5JNP*mUmDq4TywX6_8%+DKj@yYsN}P;F;x
zs~Sy06X}*#uDQ7i4t1y4@e^&gBNN(#@|4_eym;lN^{dj7Q_?EUGMmj-qU3N8NR(vr
zL5@U0AW!DyaDfW~n7L>qoU7ycb%~=uC}_($bO<ngGQ4jFA#7Ok|M8DgtY&bEa|(^;
zLpnHxUOk4}ofM&uQP(F_SuDyu)+fWP0Kx-a3}jK^h|mo7ArNlyn0CM{fr$}D#x-QL
z*%}(SPkAfv>;~RAg|+gl_}Tm%SPM9pFM`C+p(U`f$Ogj39`p#D49F9Oe2B)Y(1=eW
zw)bneg>cL|gV(T-@p*5{tE=Jcu_#{Qxp*GXIvt3kkYHpQ3rMZzl>31_u>s6-4t1k$
z+%4rq9}T342VUdi$!t^dQ!_JRmu7%?geCz#$k7y78#|!3og3_v;<;Rny}YW5!%{qk
zYr=}g#4>emYj$g9vy8LVs?h8`L_|TiBLNz~6T}mIn`7Q#x%%eXmYM^ywlbt>Y*KQW
ztPgGNM5|#@Lho##(bo(L9oRr~qe#cANDc%f=kjIw`MHHTDlBJG(mA{ekB4g&=UR+@
z#y>k2b08anAWukZCeRZa(ch0ofCOX(Es0wN+K`%qt+#QuZ7_-y0m}#2?n`dsD*wD%
zU9TxGD=jNm!ZzETgs?z(%&2dH6S29assTs?*$2o*DW}7G$(=zkCn=n0K=g91j%PTP
zO^O&KdH%vD8V)3XPz7L>;2B8w07~qv;%G|;IoyGV`0yOvTG|Z!pBsQ#a448*<@V{7
zdf2gEhBIedl9SbV5}wF0Z(rH8R)gfF3J%|GPxzE<#INuQA;=Fuj>54gr^1)E;a_nA
zo)4mW8(@oc8NVA2@UCNk;D%})%w{#z2H@ok=K_g?v+@cKVge`%egi3pAfR$7s)V8%
zDeAC@I!=iS?|Kv_iSmi9WFEB;;){P5Rf%dKM4(>OC~6j+5}g+P=`qz~g~xw9Zi~l?
z6U67mcO<+dT5?YEC%uhsrC(z<F?Q*jvT)fiU{470{DNJ_R<K`i%ef5D^-qGX{)G26
zpIDz{pUWc-jSPPv`~mF)&y1=ab<Ee_w@ThT+H3Uf2fO`Z{c8O#_<ikn+h6S;>|gAE
zO*vJ0Soy8esY(oZgqQLER6n4etX{4*s1K;GsNYi~jhAMuW{;*_b1QI4;QGKH$2>CT
zA7i<(=f?Sr+dQskyn1}e_?r{PPpF*GHsRt#zlr~zR50n=$@LGNnX+igA5%|F+cqs@
z+S}6~n7(}aZ!^p@%4hsObLz||W*(ijYF6oN<?Q$88s|0!b<g*lKX!iN{1+cy`S8Vu
z@q(obzFFv6q+MLTWbTr%C1p!TEq#9J;Ih@rl9v7Fk;df{mLGp~*@{P3PF}Ta^^<F|
z*IZxw=-RrqU#~CUz;0Nxp<u%g8*?}2Z!FwcylKhin$6#Cd30;Q);EGXpPKem&{G?p
zYTWkB)7N%9y5rE!hj*UeW!^n~55ITD-rl`$?Y$l%3K<&$PTG*1kc!Z8p~u6fge$|J
ziO@upMM@$~+DY2hsHUi2qE|%M=pNPmtY4%r(toyZ#lHP9nwU$mdt={<+ZXqK95w_R
zo-)K5vJ7^^vxZ*`gGMjoX5&jHu_*?eLDi;{`*-e--2b7u-F(ARX{ol@EjO$Stijgb
z<9|<RPdu9VucSFihmxlxrzU@%Qj~fy^^>$QX$5KDr7zAHmywn^Dl<Cs*(_r=oxL}^
zHv9D)Dras^UCxc%mvjHjTbUP@mzL+qUzq=8zN27fK}Er@g<}gl3kQpmicS`bi+2`x
zm6X|(wl7Q5%N{IiDSy7=*@~AczOH<z(pcG1`JXCj)zqpVtIyUl_4^tgXn3j7*yv~~
zZH{QJXzp$KvK1XL9r&efL0eASyX~6xpAJsxn0%!2=s%BDJ!5^Q=lB!Hn~#5R;?t8c
zC%aCWyNqW>pJ_O|<!sm4D`)Y!HRpDmmz?iBf2Vs>_m=Lh-A{Et-MyoGSNERokiok)
zBnhB3NFqWKByj{Ii5OXtL=iv-I)VcRzH|jku>?yL&Y*4VU{JsS#rOmaeBcup%p(vg
z?BW3W4M&OsA3!q@+*i8Vuj{V(uR|WXD@)op>iqEmJe@|bq0uaUO$x21Z|qu<pxfxb
z4twY}wlBG&^G65p3}`iDqb^}d*GB)V*rRLX^@J;3@BmGxq1Q2BlchX*O$sfiU(sD}
z@4DiD6?t@BNw~hT%io8vkKcG#kRym8HjF~Zx`IIPC3YqUWQhzy%o+}8AzjM)3})I;
zpK}&^P9#V(&R}#>aWJ_xUXAmZ_~hhx4bGFsw<aG{qdewdYwJM`de`}>0wse^@d)0B
zL-DjAP%<cWR&qsrO9oniAMh)vsVTCn>gua%Yc&7*ptG~HMb>n%yYV^Ir+quNu8Y~X
zOsAO}fxX6IZ{=QTe4}1~-O+ORpvERWcIMrGol^hUixhq6Nu^Kwy$j!Uz@hXT4-9Ss
z-^eat$rCh}7lHN*%g%HL&}$Su8|+c)fPpL~YD3OWLx-U)QRDO)^r8pth-2Z11unc6
zgng%-ae6tu=(e_wW5-~S1W_f(E39}MY+<0HH}t}`?3|LK9Q9xyw$l+A#;7pmon0@m
z&K*)1ESq+ndV%!`g!5xSUcduLyEub)22bZfY4K@?Qx%R1r~Nu#$Db%*0|u7If<;f-
zZs~|Wl!(S*4>TT2kOs?S>p%Q{+3%`Sh&B5C`;XrEP=ho`23o%ajYA%X+By!lcghCs
z(t*>G`3tf5iS25v9E+7>u>TlY=(eddSF1{x5@z+(?=Ec9VE;d`68_zm&3^yMUl5~Q
z0Git}{%n4T8P1e5L>?Gep2ptkLk#cJzMcm|(|{by6<_nIywA5V(E)G8Gcom+3bm`G
z563%p(Fbx;4q8>~c*j#Xi_WWWENE06tM5GgA^R;KAldIYrnu%>=<-IpTt0YLpJO5Z
z7ka_5=ykNkF$!&QjdCo4<9+{Y{}-4YM?Pfn-Sr?2iLE?(P=OM*p<nSObu4}mYw&is
z9z>d0w2DX66fl@N?-1iD^%I(}!F>Y{#DE3uA#DGd2hEe5<#MzbG*8eJ9rAVS*a7>X
z{S<r0p22qk`~aNQ1;#8VR5|1>`8p!61R*K0CV=3?EN|rl+Y>-AblM$u#nWsCFL|0B
zfQG|)pZ4~I6JVA_-Cz?4mQ3W`hJitlTLhF*gLObK6@qDS+lA0x(4E2J0agpr&cu^;
zCO{MD_+OBcSu~yntMX9y*I=$xBgAa|S3PuJ@wbLP?TrDFLn7oI!1w?W6b|fFfXJWR
zs>T5*;3zvdesBW5jGjNr;s6}*4v+5OI|y>`@(7+gbxs`u84}+uPY@vw00iu76xufo
z;xcky3)%Z&;>+Yhm+!$8%J?!scS9CB;mhtZ2z){+m9XdqJo!a-xeFw$i9EJ~O~`HB
z##U^V3ifpbIY!5;!OjkR*D9R>68VYgd@_*MUtkE$$-fkUxcc07c}E{~7;XvDp<s)E
z!A9|N6X-ao5Qxr-pGFhaTQ4li3VSkiYyF~^!(?dmLm%RY@O&li!YOcdk(3yv(EJb4
zLumBR>X)Cb|1|XFuvZq>JsB#)PveQe{;jxBiN^8{5K0jUrRqVzDg~18#Ciz@>FQUv
zymy!<s^ZW@Kr;{{`%OW;QiHBf%-vu@?gf*73y=GH!IannriosDAh?CU1N9&uwg6@>
z&*Od810Fl&u{>a&NYRqnoKmjF>yBohOh1`&!vECeGZ#-?l2ulhSKE~}#We+0>ac&U
zetlbytST=DEOI$HMPT2?V*?FMarLpa{zkN(ZYfS}NLFDp%px<jP?NeT|B1YfvDvX1
zhBU1H)2}L*bkt)Np19AVAx4iJC78;q^3-Scp5A+ES87~FTy_lJX!J`f%1lmEVzV@*
zKDDJtjn<iJ^^N;lV*K^HqeAq1WCeLe&^Na%w=z%t?3VX7^zM>@Hdbg?*+HWKXULd8
zkEK16c|6<yC|*yelr?9xs*ss!ZLDvu&@{hR^=9>z<F&^djx~WTi-JRi^gM-BXwr#J
zO_lJ%P?KM+j(RToqQ0m3Sp73~C(wqYe$D0etxZb6>UdZ=x9l%!V#N--vs)1Y?7`7@
zUn0ko6}wEv0^s#bf$8Y;nt{g#<aZvE4h9WeQGi{zyrQP6s?;C7$5fWs^UDjL$DV!;
zYs?xJzs{^ii|L-kP3O0%2<dN%*qNwVjTh7S^*#K}JL?(jac%ZDYG)tV*nU7B0QlzT
zKnrl#fR<?oZs+J$l-bHk%G9oF4uk8O%Rm>G6c;O9Rxkp~37xp$cQT7Cj!TNVhT`^&
zI&4Hw_&KKS<?+)}T!ccyy4KplbpC@8uA3QI#R+v8{;xk;nO{>_Q{rzgsVT3nbUx<z
zz|Z-Z>jS!=s=ByFFeTQM)>Kqhz5aopk1G=ntHm(bZMG8dQ$BhNn1}_Fh1}7Nti)0c
zsT@ogRyZ#PtP12$h;{@IwrJG15JZTZim@zu2-s#H3a(^DF9b*f!~-`SXB4TWX_;v%
zT*RcM)i;-FDx{sz1Pp>3(E_#;_tAw?r_B|uIG=Ss?X=o8Z{QexDBE<<q{81pgC-nI
zCzocV1ClOck~11xOEpa>7`o%{7?Ua9oUL)qyK{_Ai_VIOP#S7N&Z?ckpe>SiZNU9u
zm_q=i4bJZ5(sVGj!PB!f7mo=XL{82L5inMgk&7V{T*SK~8Nwgw=%`(Z+g00lwVjUA
zU=<3WUD{k?Dq6tekKu^y$hJ1`S7AGt=)v}92iHh2woB0rmiQX{&<vjV84k2o9?kCb
z_u0X_O+UygUoddnal3;>w_)RM|6e?WpRxG1qwgX1Z!msyPF7Ub7d7P6Vlc}3fyKQX
z{8za}`FR?A4PT@4^9plwl!99goGkcu9*=ILU}-~rO?{;X|K@0ah;2_8fQ@>SAE*Hu
zm<Fp1csmV2@-9<WSz1+AiFA%UKj(G^>0Ehb1*Q3A1^#G9oZ@s=Z~7@U&T;h6C(|Pi
z>r_B2x`_Sz(lt28)kCN2v$jPmT?xPQJ9rqtDh3Y{nDII?+Y{^5u5Q$qRByH=X89*(
zW+qsbz#re{>&mNY!JH4q<+i%|_71QcjvmY20Be`s_Y9ba=Ca)^9*q@#$RFGQTd(6C
zD%WBR767mVjOD@V9ovsqp^2K>2HSzmI?N+AtVd2c@Vk<n!h$>*_I(IXT8ZbX?y>VB
zUjx`hNA3vvLF4-_R%7+suyd>U8$5c5_dOFpf9J3&TGE@)C^juSC%r(E5|OF3M9T2A
z8F=ALyha5M-v?g!X1a!$w-VTSu>AxDq`vRwfu|HHXh4~<Xv;G6w8}xF=a3Ek`K6|`
z%vN5eJOr!<Tx#$}G*aTRc<aY_yUIm__pJn@d75HdQL!?mF%3*yyO^e0dt;@h<$V3w
z&T}#}?R&KFJG4dx13a*E?vENW?JQy1?L_3cce%|k1g}B%LAS@Ss82Ems1$xOV3LYd
zQZf7^ACwPErhQ-=qI=PN*toxoi7wD18i~ga#t4S9Cs+(>0-SQeQgF!}1ZYz~VPn9c
zflBaRv=`n3Qn*Usc#Ek45eF0^LSR7lb6Mh?HnDpSg`cyk1F(<aMGuhtCOy}2sQZ!<
z#$UIV4Z%x+24|~c4V8^X&6^<njd&4?uvpW#uEV1=`!fvbmQ-0%dS;46IhZ1y!`IiA
zSJkOMKx(x38VH4kZ*U-5uYw(Z;H4{w)*Nf<Xgs4N3|Dv!pb40Zyb@*(?^4G*)e6T<
z*8}KR*VAx4fPZz(#G<w_=O?S%cA^dYXe>JR%Ob?7Vgyf{qpy_(zgvuS>Vj=cLo{pa
z>7>`QufDBBFQFGv3;F@B7jX-I>9Oo}NgLE_GwF{*7W7V4osfp`C!~n`<x?a&ra>D{
zw)N2Ge`)&ziIhHfGEX#uH_&MpKf(LB?vesIuAl_mzgzL^#-FF3QCH;Vl;)~*24l45
z5hQEJ5XpdL?T;v<j~Tma?4=bo>L1Qt`RP}9%>a6BA^|X!|NjdB_-jxI_CZ_l=Idxa
zYiv&H$kZH3Ka|;-Ec<2Ut6=@}QDUDhSUP#7+LCO}G^NX|nW;%e<re<5;1*IN=m`B$
zh?2m14X;DTaToFiV1}mD&U8&<ZgW9X?$NZPDTk9}peK6>y{QB@UeeDHPTFlZ+|G
zw6xsRg7jROB|R<Gpq#_oiOWe1@|ZUQfao=HuG@eZe387J{Tl^<-MpOj{lY=cP6d|7
zv&?{8BBXG}xhwA2@AxVlz99blN<kB-QINLapmfMb59J2s*sd4^`f$pbO5WXzC=`&x
za>h5%56KxP0ZU4iv*KA7w1xTwa7;q_g#*D8$PI$hF$~8E;@fbZi2er?M%mste&UVe
zXw>l^U;pv=3AlcE<A*oM-|;to`~iw@_-5YiePgsE#^%sFbjpMTTY}1^x4HJY^vd=&
z+d<VErHWl_R<<o`Gh1)B)v12i&eqim=Gg5}fga=&ZS27|Tbl}v8hY}^U#V|9MGd#+
zeRuR`bsVs@gC%!6-!asMhrUoVYP&o013VF$Y}v!c#M)vvv%XnW?}_8K<El6I3{As#
z%_Ee1DkTbs-safn(El}Wy=$LRfbFw^2aRn5p!n!F?|1!!(Et8V5!?Tv0hj+?y2H{4
z&W-eaR;wCbc^|Dib=dME@s<w70i0$7iR3~!43YD}VAJyjXg|>d7Zho235`~JX|gRb
zKMD8VG5SSkg(gI)?#yI@*VMn7sL4H8YOkr6)!UoP8&pmwgM1I4LNhLF(2)Uk<K_Iz
z`%n9KnZw$5s|L%Ml$3(P6isP4_C1}8`LGTCsa8v3VNy|YZAx)U{<er|mVIf7@!5IS
z{M1@&aYDgv-5k@N?Bt~Mf+Sf%N^N{WQr6S@MR7Y)O$jN5mf}o%MoDUkr7*ELF<om5
zY2GUvcLs?T-BJzAWWM_5tuGI0&{3GgrM%1L)BbmQiaYoP*xmuWA#-d1{e#Gs%)-H1
z3V!CN*g%le07XA#MFctBD?u3lXkwE9ndA;UR5;-W2L-Vx2t4t~zOIQMeLLvuvFdfm
z&h*ChY0s+hG&GGqt6lNTHq~GdSc<yWNyi2SjfDV=HK2Xrpnvh8Arl=8_N*OvRe&^0
zJP_=>4S`SY@Fxs`Oc(;0h69>rvKnWwBS-<;xgEr(x6DibxmxA2Gpm<A^|)EH9Yid6
zlE`ZA$fdvXYEbwe1N(_|KdT-L2K5}y(0lPl;QV-m1&Pfu8I2%Hxqd*a=-hRD^_dxG
zXC7bNvBpsBx32_K<#>IW%yoQloTB&TirQB-&)3iy;JKCM^{C2fZQ!-8vmGcos@_>`
zs?06jUahZ9Zjxoy<Tte`n`(|UJ*$2ZP3XGOa@Mh}*zY@>bQv>rMOIl>wlW*yIdawc
z1=gI%9Q>fsugF}o-=uuC4DGI?OOHNR`nu}nH;VJ$(-gdSwdhq<X}ztcN?BigpzyFd
zs!?x`sfa1J)}+_xR%RYZI~n&{!VBr=Onvb$rJmk@W#7k9H=}Qc{2cNd4s%%jvfAlw
zjKypNx$@+o4rjYuNztkC8QK0BIZ;`gvb1@o{Ir7NC4Rv6u>6NdZ#d`u?6~~Z{9B`t
z1-<VZ>wD7iVv{1TrJ$)^S%f-D(W5jPFReasvb;xyJU+{ge@XLF!sW1Y>t#pxHf&n1
zT#>nH|1Pz8XL!_BlgzYrRr(xN=<d4q9U9-*bZ3U5<dp4ji?a0RQsh<iUE$mQ#V2it
zTb1^IUZ^;!Ld=)u9WQHM*>QBka^;w~<(os*A)DqVV3{f`x~wu*<2rlCTY(;`{I>jL
zIg(cYQuReK+EM8DP0?Fb7i+$1ey6Rcv#0a&>5I>wJl%P&@mbk{muvs|59Qaf*EhbW
z_U+#I{v1%Pj(mLjABWnTWxgjboH*Xqepc3gw(i1Z<%PWN^t0;pv+-Sq<A!JSha=k}
z+Q3{@Eo)t~-o9D&D5Fm-&CSWq@fy3g#5FPlH{@3=xRQIeu&KoVRB=ZE>_cH?QCUG%
zdPQ{U<|=F`!^+a9%Ut<>^NXIy4^bDT=A~pM$7F<O@5nV30ofK7Pxk}EXhDL=HeeUy
zQIe+8ljSc|epLNx<+&2Hv)J#eoSyvJVp(xRZAqo7qS02?sVS|j0sdQYpuC~XUfxvR
zR@xC&6<?lL8eI`l9a>vlUt%w-s(;S!0?Is#=3GHno8CWo>lpI)FKe$jT79zST+OkX
zwj*_?YR}i6x1XsyQCHPo(E_mQ%IeFS(o1y3!G*H?$*YP&RM{3=S)>NP*O)ZkUffX9
zT;l&u;qy61(`3n|nI*aE+#T^)mAc-5XO|S1md4@P{+a8x;&v0(<c6l}lPv!GGqf2`
zXXtYF=cg8=%=ZH$#RyaZf<pzKiALj*$a|Ed3ROC`zK?2=+;_?m-dB0zX3a-O56BwZ
zzW<P~Fc;x~QtVmjTJBedcNXIDg|oWY^3xi76py}?mH<cbol$b2>YMUovWmkUrJ&Pu
zXoQi+mlzyVO8Y8*2502splvA@57<9pE;b(RGHHC@z@yN7Q&))11UB+fcs{K&H5xCf
zKDlFG%!H&Hbw@N1lr{f|?xO7oSi+$#0O~rDel$eo146*S?V*`hq6(0H%N<Ho@!0#m
z*J19L@68?U|B#nen2SNm^i)=c(0FGtnY+4+tvIcziQ=Dj&;9znxqt4S`&O9y`;6;J
zn0pV*o!%LMy%l&q8b#}l7}G<Pi|^P!kzO=5>P%`pACJ<RY$07He*{^&%XGByqVkgg
z`(kOx(e$n(Y6RW7j@T~hj`;hm<z4dkiKTB6*!e)sw?kn73frLIev&l_tdOgCcf&jK
zNnnm7?HqPRu+U-<s+@Atz+son`wNu2Gz@F#**#!Tp2Pn2<Fge&f^iC2760HNh*9{7
zAyWyUmn&jO)g@N=ftFx6cE|^;ij{0GF)XQZH|~buw;~e!R!)N7h=-X3T|&g`46wQ4
z;756(7rN~XRtV8z;Djczm!P}u28-8LIK#nKEmR6oMgerY%larU*jlSt5{h^@wZmGg
zF3-&=%Z131{ESSEYm#dsjTTC$v4i(T;toP2t}}Q%j}}QClU$Q%cXSb)T*w<jJO+Xz
z**X|Z?351#KLds#ms~-dBrdzlj)PDT?gOg?4z}{rn;^LfeqHc1NMD8%Fa{JO8#vUZ
zwY&!EhK$8n9tk#+yQ2e%#P=t2;|9vanY3e-^J7JRaehI8+I7x#jxH#Gf+{1&-cVIm
zRavQVoO7I`D=VR(YOv>IXr6*_&%wUIKAOx$>g;p&(WnhH6fYKMq71sza*elGHFyzT
zNPIVF5n6Pb9n8$&3wSgMoXv3B$C6Mh1fewGk~#e>zp;A#;b65xG}uIkv|TbiuX_H{
zk&Epb2jy&{55H9X#uX)4CZOX@#Zq2#rw<$&plbvIOi;aXCP=0bJUn3c-RxUQ+%1X*
z{>fL~SNpafs_Cq6Q#Z8rzSI7;tgaj)tW-6%1zF{q_Q!hHHYCdG6KgDHrSE2tnfv2@
z*#3!n`zLrG>Rg06WEV2S+hbHQ5ecCgnnkz+d`6wy7t4G@cPx&bJ`uY72A&*2kiR()
z6bXoV6U+i~@qib)t=M{V>dOo`ML-S4(`fXOqhDdqDM`!8!N1|({Bm;AN^<Nu_R=PZ
z0{IR7=4ZDzmD+P;pSW%)8glD0E7h5m8O7Pz>(==Jist4j@u&|VHkfH@Du$@Qy2AQ$
zyS=B!4<fT`i|dMO^XxelvX-cbng}p&Jj_oW2vQ6N4F=H|kW=amat6_87Mi6vbH?_1
z)EOc<`9+=CV2g^HIn$3|{deEdK7nW|F)T$p{|RpZ2EL1mh8*`i(hGBPQqz(%5_4o(
z`FUBHO1$p_h~j0cD$B|$HD^1XYkL8lWaV`Y&C1ra#uPQgFhD?x*H~QhFpsjGg$gtR
zNZuj$a|a#f<`fs@YOF%s5Dvyc(!s!o@fF;TccZyDzB*fzTLocQs)~xz(h5yOmA$dJ
zF&3qCqA8z#FZ<8ODDYWiQlX$j{QhQZN4C1C#$H^bYH23n4@3`w7l94=fqx>Apu-Qm
z??=AR!Q1>cw5nx=g{6hW@|2gSS+|amKUv#qsXH{+_oKfB=iXcIlJfGBa)=<j$oKTP
zm)!rq)*PiepI4kXVS6Fw1o+83{9;b*u*Jj-ss>elxEVFOi~iUHd&I=pcASXucdT%&
zI1%%L?ZgRx=S$9)Xz&P5Vg--jbHH8UD3D7bnD#I%oeT0z8Q3~q@{90U0|W>Iq7TOh
z1NXBNgAP&M96-(t7<7ax5CV`lsF`;0Kr{)mF%V-31dg>2)dn!v5Y0Px-e3)^bLR_u
zAk-tD0EPi=Wb4oq5)tMOdh~ZfmOf-|vv(;;YY^!I0+^8?SJRo`dC@ukP#kZu9gS@X
z7<kv2;1}CsPSYvD2_V%SN;}@722TKSP%dA~y1@1bUQVzR+b6Sk<e(D#t!@=R)ba>R
zCS-&8Ac`H_`5nyExf3wSe-KjId?+zTryShb!;;qltDAkOl@Z$Z084;cCoF^bIV@Ee
zi3{;N-Umb2864mq;zq|m6=t(Nu}cM>#x8r?A+v@+MLw**Gn*WdKniw<k%W0p17qnF
z7Hk~L1DMgG-hq8$XG)Z!XUAu@7hd@AL%*IL+h;ppaHTks{TwNd6nD{o`G-4t7TR|1
z^dAfy)a=}0TiCPHe;5Sv1Q-Kfz>(tq8euTdsi8Zq0<U~zIc-H_*$#`^k>W~rrMOat
z%m0Qa9T0xxB&|C-8&94BV}cy@fj6lSv`8TpH^P5~fbH1MJPwr1O5YI>fq5L>0N%zO
zpw)L380LDgt&xsGhe10dgc}3xt5^u(a<_ofE8Q_ik&>4J5mvKj)0vr&g(IvQf*&EM
z=Wz@dRD$rSN=YG=v%iJN&b$_g?5u8v$WA1*LC~f?kA!<xw?O5$-$EQ;OEtvt<v|=D
z0YaLQ*uLr(RZ9!0{B|~kCReAVC{yxCY!?}Zv?G<NDN4tKC;{eA;^0TwT3Fxq09xX_
zfEMzu^RDwlb4(Sc1Fph5Ft$6?l#G4j+jm)%fD;_k98=$AZ;)ZCU>H=1=V$Z2@4m*i
z!)jf11|vI|n8CTKI0gr=6lqxSh(fRxsD;zUZFwYAz1w8iX;p%+pFb`A>8H=%KcT*I
z^vK~Cl@~X6uZ!LX%cM?9PfXsuNtT-rdYCFNudJd#gZ+NZs4Z-@H~OP-Um>6O(8DSS
zoDRl3UI$DI2g5tT@K!iGt*{MN6a;gygZes?bp@Y<Q+y|T6Kt059O~Q2yX-fmp**lo
zEFX}QnFa|ng+o~G3IbpPBas942S`qV#1#r*XbfEALuE+R%wv%hkH0|T_e0TmsF}<@
z%c>!A_yRcap%RV1Aj6_&7Kx;2d?<pB7w~wfnFKWq`VNg_@i=BDior|KQXJDGL*oGC
zI4O?V30BahJ8_H*kAt@3m>wJhEtaB~olpbt#z|334}xAjC<c118n(C(eCF5#Vdr28
zhfs6)`GWPw6Vb$5knrR}JjqVP6Cns42{o9+qa&z+h;-6})RHXpqEQT52cDgK*8Cc0
zG&z7K-0}nR#z*l{KRgM7F@e-4Oc-Kq2)w|D%n-PF$ltNZv4}=v761g|P8#_5cQ9Pg
zfzAgN_$6F~iy<6bG~wjS^VH4)XFmPX{1X$gNO>m}zo^*y)xKLutVI8W?{JDyFB1Q@
zZ_8I|ht9Q2;aCbEKK)ESZ-CDnes(Q&ErZV-ejfVF;b+G(wNC)OE>Uz9__G-Nz3=RO
zZ6z2L7<36;qB{jz2UcO}R4@MkgsPa&d5c9es2Nn#RuU84VO2XdgMo<WKm0e{*+Do1
zX$_w_T|$>>XE1Z^x!2y&xJLkH-3zbN3m%kH8KljihA<z?zDB)DACnGnmoKL+z=1lg
zYKY9gOv5Zz?uFpKi|pFxHeDmGDhSBGL?7oZ>JNb<b)JA&LAPnkDM0FMAdCs@xyuu;
zvyRuvCo)4LG(dQ428e?&aScs^xnnv77#8?Krn5r>-ug>0nsnuBd*6X?d6;)zd+r*T
zW2CS(mmnq)+H`6@{E%?I6J&tp0rb`DATh%L%b^w|O)E&6u#ND-5T68qh?oB|I~X|p
z2@cFJ@H7ifZHSfthPe--wSja<bhq6>qP6Yd#K)hyrfmUFjYbnTCJU^_5+x3N53hR#
z%hh$(x|pT}S$1`GUZbk5zWG3NVQWdVrl`BPyIbklk4}H?SP7qr0PoF%gUtaaGMsqM
zLWgx1?>y+dy%z!%qyh8|Q3L#d1ncPA3r`1b?*eB7@SU5^Ai{UTK*kTiV-(5hX({SM
zd~#Y-s|GzOZEb1-=Sncs(wLU4DMm9C=_P4d;9uOpB&F3gYEqmc8a&F?73#_=d%0bO
zOpM)LR8Xa<E-cT>QxY8$jL6_Ykc&_$lHY{ri9Qr?lgOz-=rM)PkfMXZbcU8L&C61U
zPD*?Y2U(X+x>f4h?fg<fzFr}*Ag4N$GwO&cJCAl`--PPWZEC!YfzJ%)(G7K-t*2B%
zn6$$S+t;wJ^^7~_+#QSl*aqWH(EH_hAyCyqx+ikO(Z^Kx2dHaY>lZ<A(=7g`ONjC!
zU&w%q7M#wYO#wgt;shv~&OXHi7JM}`pTG~}$JM*`hV0rIBFoH3&r1e3!?CW`>c;v8
z4XQz@C<#qQf2!cj1MkmH#g|cl&Gf^j-P?oJ;GFSuJ$4<3t(D<3({U9}#P2J0<+>`p
zx+3xLwwx_^=b~}Sgz9{Iih9qH1F>&>{Td2=L3RG-`qbw&u{VB6y{SUe(A4wqAe9D;
z`f9Wr?Y)Yw${Ma#zj>8d_#v(fJp@s(pg{&fWG{s1xT8FPC^iG04cu0s8#oI-dO3!C
z)ukmxrS$QQT{BkW8dtF1<*URuP!?W^j$vPQNohq19dkwZ{d=g!5q!$w3*la{n*$Ow
zUgQWy<G~XIlnkz$NANMc9)F<19{|wx2r4u`0*@OF#)es+$RgGzEM20+!Y5$lgGgk%
zp73!5klC!oV0sw!Bc3E1j0Am&#K2=_`gF{E#5?~3I_9BVaD%GRO91;@6o!!sBm6B6
z15UVJlbMifg<$;}h~Zb`mtatMfFQsDp>I(rdKs<P)VJ4k9D*6c{^bo%*zqFTqrj`K
z0}ED-L-YQ7lH=Y5N2KgwHM;7!Gd!UgSo#r<?xL-Vy8PPgDu`R3{%W!+Gdn*!U-Q%y
zx^DBvs!hZp`8p24ui%}UjKn;%MQN_KG$g9wdzQ!0!*~kXHdmINqJJt?nOT)xo3GAn
z|8YyJs;;i8wo23U9~ABIMq0-RlKO+!v$9Ni#<f*(wCHHTQ8hm4DxuM%Z^0K9i_k`;
zBN)pShx6Jq4yu=e0r>&+03P}IdMxon^wJ+EegJG^7B0Xxyc%CLKZ^bQ;6Uhr6Dl5U
z*PMIqT+i`;$Qlk-w;v`8L*z602~b(lJVNvDvqSXW2=x9Z55$h2lomT!MMg<l_7J_-
zHfT~Dd+4h2+UiO|?^Q$eo=@n#pgcEU8RIZ2y5mC+?^f*&-5(yW8MK2G8y{J^t0U4s
zvg29HbE<7D^W3wY=Q=bU=SrWCKPMwN3vw}?0!KqA{8OyEii6foor;5Skl)~RG&j3E
zvoJT$pE!>4@`|!bbNtJ)t8(lGj!JyO57)!Bt(Pt>F0vKDH>o6MXX+Gi=;uJYQV7SX
zDF7jBiywIBDywp93TsRJOKtE~7}!oUH*Z3GK79S*zYT3e^>CeVRgw<&V*iqIh%Zr9
zSC>^(g0^$Bwx+V7sNNq3IoG3kXx`16S5eTqtNx(10=0Et1*sM6Fn;`rt0#cl1;ImD
zSRpS5K1Zw^3dHeOM<bRUgm`}iaKLWdCxwbW@|`n8%mDsF3rFyvRY4jBJB8F;=J?@G
z?zYa`1?;+57Imk?mw<w~1wKu1S_kjJb%SGBfX@NRJFNeseHdshAf7qlKEXO?44hY>
zu@muwpA$d5brnd044QhC_)A~aod2Qw`<K6x4GavPS40CWDW19x{E6OMG!fIoX9Ula
z;Rdt;t?NQBs6bMM?-z4Ge3iIcehg0gc3g+W8q1UBErzW!Ol@613;W+2<YQnF?Whik
zHJz^<diB`LGDN-n#tr1JLUuTQn$mNI!u^CVo6w&w_!i5R3)ckl6*hv8Y`)<uE$yiX
z%bpRmHqdEaWbrzz&j}61dibytPIBpstZN$jH|}w4acn{VMm>&c>N|F)9h5%!0F8W~
zOX7qE><;<;HLE}y1wH9Hs3Sy80@-H}q@3Y{UXUS<^Hw5*49O3md?gc|=`UFU{A{4D
zfsjB9Qhx~vM5zLGEd^u)kVD*p1(97&Lo5)Q4r>Qeb258EQC(D1Sf$265MffCpAA7}
zu0Bx7gPCP)Q$bU99Yk<~t)Ve9xh6@Kl$@ImT2Y@%PG@Hoq@^K<Fbala4CHfjLF5_u
zYHBmYTRMUudN&v%n?YDVRCz8+4Kb7Od2C*JE+`0K!waUt?g3F^4uFG$XY*;5X|-7z
zM4+bEx8@vB9RU3Sv?elcZCzy(M2Q8uy~vCCQMNi_L3Lx4ugNL@+b8t{?&7K^pIilB
z$@rdxctkiv`qr`mcoXw{cF8AeAui}Pk3!+&-NCDLx?%USSa=6uOYlMRG^Vz#5rSMg
zat;+9R1X|OKD4jyWJTxE6J&f6a|#i-LH$5h`*}Q~e<B*8hLDBF$Z7&lk^Vn~y$4`Z
zRsTK?LWWU!l_c0E&AmkgSt6r=3|TT1WQPs1M=33(P4`My+N4dIr0L#^(H1CB1!V70
z2VzA<P<T-hQ4}qno17GX&q*q;@Av=x{r*Opb?><+=iYnHXFSjIS=0spgd563i}N>f
zk6XpVsBFQsxjg;O?JtUpi3k7a-Q)VbjFx<d$WDE7jwjy<{-Tny;+h&2c=|v8CUZA$
z)1+o2NNcr`@A2LJ8|1aC<OQ{}RSXycb%1rws4H3^Sz*o9$sRkXmqh))#z^p^5OnrL
zT96U!bV&0UA4tr&01wKZ1Wri52v{5IWWTVOE3@X5)PWd`LyY*b0<Rx?Un5OP9qk>T
zvu)6pLrfF{lxH+gg0LQH5P-V>h`o9|_GVmVuA$1Ut2S;}6C%w{$x2C4(R#2LTireA
zGXTz?AH*3;N=>Ee2jA~L^BMn|dECX&Z;-VqG#0AMi!9bMen9!STMt!W*k*AJ@r}uQ
zOwxJ#0$W;D`|_L0>bXB)X}$J3c{4?dR8nb)ib(I>Bhm|}!`AHMjyMjLHP^%~-Mo6`
zw)brZ^7oZWu@o)zM-Yj0asEV>kgepk&VHgHWG&VNHI`!fX8XTrvGZR*G;ak;<wY@>
z_W2{SfrA;dl|CgNoxWurPdk&P60(Nu^~V4|r@17&e~&0W^3bDNU~(%E9)-op%uY-c
z!!*o*9Hxl@^o{X&85^7#&^;#N47#r>34Hv6m?MO%%Dp&A&K~$gK==z0Z!KOreIzYJ
zA#wr=C8jcPn25upDggj}Cvm6@vF=Xfc`&lY418P3?p#c^TJ*y6+{M}Iawy-Ig>1DK
zY~u>H*|&zM-k0?pe*4j*+qWO>+>w@4$0gOJ?bxYe?;qVB-jj3QZPzMy(gsqpp^5YA
zFX&!-O}Fjd=*mbQYb6XH(N}FJ(GedN384c>e;Q10bUcFbZU6}(KwzBws*Q6FYaiCZ
zZ#>h|a>fHt=4mJiy?<X6S$X5tI(7BkyVX3x{}Hl`$itDual>OObZ6j8`8bz?L28{2
zw?jE)-rUJk=AOM;r}^|8;JYqI*Z+LN$?fbzkl5X$ltsyf3BcYCtWMdHv^{aV?~eVu
z_U_y-&9MQ@s@g$iq|>$<&YF(d2q6oj0kB)y(C~t={B60uI#4%?j0yP(YC21tkd&N|
z!6z;?Xbnq3Q^JzN5~<{SpB&GQAwU;D7aGMQZ2-R`&61Xr&NZyxwPDBF#4vqW>NfgX
zxDR65@rf!rQ<9LESY+hLz;MUbg3zK+-;i~|8$#AgK|X~5LkN-i*M)PyeIgfQ&ov|Y
zKxE(5B-QHcQhlqzLP;<e>5J54mbj=OuLx1%qt?^b<J)s<^J)u;WQb3OXUCnOBXrwd
z!CS3nyIqx=X~Tpbv@f`6>w&`B{My_)@>-2gp*gR(Pz9{PZ%WcbGeJfMYUJa}R{xq(
z!4Wm+0@+>hv3$}5nLGtwdB2d)!dJ|$Z2<q#=};np0W!PWmvokKvG6zmVo=VOkrDU)
zKtN@rsv8tb$qS4OD<~exz@K@6tixsUHc<>BieX4oF0#rORpS2BDwoUT1t*y&<5l|L
z6PbO#Ve63PCayBPXnBxIzSa7(#u8(Wjs~D<?7LLN(dTgeQ}LdC^eMi4!E1ZxtAm`1
z1~D4Fj@;$=bBFla`kMM3JAZz-nY`Nap_FK<C(x&HdIyX^DiJ8l2^usCl(LLGO*xKE
z7Vo-LcHvXi#~J6gT;zUU`btwT78=ln^RM)4phIWRSife|s=$>}bToL~v?1%ZN$GZW
z!(kqL9+nsmT)E>$aPm%m1+I3V)#N2Ly7HrVueeoKd$91>F;#VDO?nmAaHRC?IaN1U
zZ&vT<U}Vl?WY%M3o+FBBPX=f9N%<F<K!&uk!5R7AgX3;hPMbCREsnN{>C^W|P??H8
zt(!nK+>8$!$*cVzZrvGPA673t_b$aqj8zAT<+D#>a3p8$?kzvX?;}qU@g5?BC5kU9
zNte%;U|{64t<grnWVrWq)$>-UaPaW-@T5p?cToA-<*J~B<&ohWw)w!cW5@;|KTS&P
zdM@^C&=Jm7WvQuF;Sk3XkA)rN%thJ7MXHv_mUYKCt3-bAB$=I!*|QU!uBKhZbP#=E
z{Sx{zpByqec&nOX;AWqEGK|~B`?q~EWY@agEBCD0xAy$>Ep+Iw{iNP-%OAfs{d|<K
zzmRj`Bh{xd7lJQwWU`oz5l3(e3CDM9|BKOtfzE5iJm)96rOcAd(1@}m-6~^bjM?Lw
zpoEb&la9;G<lJAn&$z!5&Wem?I+KheGgM?ZnRg!cb3dpD44-*|s|h)lRkKri=>!=I
z%ex;^FJ#^vx*H}$k2uZ0HJ)?}>4_CsabMZA&Jc#Ys@R)F(Rw9Lnly(JKiTo73>MNq
zq;8P#^nSs+0)*yGh>sxm?VNs(q>+3~)5-AR<@jg7zvM1>+fC`5PU709ONw3o%D0y+
z7|mswByTJ^_0cCMPF%l!bkVeIUby+#Unxi=_cmXCea8A#Yht<dcvEDQ%I!nrvLqJg
z)0rP;sYqNOqxi_K(^tsz%#RoMrGdQrB?(D(#rkX9f3X>s;gSNn2s#9Pz3USvXoF>*
z1qz5+X8?tr|2n`1gQ*WEI3#r%uqSZ+d-PuzdxCevO7{WvelUF<ye*-x_zM`Oah5`T
zi5iP_R!L!DiJ3btM2UonutgqV77^N}GKeN^MUqcDO4d>a4`d{OX2>D4?1)DchD@fD
zkx%dkAp|kmQ5vKI{Ml#3kIgO2u;~m?lEMpM-UP%pX}gRT#qSnQ+qz-D6$q_np!we%
z#v?kG2bBWvH=AG#w*FfNQ__W`u+YjV21KEFU3k~oQ%RRJQ(xlui|RfS2y{pT?e^Yl
zoa-{#q3lO}fkjxdhI{XB1CWzLfSViu(}yU&meJ<>;tZL)HC{G=GR2dFGCGgM(hcOp
zc<#XBrr@#!>B(h9OJ=BM1i{H1Fk=7*NWK%0{1(am0WAXt1hurZ6dgNxgexm*+I8T#
zlzdnWQp*O$sKYg~>3mgubySt5{$3Fhd@G5fmb|miIhNGRb505zc}JO(V|1k3puUlv
zVK8KvQ|##wWHRMgrSb{-)fbf+_Ed`@!;qN;Vuv*?H#5f~&5~GivT_Y}>8uM%b55o;
z-2&{m$(U)(uo!Ha)=Zn(Y?0OnDswC*yTN9#rXh)#k(r%lO}85C#+)1}!T?>BW?Q-)
z$N&gO7?C!&r8$gJd2c<)gch?+dfA|~r&?1?TuPcDJ<V>&%jV_J>m7EhjX#&CG}$0P
zV@ffmr)Q^Sg970&18-w9*`%(;t~pG_3l3q!?yMtxnd!T?G&{m;R=oLg7VQ$ITGp7=
z0HX<~kKqLViyF`ZX25vy#L&qLUWauretq((&qI0l`2SD>mMinB4LhRCn7V~eVN$Fu
zP8}EPK`3b5+K*vxxV7R}@zhr)XmR%Is!M9}cy<ODXb$ND@}ghYCgxcaZirX*0kIH8
zPed@g9wrK4O67Kc`A~#{lblPHh)*p=Y^G!o7xFsonl=>4h%WV1ykvRAQnh@pe{fv&
z4*p=(dxuqWYvqlw>o-&+{ZrCN-X*Vc=MP?M_+-0u_wDcZ{HT^2{IRNumXT-n?|1B1
z=UB5$IlSCH!4a1o75#4VyDL-+@<Yp)aV2XV^k2&{Kbw^;dg7})2V=-TsUf9k@1eZI
zhg5&8xk1I;y7kHH>C;qngg&E|n<mgMFK<}W8?E_|-7;dsv_(>?r_%!H$Fxa>!;Y#Q
zJ9<UL>g6hQci^?554dATb{-)j(lvyL)qjwGIrcmNyA&2j9QlLX#>zGk0YGw8Y0t7}
z+PSpKrBzXR^BU&X&u^5LYzx}8W!6yo_5yY2rrM%#o=*P_5TfpV$aHB!P1v68r^wsi
zT~yTvH^kL(o6l@H7j!ncBI0PIU5a>aR+@U_l(_iK{L;vv`C;!$gXTofeoHlI-^ltA
zT-B`Yb9QUn=r{!HR+Diroen%7dND$}<<__Be^h^bp}gTdf2j6ML*-FvabwA+ds(pZ
zfy~tgkh^zYV6#uF7?F{H%UG1<8ZS<g_sWv}Tjow;P7#}Ez!}9(ezcFoJx1t1^w!W3
z?lJEEBEPQJT@FbBQ7n+nWBkAS%56!Dlf${UgK76=Q`K+H+Fm`Ar(3_KDI{eK*(%jt
z3cdKT>dFz){i9u6Ud{1>I<?i7Wwi2T>7Ua+C0nKW(N#L#O8VmTb*iYcu)G-VbL#WM
zVB#}Tnp{>JQ?dU;^5Q{tb#;WkoZk^g`b@ONNX>?<bOUB1XPH0aM0P&tC+>@cw$|lV
z&JBAfW_sGk2aaE^xi)jdl+Z~D(#vy3?jNKE2l!>$n@$b0gjsPmDvM|;F6?1sv2^RQ
zIPGi|?RvKFzvprb%}a_`)ksZQMw5yTAzf$>(l?k(3k}H#QAb9ZEm3?k?uKUuk(V;1
z0kjJRW^{l$G%VY)jeiZi*l`QV47KnB`AX0W<BPJ2Ohx?V!>7+4Y>~o`MOdo|%T7~g
ztikuX2)V9J2nk6(w;zD`)Jvp^Mu}<N0PrW`cm#XL(Izsk{X=rk`5}pt>>^E~ZbSS;
z*Zo|tkcpTS>s^~L9X82BTR}R4cv3St*PGj)R#a0_X1e$m*diS>$m?OMsKW65c8;8T
z2qltca@XV1dl(1Eoof*~XJi8x{H;z{FSP9exv)nilVk%B2LX|SCB|DoZk;N_`j5Ha
zfm4p<M|=y4ULFmPzn$I$aGoU|T@S?Jkbmijnd$tTdL%n#EOep74qIeGTfWBKCV?o8
z3(zDb1_ZV=yl(7fg)b;(cNO#HcFCYH_%CHF2^<7nGeqOzT@9owex!k=QX|ffIxa3<
zujdnz(qdy&R4v(D8`=nt-?<QCpoX3FoOH8gdl*omYH?X9at3@sp{>+ZCKVh;WeoWp
z!RedSOtNV<G1Hsm75aOb)MUysu~w6k_{5~NIJDYJvbk>SZX+jr6)3EAuWfXHB@Hz1
z*tT1Z%x77N9dMLF)@rHLlYr?8v#Bd{f!E2LX(Zsj_iYzfEdpHoG0XPApRP0j%oYmH
zH372)r{QV58!G6OWQY(cDz%mumZ_c9;<EC@SolLj+>s(E!38L{r&g!da&(FCyXaHh
zTSq6V+pEPB-a39%*a-$kimsk%@VZH>T5DAQEB)a1F&9uXUySp`T0k{@LV^lE`2
z)43IDw=N!0st66~CZ0kgZqupf=+wI-NWS?J>DKd`AvZoHk~h9?2HX3Y1LW5basVP9
zQ)yo**yCs^M#IQ5Nb|UVQ_>=`oZ5(p+IL7vwS?Gr5E~-s_*B}>pE|w<1xf*0YgcA)
zb+^h|zWy3{CmmLekB({(b8c4RO;#JZO1@Pg9MStcc@vM`bLbNKZ5zFcKtUEbn>}!p
zZGeE@CEuw?1bqojhSYJ^d`n@WYLZO8n}rw>Es0jd(eU;o`W^ijy-SPeHf|?YHBcUY
z)exx$>suGuI|zWULPQ5<G$L*G?di3?X|+)mVKzGFh|ftf%U+wdJ-W;{raaUhmPThS
z^jSvcq<c2--DcmG784E5Der^i1@GO&H!M9WDm`o&o#R7ioeZ;wViVu!m^AEulFss>
zbC$6U(!zYx@m+ZgR#f1G@P}<;3-h&yRYcXMlR3+L7SdU1o=tqqqPM5j+R3bwK1b*r
zTUdEiU7Bxg`gVI+Ir1)?57IN7D50=CwOnnpXJ^~^T6;x>t@a3+<3naGME9|wFZ*d}
zwF}8CA2R1it*xTMUh8Y~{4{B|)9fZ5g4hilQ#m<?bFr1pNm`|R@0HOSc~leZ)K=xx
zIdi6R^joplA^jQNV^fDi`s}#nv@24AbUe<(n_fTk(bex;TlnU~`wlmrEUYZ3DyWiH
z=N~819cr|9>srtNTrC5pzoQab;fOx*LftZPakKsXgDT($l>er~IP`$3R?+c;=JLVI
z1J`U^Bi$S_ZTK?gH^FH_7yfoXFF)82agksD$D=KztGZQI*;IJI@}88uA%@nc6z-8f
z&wl1HB8TrijVRaR_cE(h9`ZU)Kc*b{p2ZNI8;4W}8t*dcC_(EXhsv|dEoI#5YTenx
zsv28OK_w^O`g&kP^nnjl4MiVR*0AxII_LbAPcB~g7-E`YdF1Pt2Yg5rs{7X(Zf!qC
zMY;m6Kv$qEifCN8Z$<M8>7<scLmsIc;4}bqQp|>x-8rmP{Gw&kZa0ST8=C{0gFle|
zICm8pPgQEhS_q(TthBExUc+O2aIMH-yl~)+Nh$kX_>Gp;g=;G}NYP;<Qp5?W8I>~*
zEaC8zOa>91Zz8H*jAQmxTSL=B{HoWhE<R|wtcF*>Vq`3j^3St>Nh80zDn<t4h105(
z-<^Zgs=(6orYfi%9`vIjS0}Y__$B)mHbxdk6hstAb@^0!e23~*!?pXI{D)ga@!^2W
z2xdz@zHspqPX6nvE9d7-<>|K)IayU%^FdLA`hx?}fepwKVnEe6z~QsH)z!SEtlSJ~
z$L9`@rw}qxSe0ZZ?E;f?u94fn1iwd}5N|Rj@NzO|L*?4S)fSvu3Gv4ONTGAbVL)UE
zVz_0J;x()6E7kOk0N60YsEUkV_2XRrgJ6v5MkzYe7;<~sG8Ju>u%5nx=sX((KqW6X
zJ*c|K?fawt5$WoQPW;bH1;di#y$@)YrIV<PlNaoaj?T8k@T+z6bSu>1;kJTEJ}_u)
z^m6s)mBkg?JU@AF6T54s&A#|ChY@*a`T(j>4+y$;YdaAgt1jTH3#tpMicU7-E@_sw
zwtRo}k*Yx=|D?&OK*%B|6xm<}E=lxPfoPLg3Koi|I5P6v=niqTW1OA}YTNLTi@3Pq
z!DSVGiT8Rc*ojLFcL;vzvf<M<hj6%c0+$C%2hFA9S5cLU4!Z+ue(ZZ6*h==ZVWS+P
zKr+mMNWb8I@#k=OjPU-P!$rHxrRb8esMu7bZfU7JaE!N_MDo2&WWMPSq9X4w@2nKC
zEa35r5T$%824FO|NsP4LT#@_`&{n{2$=`yk)ZoEHBV+ii!9VQK0D6p9XJj1!uMX4~
z+ea{o<%b?=7+$TTVFwz{kuUZWi4e;jg?1<uA<#BpXb|R;FZN3F4gTQ?=F9^`n@i@3
z+_?5NXprI!JLabzF^L}rGY}ZMMRIz`4>1T9JAemRW@W%KrRN}jqujjEH*af_w`GD!
zLeWhkmhC`e<na$boRKkwioct*ngA>N@d85;c?QJO>>Spt9L=(xV;sbuabP_HIL-T`
zC2wooCJCsBb3KFN>7F(FNn0GrJWYBNxzRy1Ao~`Vm6sMD#;yUR^Pr-vx<5;^t9Fw<
zI15L}l*a2fQ>s4LQRg^Pk$WPtf=C_mo3HHFuhz)F#S_`?E>q^)kyOga&vaxYrby+#
z;A4ov=A;=x&dA6}sf!Pci8V`eO=0obsuV*~R$5A`K0i7>Cp}STPfo~Biip)0Cudmo
z$>}+e)=SGUXBQ+}Oj3g}Bg3G!Ch8MXQj=44shP%@*rc$AG--C$W>YqAPO@%_EKIhh
z@5s#0EHGuI79_?S^YwPAr+a!^9Ng!4z21^pnvt5DWXd!o13qs{%-b3pZ<l80WEd@c
zW>T6xJ;U2$c+|=1hQhFf@a#}&RN<y_J?Aj&GCFdKY{q<Rfwj<PE6#r3ym^x$CKH)-
zW0KTpwPu6*VzF9Ow&aRTTY){PD8-y0O)wY6N2?P`wi?WQepFd%MV>S@GeU3Vl8w=o
zIr*lH%*;$<azU&urNC$jZMK;c(zoU%+9UHbcW%pw$uy)z=1LQ+NqGtOILB5;dPRDP
zxjdy<nwK6^k*F@v?XqO@`ogt)ax2W1h}?|qoaAi1HN}clx-B_BrZ~bFk(iNFo{^hY
z7@cJ(lA57-fPbZ7ML7it`B|08MJa{xwnT@)vhi@CCCZTefu%s2Wz5P}<rJh{D#_NH
zGPhf-=A7dAY$L+@xkd!f_4YJdrg?3h$&_O^+0(7k==i*y$rfOsEg3oI?%w$F{z+-J
zEjKGyU1rRSFHA98GjcPnmfQlf-R#INH0NfdLw`0c*JR3szJuNonUtI%P0dI%q-GkB
z@6X6)=j^6_S4n+QV?N^RX^ymXYnINOA<d7k$tyM;N>6$AWqWc~JfQB5#5|kBoKt4C
zLEIt9o(T-WI!k%AJ-0R^*MN2g9M|Wk7wF@Y?WV>QL!#7Xu{v_q4wE@D$50ejb1cUg
zW8V#AlRYy(JdqtZV~;*RIXfZ>Qpa)SiShVk+HQSHat1K=2?^2Jv1Yp|LTAii+5*N@
zW3pLqNG`QHwxpRVEu~o%Y2Fr!43)Ura%|<9He*40cA`a}6JHosnrksvK?)Sxytqf7
zYELQ4&CAU%w^)myV;YoMs>&<0m<qVO_=*BMmuEYY>_~T{??CX!>wb7{u-r6zd;(%Q
zb;&X5_$@|Tjy)&G?l725`BgR(epg~ndQM7yW=@LK4so*Tbi1)U-xM#+$uV29RoMx)
zxKcB;Aft_$TzX2pImM7^3Xim8CKg9##o}rMjWaDZBNaa{Gs6&LFy)!8`MIpaxQXe=
z$DNfXt0^yAWhyDnHx=V%Vq~n+;(~(wf_zJLW|5&Lt2U!1JH6D51T;>z)sAG49XyXb
zTV-`YLS9l>Vxc}KH=`gox1=mTs>D!gu%#F3Gjb~I=4@$sPOiQ%xhT0R%@~zuv}Hmi
zJ|iCyu-E$2ZqukHoZ0wEe&V3cm44zt&~92LX`DX7>q`3KiI>_Ikr&(FXn(_pW$+&%
zPp8p1$2rG|oZW<J%8I1os3;mw`$TW%=z>2*U~mEk`G&}0v*+il3ep|PcCLBWz^X~=
zbeR{?1gV0#WITwLQ!n%R4F%1OK-O4fojrUR7aT~IEJWV$u>)yb7AEy171>LcO(cr;
zR%N)%>FC<=2O$xv&}nW!#3s(K>sKAJ8E{a=Oe!PUo$TX|m6S8NaajjR#~CXTl7-~I
zr8AHgvNAm`rpg7Em>HJ}Kde{7a4Z1_cPiRJs1AU-Cp4{F8vxyH4{+<CZPo68r$-%?
zGBMzRZJ_PYHpqTXlDS(mG;q+`7=y>Hu*oC<7W#?0xT2I0<9ZouT}fIhTo|C$-CFTB
zU0irFpRBWPg-e02eSp})1OGvj+tbBr-x`k+NQeFdNE9_7QP{mC3Ol4p*_On!7xu*K
ziyHE(jJ@z-&3L{+!%TgGMFyda%v3IM9OOSc^v;;7m92wuD|`>1YSFcj?|)ELnX4>S
zT>Pq)sVk_u*R4o3m0M`-Xxio8vR`?k5`X;ly+eOkq^>jVFFaAw3Pcp0r_1qpp74QC
z()zPM3GfJM1^mf$v>rq7y?r8L=59q0g4Z-cdBZ|#0iBENHG-<uD^}(fui`Bc=7<Q@
z;E*AUCUAdb-h21sF;X1v*Lz2RdjV_W(L^ZQz!hCU7x;-+Jr2N&;ojg!_Yz>VwcZcs
z)1hR(d{QTQN+&;26TEgZUL%T)2}=o6gGo>ZtkxQ`mMOm0)~a?DR99ATn;UnmJFb31
zCV!#R@pU^kH*%E~)%iQ2Xqy~U#*=k)ov17(FMOM-eZF&nGB`;W8O1ej-nxIWnt82@
z_it_7%tuD)l0!P$$Fb=;vhKD9NzT6;Swq*dMxdJOlD98Vei`za_B6+~5}jHwao2eD
z*oi^&wfwL<qBEx(u3cAs5z@Ng6zA!ve4zgHf=Ro^spWq9@smOpEZ~`xr)a^#EfdG9
z#_fFbw3hR9alxt2gTGK;|GeSMY5vYI(R^*tgt01lRj2VgC!JcrLELk#NPDU^_)9e!
z{GxT|sZ%^~hkZn+PS$_^1q^5}2t*GMh@Jtgrc*iMG~*6%bq9(+I1b_Z8FO{(R&r$a
zI4CZ>NH=?g>*KQ_%`$LuPx>02)`435k8r&|i!pVE%qzRGfK4EGl<q9&ZjsQdMe;Rp
z@okzGcQ3I!t9`eKJ~QEDf6`N(s1cK%Cx0f-OyJuqoPo-UdQp=^=tUInUXl%xiY=tp
zJyF!YT<lT23eZO)aT0)pN)|#itt6m(HW`Kh!LaNW;0@$kCxAk(fDJxW^R<SYoS_tQ
zgqO++9aY9k-^{AS^5{IOXz{V8B%71fdVQuLg>Rqgevv-)QHB|hY+pxxPGe?c%I{Mj
z(5J3QPmSoe>s9rT@u7?6^Ya#kjJLnx=zXOx={!Zc;MRlSd+IaC^D7SWHdaw0ophVz
zBTwx_yG=?-PfJTr@vT_7IDfwS)xN<CMgVy+WoqA)-f3^_9DjD9bn%%lLcdqbXLq{y
z%dS(mc=FVzSAKE)ZfVSDOsUVw&N1hh$x|QsIMFeY!NIhb^RRY6WLVKso>y3IsRFGx
zr7EUS>PMG5`zXV=tw~y;me+KeHKk(zES`4yWc_a!&q!<PnhOrAFI<gUcabNZrtRx9
zYrb?udxCP^tjSTc)$-?0u2@r0vyGp1X5Vk5(<jpX7eB4Is6O1B6kf+)oWH&g?KDkF
zrV36aG&0QQY>UM=*KW(r&8@5RxxPFhRTPz!2)P|SfE{$Sk_HUeR+pNao|~HMn`t&?
z8!aihJ_w?Th=_3j;U3Ls*ST9oLYo`J$m`^5D-?k&Ilg2H;e=B6Kuk>3u?F)oPAi*|
zVID(ErQ?m~wfsSopSUtn16rkc-I7?{I-cBsr#c7IZ-98=#4Q^(@a}<VXv2mr6Bnx5
zAxwZhgl5`&o`p8CRWl7#${$>TX#EKZz2_XS^t=*Mfh+Lt0|b$SfxsYJDFlGY6(B(i
zPQ~LkCDS_qEKE)Yd%u#fHRyRFclCf&h=n}gIS0KqVHGPNa$NE8WPtL{hFkAk;*huf
zN_1e|g6jEd`qc2@^eJt%_P{z`7~~!V8Y`5v)Rkw?R^mC`#=8dzgGBKq$(2>A{X2K;
ztEx(gFG1+i{S_n>Y8Po$Bi?yu#Dayj`_^;qrOq<ZvLjnR{6qy+@uv{8d+C!8gO?xS
zgx}qNc-nQSE6<$yyWNKfJAO8)UvqCYuuEgW0NK>%y?$5U<eg>hrJ|XaZmqwg2KDe6
zJO=YXLO{X>CqO`|kw5{0-Nfv{)E@*mw~#YIS{Z{hN!E^K&mBM&?0$D+yaf*+TvD+=
zE}@7gyXkIGVPff;Xw_qd#O-h)a7wk_xGBPjPh*u0Qg+BhG?K;+nFvhnBE~_3{3hd=
zx!U|SSq|Af$eSY`s#R*SSJ#d|z*#$FEl~~VFN-yIMFk=B254^bHbmEpWULknV70Ec
zUH{7$PHosfw__I{>5OU7(eD?cc(9W=%JEk5pnJoka`Mb3K(L=C@|WA>)Ahm&Bb8TH
zo_MQ-`-w<IcTmM-;3J9rl>bSIyvo0!(cGXmNmi}fym;e^y7@lMmX^%<s<j(pH?3b&
zQ5(w58z#c0?;dK2ysZA?_9ck6nVk~lOJey`{<EoEV)=>$H<G@p2V+hT8b58_)Xm(j
zIiek<J9q6-H&(_3@*`>FRytD^W5I(XkHvnXWE#+fK)l}dg;M^M9u|=N`R9ecJtfHd
z%CC+uFRduf$5fFd9&H*uTIDa6D<<?McO*c)H#cEH4wGSX1hdHtvMQNJ9txu-m+GO%
z0TRX|p&l1B^g<iJ$Ge3Y)z3L$1uHV+$$k6LE-gxDoH(>BsB~lLv|aP6mKD*Lng_kV
z@{n}pp@_prRp+XX9@@|CKXkF;3-#AmgJ+%RcW>M?ZFip{qtCbL1s0K|#0>Do`-Y1t
z*SWM4X$R8kCf3X;S(z&>n5ea{SJR2~#nmH*@<T_w+Z)<IvrFD-n9dc4l!lgvONAW3
zcdl$b99|VIJ+*ZEqWP*7ajUkh<$~9)30WK~jajlhWwm<wyY}L>{F<Uo<smg8HKFz4
z`}Jk9$2Z*SZS+|wiOSeyUc%)JFfK|}B}();X*n5OUTSs9em(JD>l69;N5<3YZ$7pc
zo#amz9;-eE!QZ{xYpNR?t9KVSNq1Z+y!x4{(O3`UIWh;C6bxe5v3o;)9Db)eN*f$<
zMv|_h{*;^L3y%1SdMa-kk0zApr1^2S$+WwQ-j=*<9h<M^ipT7jeqhOgB^TCQ-y+>|
z{ik^Hl=|me`BklaYt@BaN1Kl9+t*xouyj{ZbKY@09va91soatvbW1JEQkiOv6@{vD
zTcN|jS*_cxAJ}(h??43)DLjZghst3r&8X#K%<rU4PoHM-rK|HqBgMwyMVY^vcV^cm
zRq1?-;_bnfRXKBf<aNa~ve1BayjR-zgr!{arr34c)+R+oM4}pT_~y7xdTI2W87b4$
zbLSK{%;wir1=Z{JZ#fnBamI}V(&%F%F`{Bqd5!9;oP&-$uC%Z&_kjBSW{d3r?~oW<
zvWrBf_z=@-^(xa+;{v4H^Nia0rg`eP`DW{@yk*_`>`m%~#4J-HZ^6B>pdhn2tIQ<z
z6J7ChC4pK8PN-OJ*2!9yVp2G{lp$HLMlektq_lC<lYrSsHg#PO`~|{pO!bmsCKz6k
z2Q!crL1HSYo+>s#UZW_8VjT<+r(+%4s}GyoysBgnvww{23nm_@wD$26ukXAae*n|i
z?wYOi|C6!2{`41-K|P@3o>aimrDQ3BNO3ksw`BPyKbH&tBMg;}P!-bj1xXxPN|!Rr
zKOIy`8*Fwz5$;zph?F*PE&W`F$-Lt-fbM;iv&rJwOo)~}U!aRGki}&21(7q%J>s~m
zJ<>V!xQ7m`0X(hy_Z@SyoWQ!eF9Y(@q1+|Ou@ze^99cvbi7b|4TaKCx70Z7G3?1sS
zj{BI*8IJfdD7_vg_r_&WVPOc)BH6!Gq}Aq)ovea(@x-t4j`1yGZ>~k*eLnV8^5-5j
zL5p(;83RNq1O1p`FZLr=#9ZePYZqiMKS5-xn$*x|IOD184~x!8vx+Z$O9U?LXjUtr
zJmQaT-TZX-!gr>;`;x9dH!AwV+h40mpI^vqvJHs?F{nywXaW+uljy>?Dwfx8;EQ6-
z>4vC`gw(){L_-wFt9GgX!6m>=G0Y}7EX6`65YZOUK#+n?)3G#yX1)H#q2t@Qcj=Ur
zz${hVoXvAWR!Ad1{Y?Lb+7sLR(%FxUB0V5!&=-$v>^;jvyJR^~;5KH6(@&@TS#_6n
z{2S87g&)oO3?1+K;kP%gG%lJsb!9Kz0B$roeqBvo{ux02tz-;bk>?>z9Sgr|Jk`Ec
zv0@iG9%oL2v<o_(nP`aibi5}z4hdKl*+Qm+InhquC-dnBbon?G7s6@gsBs66hjJ_F
zo2=uJqZW+byj<!Gd|BhtLVKbcJY#JOEa@zAO&CR9MN-w##W{MO60!6bzv_LJGsnhd
z#i@Z}nz4n-r}6Z;y+rgn>8=)@7u%~X44i$K{Gr_Ze(<p~s`~u8yq#R8F5IXCv@M|V
z)<oVtMM9pvL|!Z>D!^kV3b{%$a5Pj}W>TLSREi+|z+V9Zm`XGsJRsdT*M=Y9`QpK>
zGvpy0%tpYX>9{W*C<9C$!EYJTYomDNxjK=7O=OH(cw0=>GoV^1E(|Wrsf?ChnbAl)
z4+a-1JOaH|k`s$*qe`2&aNAOFFaeOEj=Mtj1rmFKATL9vT!#%fb36t-f-K!nW=@Bx
zQv&<GeU+=zRe+u8=-E)#4WL^-hdHs1WS$_emzHE1OSy-W$%}7Le-(Y!`MZ*KTizE0
z)v2!#fJ2F_1?S^MGV><Qp#wSkB6v9Oypxs)lX?1DI9DU+JQakY#?oG7VRg0HQp3sV
zs^l$65t~)>>z6dH;^;I3tzR*ez9o%Z9k*h+ipG=bF}Rldk|7Nbh=fDuZhe0GM;K&{
z^yG2ahCW1BLCSD7Eg{eKy@c;8k<cdU>muO+mM}JcOz5qBRmaeR5iX}l?y=!TCcPi#
zIi#V5W<0gYuAXIISed#89JTv+(`=N)g~jW`BgcL1gFa|PMC{fA+|E#52%k)c$U!2m
zw+&D<NTk#0bb@D~v^KzGOeE6*j6)xwmIY}nF*LG(lbfOeo)vdL7?-f}oQ!|q>;x?U
z3M~MeY_bNN{Z^s%E+8oLG)%j|!QNmFoh5tx7Yp2UZV>=zRJdB9M(NhNwU`mpFe4%u
z!z4_Bg6r5U3!4e8uqh6(a!{}j!N>&035-k#uX*r&_~nSmyr2O}DWFG^#?|Ho?NSd{
z0-ERUHt3-%9=G9Vf>FT4$1#7yj_H`d+mkSlN8Lq>^Vl>$3rYhsSU=f&blUr+lXV(a
zj!x5nU*`N+8N3-KSHoZ)i!iB(L0*(eXO8SOo_6-=pwrI1zPL1!rz6QTbSyIFqlsuk
zZQ#z}Mrr#V1cqF#UGGf#EC9&%31a_+Bl`{hjf$==<52;w6B&YkkbacD`yqMiwHqEi
z_8a7>yN5o+*Dx}N;C2~II!W(b{N^{7&~lC-g>(#gxqCVJ#`%EUl!uasu3k#|&Es(L
zjkwZJ^ny~}^s{No=Tw9{dE&(W1Fw!pki?uNCX&y-_{qfkb+xnyE6G_%2)#suIe93Z
z`bOVrt9W^n8R4dz;;fuO8IOB#S>&d0OtQ<eGq;L6Uvui>571FM0^$+x-cD{xy8WPm
zRS&UL`4zC81!$v!96bh^{rO{oD(uMtSEIZLm<o$KF?G~`|45!cMqX4yNb!lxwMXuH
z7=7h4BJal=?4X%tOPOAD8dFErP)@HmMdu~@%A0_X>_fKnAu;N|6|cbuV6n+Foe$s-
z;41f_<_8AcUtkw89`yPxaiO6+yL-T%?2aNm)`CJ+p`jqf!3FQC+Im=BSDjZ@&hOoQ
zWbY}JS6kdYP#B0f3@R6?7i?U%F_4dmPDW9r6+0q!1#^xRD7mN;lME>+J@^~_O_YL6
zN}?*!n&e2~b_GZ5SfSpggYX`|F>u+&1s&y&1m<jdD<GZPdt~o!zN|bCb;}NKtqR`7
zr8o)<h3dirs{>9u`p9CDp`meG)~ldk&6wMNxjX$$d;XJj0_!;fat`|IxL^gvNVqzJ
zcBD+0;Eqs!`0nmek)uO<zclmZ?Y+0pY}@{O)r~8suiJhTy8AeG@2T9K0&wK+l!{HM
zfl}cNeK=Ca#3iw_mZAx8le9`9#ob5f<4(b&4<0FW!cjWrE`6DcDX7p_slh4~S5@Sf
z74UaS@1tZinb5&uMU4TM0uHwTIvysFdOYy#4p_C)VqL}0v#nY`>dn{Y^;zv(cewU+
z`PJ?BeFBb&=)_-M0UWBIiqs=YlPCmm%nVWf%}nF6Bp!0we)=cKY5W~cgtaWL0(?%h
zdKXh=V#^BbGub^%b6Ol5OF=2B^dJ<6bz?I9aM5C`V+p@7Z{?P#gvi9mB;P&X_CF({
ziq9uLB2THX4wM45@*!fsT><syV>N#R|9R(SKe|=<1o1x`l_~zBj(jNlyX0M5Pea%q
zSAi{2osnTOW$<qQeq_9`R%C&al1)_|RUXD=(HoM;n8avEMXYH5C^I1^z988*Ew>;e
zA38W$(7_S<|3;UzA2mc4MpmWynygk+j=HQQuQ-<%n*6$^+lw<x^##6Qy0o(J80KjK
zoDoGR$WbWN#PC`_!i*)_`>*4y!Mmodsj~Z2%hU~7(MqZv0H7{yh2A3EY|j?h2UECq
zK<fYPBQ~2t8G!_J_WJZt*>)~g+9M-#BGeI)8EKKc`%B4Nvu3^Z)~t&kkHb_<dE}O#
z26PC|c<>ySnqx|fM@3xdHpDF=o83~iTjuUeH@myN#+!^;#!S^Fjl+(_1b6D(seRw5
zf4WH|vO;wcQORzc|4IGR4ZJN<7vk+ry#40X`UU6sbh{lix<Lr2JA;(0&QL!|Wp-6|
zWo4DA^4XUmf8*h2TX8qA>%n6KIbiTRv05rYxKMba4FSlTw?mw!(f}m(7FkOITv{(|
zZ3g5(+5=!<NT8Z~LPP?wn>W<JrU89^k|QzK7ttfsI%q|_0YM^2QDSLuHMbz5=&g^|
z3;3F7A|LJfME--^<<v2hT^IV|(Lvw@Zql6mGj!@E&FfGX3+ORp+gq!ba?ueHhz3s{
zG5eKdDw5J@aWWCj{&n)LsOpHy=ql;zquV}iQGtuTk~~Y^1^@&6l9lvXx*TD>9*Bq+
z04Z+6qX5@=?aRA|UK!8HU025c;GgR+4T+5j+N=t9=t^R_xY!h3xN380@QxTRHNg-Y
zr;`6L{rHx1+}yfz>o2P>pWAn?jz4$2{zD{$Qj7Q<Wu?0sRW&Kqaiv^IsiV3^)fm$d
zQp)B2`NN-AzK~w|V)u8y<A%pDclWG?px$~UqWoNyC=?c7M7lzZ4f1QLOn&V%sF<PP
zFiu`s5z09dEg(72?{w-wXT=T@RG}5Es?~Tnt{(eE=T-HL0d7#4l98I8#?e10VpQoV
znW^fORErHXCZOOEw915<W+|gp169_B;EhL4fz`jl_c}<afB>Xh0NOs(lKyVf8K8_!
zh=<Ro1HB~L8(wOoItg<C9T3(>4S+w$AE+<su;ikcZzh6ln916p4`%^c5e+skfJhQ>
z*<iMKuQUhM;m*W0ApZl`o)*_8iz726$0L_xmXa)6OFpJ4C%B!{(%zV)y8iCb5rb8i
z01Nb%X22_{rbUAAam03*j&G0bwccOt-)rATBl(EX%*~rsbfx5r&=ZqJtK@%pxokI%
zw|)uPzLk<@TXP8nQ&SeL1)3Q!l97K7AORdJM-uSoeFDqFTaAPj9bj^-tsdzt;<(Is
z1^8cntdsn)p2;sjh4s<!+dbcxjXXK!S}5n{e}d~IA;H_>!Xa;>f|WN;lWs7X4BY;R
z)!Ub;Jw=|YtL*vZyt~g&GNF$|UtX0~t@a`Xm#q$67r~?XYy<S#)v(rJ8qx)DfdWV{
z6bXGKbQAEM6Chg94^|#8K{u{#Kl$mmxTIKI5Z+5O(IJeeBh9c3*^I7WIkxt;9y1<j
zTEre`Gn|Tlq|spLb#gf%n(S)+O*R8>TEJEHKdNz_1?2GmfhJ^ib)KLJIiLyuCzkL(
zNJ1tz%g!(R$I_4<46OoeLv98Vp<>1+<bix`cPL>C<7d33X+eB}u=<s(Y$Ytwftq$5
z>hC$Vq&FDtl4!uQ5EAy})F6=!V^wt0GqI6g8gRupETL01|9su9kc>Vt>5EXVy`rPy
zlCwhc#r6}eH&jf|89ZbMQX=52G-E#<7J;4Y672$jH&vWR-#sN2Tn++KO1pN2hA~ng
z!2X)%?>CPX?q((GEuc^A($1B2wlHl)qWfF9-O=K$1n#XnJ;Pg6dIn>smvW3TkGmVY
zwhqIj3lqXqdiwvm(f`lauV9u$W2kQR6=J%Hm?%2Iy8y_T(VLlj;e>k;1NVaU_Pp$S
zhET$!PZU3Sfq!Jde|H=NY3bxaAlkP#f93HOf)IPwzAlrei5iH5xe0E@%JC5T?*qFC
zuriYZ0ARO63Sa>IsRWr^2KV}DnLJ~P;Ap^rLvKJV53NV009CDMGom8!j5>LH1^_kO
z5zicfD2!JXf-Oy$jO5NrL}Nz&9gWGh0o!V2(HI~3pC_$3`8l?1DH)2>$?PClWC~}1
zQT7ocuJE3kmDn2^X6$;RtstXsTIz|;{CUz7o(T(!TDnPv%VuZD9xM`K+7q-Q1pDz2
z+fbI>6R7dNCMYxjwF;-hyI^7j9q=4$Fg*m^XMM!nAmF(2KlLBU@UDuzf}yDExE=A)
zV?~dk2bu;kMh=;9+}{7VB?H(k*(xDz?3N6|n+6YkJgWhdr6b7mKhZXHX9CXhM*IO-
zGApZrHn(uJt%2%VL^B{tgjxOynWh;4(!F>_Pz$m)@*8+bwL~WxAPx$GJZ3`>QKU+!
zHe7TNHgLEol`4XQs$>m8B6;I|F%G5^L2Wt!dt+V{-$!dxnFLdt2=8?*q^&^&p^2=9
zEDuN?7fp8!D=&bsi2}Z6{Kl+t>dDZX<RN~DlL-a_@aishd1C_7ik!`o7IZ)>LO3Ic
zDnxD_dul-hqm@l^s8~xjaruv+h7On|idw)tm2~rvD6~qbxwX0-*zj$cO96ZsZAEYr
z?=<jj5r7Nu*RTyC{nm>3B-<F9Li???NyE0qmX9<-vp@W6)XwaD$hEYKQv!Xnc+^?+
zmZV=I!<st<_kEB5&u6e(*iMDmNw|xFUj+N8$08>APkOqRl4mh}C`aJ4t|L63P4s+*
zm2)^+>pEQ4?eSlpV+z-COqWiHy7yCL|2#;?28Gzb)BgXhAUW1_R-~Mj@=528E!n^X
z`AC&;o%Ns%Jz#H7dEPpkad21%I!%XWs!b*|16I%I1<qCpn|o0AegcX{I9pLbi$!LI
zW^urbX&WGUz$6Lx%=qk<N@y-)*1&yO!82Im7l4lL?Ba)><hz(|9ZFYg0_H`K8Xkk^
z)(*gYkh4t*yTaA#wGzRxi^g|bU?Q)gI>v6ml{rAX@UvBS*x^CMLvgM968Z7RT?Z(?
z)39>CJbpwLj@8206k{}9aN|$H&=Taf+R>0p3meqiIx2W0Afi>?dGoVjsQu%OFFRYy
zG>?<Aj*x&OP)KcHtK)s4^?nxtWMk#ZjT=|4+_?AP!M*I?Mx@-=v3A`K3<}y698Y?U
z>a5>+st<GDF+^;Mk~t1uFX6ERs|hZ>E`N)wIf1@FWfstEn5Zk}Fx(6dp*0Yfsh|k-
z*3LrWi_LEAn<7~td_Jc(5K4?ID`m^DY^UM2t3{ICi7`c&bhuvw0J@OJ3iw9(_4Jmp
zV`j`4Gp1$6*PJ}_`iCuF^TK4R^?;@Sma~`)<gU97w;&shJPoU2&h!L0S1s<nOG}Q;
zoVjGlOd`b?g6{D1*z5iKj~&~;|MjsybMl@)U%&n{j{#zYlIC{U@g1u?$_&c8zagXR
z1TEMZ3~(dvgd+eV>eUbP6ZiKhhzalmy6TB!HCQ^34Ra4XM{ht}1@Se6s2py`KSES^
zm&9_PItlXCdtY~NTVq_4xrR5zWyHj(q6^|GitP40J6Bu@`Rr;bqH&+1W`sZH8mjmS
zc8(7ARd;}eP@o2**{b{!gWBUu$m92*=V{||n#s|zVhGeVegGQvt3M)8I`X5Iq?8Z&
z)DtH%PpVIzu;iZL9UomT_z2(ph+rxz!RW|jCF!%4@B@g5D?8;ldscNV_FCX4939-}
ztwHn|zH0EmyjRt|dg;Ua@b~DmeXh`<>cDBS6DFwUIp&sWxdF86T7a(msA!jb`poe@
z9D?;4L8&99YEnr4s)HJ^4}a`oK9NBf&r1}Bc?t6Zw-f3WV(wrj6|^Fu1%cbarTq%`
z6za~cTFB%6!D6QU-*iPVzv3dqCB^31Ht*7D^bn682@jR=DTyh14pMM`iB<<r*}PEw
zxK$jn#lEB1QRo1;=>x=hns<NNcv6i!wW>aCE0*CbGEzC%fAM6<k~rumBImuYGy*Gs
zhi+ii6pVDFvc^!IT*<WrPpgMk3a(M{XELTs(jj<)roJbgPkefo2kiDQ(e~Zt?;V7R
ziU$m8ExN0<KuRD)Sgp+89ld9BeqjE_Xnmm6e{<hAX;%c=`hNE3W?sL0WBF!%pgww2
zexQ^NKL*}WC-R0GdWo+fiZ}ci?@ke>_0vSa8o>|uwn#20$?zrMD|Mo80PKz^b0<1{
z39k<<-?UrbsNY+jzgzle<Acw@g&<uYyy@S_=LhbM-ksl=U%p4bTZ$HMvlf{HZ$YUY
zK|4*>u4u!Z3>9yOpzY`Jh_o|Evk*YESoYzOoy3BF$k~ccye6aCT8%s!73dX^rqou+
zbTauNqF9RG{60J^#ZnE1N(=AmAhP!}V4XNHamu4Tvdl3WPJZa>*?E(B7Ny3gf2%;_
z<Hri|dm6eIn$R(1FBFPoN3Va+L)C8V_e8do^ibnN2u72=Odtan!+$>>!GOYtUh9s1
zC4bxi?2*vbtO;NiUz=G&b*QY3`F4PWA#30gqPRASY-63qmjN0q+5u*byl1CQ?QQ?H
zp|j1qVSC4h-W?8W<Y5y!1!$hU8*J@#7|mb+J<Y*ZLTjPDv5EKP;Ac7w@U~0SXjjSq
z))hb=UdEcCcVYHNr#)&u9Nsf1|6`BPLV%~LU?r&`3h*r<K&Z&YkK?gCK@%=Xxp^ji
zb`I^LUKf~)f<#-3L?``UIZg+u<=-H}I(E@4>cb27p`Zfe@iI|@v_zzf7yijdyni(L
zBmt7pEkWGdxl1X3*IWLGlP4~(TeB~MRY3C86q0|#Y9Jkf`zMpX`?E~`O*HCbMX=gN
z^2Cod1*}3A>5Sf7#8;L1MO8H{3gGGN3#SW(!9-z40t4OMi%Y3dNuN)qFR!4|1yV8-
zg|E+&SB{cy`O+$xFrq7c-aubkL}jz2WUhofb&>QvPrBQr6!lD7-D{ux(!gL_ekf1o
zND^}rt%)}2SqQN`e~J!BPX}X`gh|Y$CD|ovGT`2VxkSPjrWYCtGo*0miE0fQ_VEvg
zr1Tw$Fuv>H#dO#>s@f+dizVr`b;j)&4S9DumyHK`>{)n1W&b@CY#`**kI3Z77>u7~
zPX?l6806F0K)iQR)-eoBo*FWc;_xm4g5;4JSBrbaRM}(rSuXI<aL3W7{6xz;AM%s*
z=_0ff4U|!tq2|MEH7KJlIR-IQ_XhU{D4+^>g6!$BV<O>>>x9x;np_rZomuJ=XN^fV
z#JZpMb3O7wEti;5!=+fC5<^*@wN!Z8PxOqBvv)fm=>cNE7GbN4pJ+N3G~keyD&0MW
zp7m(Er|^>KiV3qq1AwM6WCJLcuW_I$LlmHu?kty*Vv~mCK+-jqaEosZ{Ec?qP2UQk
zb*6YnLa{*#$?PnPx**?{Z{_WU$V8kc>r|-M>esbe_(HjKdBNKkfG@pD#?Gl1xfV$v
z{e5lM?2nR(ut-D}6(|qBpYYyn2P(SycuKl%PlzpwQD;eFViH0Vc^ctf<~B{5oszKn
z{Z+m~C;I1bccy4%TFJJ0b$(G!ZZR(`AbNq7e@!h0y+K`HQg<+oA1-8)zsR4We_(uL
z{JPdC3u_I#qROR(o}7DfvJt2~cp>eIZHWoN_7L9?du`M%Cd<_-4z38>nZ~i`t5sc7
zRalkJI{{E)+Uc))%^%?urZ`x#cSY{Il6J)*&ufWrsyzTj7j@3NVv<En1%+7^Nrk?N
z1yNavNszP5@@c<|NoWkVzy%5EEM0pW87y<Q!IF591vhk%!EA?be!oa|zu|uOtoxXW
zv*3B%M?hD8A6|)P9X%l9Q1_LssCb3=lft{Y%E<v4f8P_g;pF$<mrdS3^Xz8cNZ0pV
zNoUPj8owZRahOz2gmlT-#{HMBs^s^*As)|mT>C}9;O1>!H*>P8=k4Jhd8DiBF3oG?
z>Lfp(s3F6Sp;j+`^Vb&AF7@v3!P08yL<#{d0({`_uyDYlBj5e~P9CQhW{@(wjJ&bt
zbIip;Glr&B45f{t1RyJ*10mPz{kr~!{(l+#*#h8Mza!tpmPQvw75K)0n7y6u=m5?F
zfxB_zjO>kjeQ6y&PK_yuDvU0T^~Dj$zv-P0VCt8jJwc_OKDFz!FIDb#=O(56*-l9n
ziRH1S^xx!;j~5C%?#(ASSnYz~H^-^Q?RxVRaIoLe?@D9K6DyKf%Vi{uZYSGsYijc9
z)O9r;EN>k?Ni7pOpBwo<e~P<VX#QLy3?OgVnKP8dY&b2`|8gpr9oI{S<e~TkqJUfr
zs`|bH>$)#iQ$JBB7NcRH3IJUllabj3ll>QA4#dbvbH`UY_ElfmF8I@XvbXNs#Oio%
z+8VMco8Qsy5N*od6#{j0hj`DfoqO<+(;)(yXp9g{x^IM#%YAT!{6zC{*8wFVKP#^-
z(#X%=0YK|ZWFR$?M49si=f9P-`xqK8E&_M`Rs~5@5#K(yXzvlTf;Qil?JnD=KKa3>
zMZEkhc~cf`PT(w|A|YSg4RM|BShL3<x!;GRlCe0<f{w$q$^{&z^^S}kLyTj{mpnPd
z%j^)TPY$sq@V2b6Y`^yqI6zExJeT7=fM?D325I3XDccds`VNwKPL#zS<$POtIC90#
zC2D-SVN=}JaIBV(9h=vuF05GLyQ1=N$T2mxYuZz?qXy-)U5>_mxhJCzLq)PQvMv&s
z_Zi)V2r@$+iZyh)vTg3qRKiiYw*OT1rY%)9IzFU6{os45oB1~jZ*b;3`*}-_)GU!V
zr6Z*)-bN+r$rE?n1l*Q%fh3BGbRK@bchCN)I)^rX)=pJzir5ma<3hHqOkb@YH7dVw
zG@opq1C3s(JQSXli6ug~LStEGIsW-3-ngm1sebREZD&1SQ(aZR=Su(6M6M!|pU<`Z
zetQn>%+YSNOAviZHR|)NSO55<BJX!DJ3k-s1FIza%b;stsXpIz>}!rZ)d2crH#O;e
z{`T+8!DN*`tavCwk>+ki6mhLal8y?H9$8q}Y=|U6ujME_u}sn&#O32M1P%zv0}ud^
zO6}>%-s1%@|Hy^m8IQ>vW>i?ZKESH}%G!RN)ChN!DSOlR?S}-1r^)ffZ*G5^`|UT8
z>w)k9OWLTLJ`WL~8-)LTT4Xmz`8?DRJF)wGy6WqYTPf0f7La6JNtaEWQr<9&gECsu
z?xwVT>c5YPkd*|Wmv)i+dE%oa-QK0L?<l8cVNC>)ot+_yjN)TOutht&S`mYFwIX~0
zERce}=s%Jh^UkQ{i$kTX9Jm(IQmDc?SiF!$UL6wmDB(6Ouhnx1ix?dMDCa)=a&5kF
zo0JQq;Km?-gxIK$CwwUU!}{z3%!)$ka_BTTosZ$|!a|+_!?<}VAZ8lc417V4wNF0r
z0LNA%hI$VT-S1AC?<1s!DPG<rIDjIU1W&m^2|dVE8<S--r>Tv`EK?@$)(#LQWa<;+
zRrIvjQDKELqu1{Z$_ptD<K#cm71Lj#3k}I}L51t<<D>>ho-q#+8EmaGXG7e5E7_#R
zH6f-w*1n2MsF$j}*;|SM5h_3lp2GUxXBYPniZAi`iA9;fRtyk5(PD*Mjl3z>mgC4{
zj;RjJh|Uf815|P)U>O}t4;HLuWm#NN46@zx$51o1aP#KQd3*L`_rIcil1<4-&oHS0
zpR^=%T%NvVhL5-84(x?&3r}|5V&L8pbZ4gCl9Zd`ix3%dLXd&80n&{cGzy|~*lc;(
zdA=3Gzph^R==`~}zL1AXxeLtKEf|?l8=gtNMzm1;HN8%*%WwIKKXv9PcMzWt;ydOS
z=`UmHzs`Uf;s+5f@+$qBa2m2-%>KS1-n%O)vXn22v<9VaqEp*jeaOGXz$m=#%z@1S
zc`78WEKug}Nr1c5xR(k`ed=Wbd-_)Mu(wZ(hF+i-d{8~|LW{;%s1ka5sH=bP=3MRB
z4LbDoOa$(N55*rCS`Qz7i>;Tsm$IEYAHqKGXuSIXB4|b<H_@(Vb9&*X#<%xFW}M-m
z|JjRFq2+(<h6*mp|LNevhv**b6naK3V6TV0aU@c;nDy>2L4OA`_1n-^_~3@d_1HCD
z**-#CjDibJAMp}*Go^h+rVI&v{A&cM7m+u`h2WbnUPzXltRm4Ow;*0Fz<iU6{iuO_
zfN-y$YmIP~ju5V)SQM7Aav<-WU4ZACPmXmi@Du;oh3o(@ZVo*Lt(ZgN^>n_-k4_WM
z?RY<p-gU>);qK97_)hYQh#nJ9rh;=8t#BSfD52a>G@P{u&mZ0=b4U9Mdc@~Y9T3SD
zJ?SgI=+a{81l6qdF|)VY#ED6%Ne14KWJz=+|N4s05J>7y97dOhN}XyrrUN{6542>Y
z_=|%lZvF&1N|bEiiBVsy<mgC={2J8`!~`V22oWT0yIM?_Zl@FP3?a|``m2v4&jLh<
zw7`&*o2bTd<gJ5nI7wn|Qh|Xp<>Vka&*Y7N{80pk@DQ?xK1VL8$t3_-o&#BJ2>&Ah
z`kss0TjWOmQ-L)XC=<-jm65pl|5>=!)r{m&yRJ!dLh~w84CA2Gh<ryY%G3%hk<YG6
zUvRDPrwc2!J`odVkG;%i{M(^^PxCwbj6Q;9FI0QkGdVyW;ql~|eU7;wg&qD99D?G8
z23Y)N<#>cc5rlj4)XmS82TfOjq4jZxk4LPgYsVjm*t^2Xd+3IPJ$FIO5AOaSuPU=s
zGE&lszoxL%#K%LGXcQSmR~JiTvlEHG%;v~(n8@W=RN*z1(#ui-YI@m7-KJrOBDRAt
z3}Wa%xQDSF60n2aZpkwVrLn>&_oz}gG)v!e&G(1$@M?6py+w)36$#{IeWo7V8;doW
zk19yQ{OD9jstYPB3b=~=T2x#{LcZ0fLSF!Si7qKJO3y0Yuk;h=(f7!E-A}P<NF8tF
zR+qzdMd`lczCy0hD*_%OcTM{MWi#m8<T1Mq9}ACbYTnDL3HN2xZ+2VNlq<b8VBjge
z;4@(Y?Ups++JtLaq}#P?Qh6nl3tw^!rD2x%$~QgcA9G7k9Ol?kUx<izzjlpu%epqf
z_u7Ok+HJxFpjW&599@b!Ge=##%ipIPriIruP>uamh7f=X>x0-E*QbBg;7l=8i{cg*
zbsds+tw`FzkVY6mp`3-62sbm`w^k4C?lQg~$q)%RTP!-;#bt4gQs!4>Y>z8PYC+)>
zzH>=dcnE}O6+Us%nW1?R&~~UwsKqVQu7HsVhHV-W>j6}onrs4$$yaYJNGm|0@=<VE
za9h<@6;<a$g)j-Lr1t;~uu+qeQuVkM_1OuzDNOCyC(Z6ark}L7D{3`i5%Cdv&=xJB
zCDBar*vBh6$h3}qO3HJ#>#L<z`iSB%#NM2Lh|3a+V^B~?Lqc4)NJtmwM9*|bUBrc{
z)jQp7q#J#Njl#w!(_J#Df+{WU<TJ=(m9h)9<7F42uP{NmXpxB3>yn%RprcsWuT0BL
zFrre|L3$9Cx{L{+@}?G<9S(Ak97Lrqb5W`tvX|{sm9!aoJ)v2^6Kcn`w0J(ad$+0S
zQdZLjUsn06X+ze`4S0Eo9P-HP?s3I>Fy@|ToJ~L%w#Dgm;9#OI7Aq2GD}<Ib%qn-i
zyZ*_1UoX_N&!Wo=JMwCLtMjT$>ePa6y~eFW21sytS`L845#YH6+aO=)N(P(OTc8Kk
z=PYS_cwQV3WDuXGvwH?loyAWY6;1o^qUq*@)PzKX)R<u~WvEk9a&pu7^jt6?7m*bz
zlJyagd{==&k<^t`*W_{4c{>bc(G2H+<xWbbk=y>L;({!^HyqpS2~Q(v4)cM<^+X6w
ztyLm-<q)~|OK`0rg+_Xzw#qyJ(qfzsl{l+~;p%pUI6NW|SAag!ysd~`0a~Bfg7V}_
zb!B;8DWbNmg0`sKT&J(}CF=IM21HA|`4E@Zleb{`5Qj_4@?dCI7^%#!G}a}si&^Va
zniQQEsg8_Jj>WK|;e=@8w){xni2SO=8nsg)_PX)V&MEkRHS20c_`fo_Jhp&y!+(n|
z+GdW_`$p&!Bf?d%AHxeHs`Ol?zRp};gte*Fr?eoiyix@fa2<@m$Ee}s(k_+ZpXRZa
zrR>mEcKb!c<ek!n^*4<~c78x|y13Sn7^N}Skeg~vg*rfLrqnb!JU|Wn^UJ@G!KfAV
zqPVy)zu3apkgjHjF)uyW*Padq0kwOj*tp-c+tMHv;>9H$n~2Sh%)E5FZ*F=@4mQ~&
zCjCApJ%1o$uYMAntu8f`=H-;WPloxJb4`v6y8%)Gsb*<*#_+0MYOvQFbQWzK%J+jR
zrFgLBW3h2l*81!q>DwUmP?5yL==n)ZKlm1??m6T`HF@^O2H@0+t&Wn65~*i)*-ST+
z5ENBdBq&K70!OHCIg~`o<6Tyv7nbJ{V);=ln{T^^O62j_?A$jp@?x2co+ClxhhKa`
zM8DmhX3FMl1{7q>c4RXY*zZK{lUHaePs*2C(*g1ZzDZ5(C{HnpM)Nd$Ao-VuzBpL(
zlUv@Ob+bQ2%;zAchS&)MPkch`56H4MV(a4C0Ps3Vr|WLecdl~urPH+A2ai-g+_?-~
zR)6xGKMtFlj=?kMW#`(gjvJ)U|LN;Hpqse1u4Qb^3>uphdx$MrBU<PXY6vwBCA2s+
zhi*CsgS+M4BzIeKkuAw`?*$At1_w+C1PCo5gir#clR!ua>B-BLeP!Oi$MD|wul29*
zUjj>-raLot&OP^>v-kEaD#-!udsYF0^8M)MI*!aoQ&p&JNCNbC5leS&N4@@7`i7Dg
z5bZ>=Xg+wP-Xe;PW0X`rc+DutK@1{FV~!}1M1t!vH#I9WeHb{OQd5lamXyK_OdbZ2
z<UJgn$AWmj0(--L&tu8%Hq4`^k}b*sy}j^lUqK1wFfs#$g#(>?2KJo7b$pf4osB-R
zx054D(-nV!IrJuOnb(s$L|z2((f2!jIy8=nGZZf(!}%&hokD28<#aw057I?)XP=f|
ztw449NV<SbrEhU*CUcfWMb<DjYY_r?CTmV<ni&y5U$El&(NU00j%h+(;<^3Ak3%>C
zmpBpSm5<5HyJVIVu(dj8`)>m)$|R`F*W~Eeia&9&j@~6lrz`$qD<tzx_cG*){dS?n
z08GD^B|~r0spu*OCEak?=qkr6v74i|#&7XT=twvc`3~J{pY3VQ&=sYqpg5z`Ny;CY
zdP7Q~o_y-K<%xdwoMPw%*rAtc=M2R#=@b1LAvEH{U=oOyX%Tf%kU%_}ffBoGg8ejD
zWqX=?HVafacA`{s3|qlKM~x+ef0{^@8loRaEc8vXh*paRM>{%JZ-0d2(7#6E=vv?r
zw7AM1eV_fLUz&;A<!8@BANHtJIP&moKQeGyaRdU|OsV)3@fyq)TNQRIc_`$7P(zuE
z%GBIEA9C80Oa>FNhd`s4y<jybT?{I_J%oe!e2r<7?C6IMAMBI!bf5eg<ULRFGIK!M
zC!PtE_BqMe-KgaDE@Y(r$cueYI?VJQke~C-Io3W~p@*%YY9=vRpOmc`PaSfH6Eli9
z<fU0DmP8eOo=L)*EX`=Fp_G{@t;tT7)~Ff1J`=-!28cBB_4nz|p47=-O@=TB&^kH~
zA9tW?!r_o7<%1VOh4U%vW9CN435A_I88Sm!SMYRQsZC<-Ca*~A0?~UMDVby$U4o3~
z_hdY#DN8^hHcd*nSAl17JoshDyD#;2DL?`@9_uDvqU+)O71Hsn`zo5lZr_%hyG=t3
z()Xpbkj>q*#}I^IG2IQ>TVMJLOXPW&Ju5$~-nG}Hp+^8}GUS>-Q*OvqIfk<_*(pI=
zREE49D$f&x=u)}+QnHab)Sla}qQ$Jc0Szc*a^LPW99Gc+`~togGsId-7JXDlvMR}%
zm%gLJ+c@{P?{&TZMKbZ?=w8R$0$oKvuN^9q2kc+ubFiOk=G(&r;0_zAr-<Yt`yFX$
zqKgrcNqGas@bCk=&B3?~@KE|93dp9zdGfdm7ix;op>XK{oo}!jAQr;d4`CK>{uiu3
zKhi;-Iiu)toKQcm7^+5b+*gY3JK(yWrpQUvB<0BSSgZB6f+VtCiu*l}AE^Nb@wpA0
z8~vZ%agF<qct-Kp33b}jgY-<KxsTMh)swtk9^zoRI9RqkpS4<U147d4LlT)T#elPL
zH6IOCf2HW@q#lybIbVT_ihk-|4}9>z2Z!H$DOcG~P0f%rLD_)%EReH%(L?*bPgh`Y
zyeS=^dx{+gc(S?l6m|RIaD7Ml@3)(M2Y1Gy2xdT1n*(F+D@f#B*ss1rq<*qR5!}7C
z2&DyB+cN~4-G?*q&0R!w^nF|Gps7XbectlMEmC2Egg=ItghTlWyFx;D?+R^hZ)<L8
zuSMc=%5_9*5DD_Y6WciwN5Py^_-pdLp(5SK6t`L|)>^LVy_WM|DeoA_LaHrMh+DR%
z`0AFYtk5mnu_GubaLX?L%`3)GJ|LUhlN}nmN7*Z|yZ412%oW>mFGhbD#RVXxtJ+A0
zsw$Y<hy)o5RQHTm8jqgp=&HDsajT)e>VV~t^@!n!4h+a;@8q21O0)LqTE&BhYtEgP
zLQpgNYLB3717AXD4{1jGLwD_N4rxa<DIP<!0CJJeR^%jDX+h^soDAF3OGXLS)`ESd
z%Dj__?*t#Q9>NbC(I1LE5K(Ws6@O`G*OpU@8z&pNtRzF6>QyG5p+l)^V*r(D-iTTj
zy*rl+%nc5O>ZZW%X$}RU=ArCIls~qj-T&a0{XvI!SeKQour4q0J-U^PgpI_tx${-<
z`SABNx>~&@t(7DDn7_We_m@#~I{JKI2ZDyEIV6KF5$^2Wi>Iy;kB{vcKVeoMLZ*EB
z{gq7*N<L1xT=;S3*^(=|3k6rMeEby&ac{qQsAmjG687j5Vs*Mgs}9*)baHE<RaIq)
z-=m=uS@*0Vg@E!65Du~jtrfMk*7!g%knMerR0%Y3!Do({pW0`B!4Usd!XkQriIz;E
zDmAmvlB<C^HkT<$-C}x_4bO`&(2238veX)oZItb6)NKEl1vJqtwB^)imWj=2aiw9Z
zNNrMhw1!Mw<j{ZI&W6?2CDo~#DodM-HHBq)4F^ChRv|28!gCX|l<`^m)C84QS8-7D
z5%Ids9!_sgsuCOG!fbm~fnmuJv6`gVjL;oQDzH0>LQ3Prh^nUKHr2sqTT`W`7%WzK
zWt_3dSX!%etm*z#IH;?Pj?%{kqE>?qw8YoeSSt>S_I-{sNTq+e<z>T!m}z42iVa&<
zrgMoB9>ze`FyeSGqiW5{q76rr&vP-~7#`e(l;yX^2UTB-whJeYo;Pu2kcR_)M-4_v
zyeATG&AE&dTS}L6Rj(K(OvTo{S=}0e`oBi}+4T0r_ad()9*;ksc%1u;IZfA`0#5W6
zLpC_vgdOR@K+HzOh9~0$!)*<5nxv}q76gO`vWJUWN^$O$jkbfT1C7ZMRhrV+q7a<>
zKo(-3uEG&EI4mMDLKU58u1wctmE=@l;&S|B+Q7Q^<75ejH26_EBOF7Ot<+LerXlSg
zI~dl!h@8Vj$PA3@s~2t&=GLu;<h8HAB7c0D5%YW#3Hpt=&j0P==kec{f0p*Z^i9Hp
z(rcLy^6xz$&hEW8TY5H=F5y^mU3g+*dSaqT&#;MQW$9(AMq6!{hCCnvLRVs4aB!k_
zlj#3F`*=O65FA%Vu<rIx*V?ZPc}PAQhqgPe2xr)By8>hOszRbm8qzeGW!ZIYO1tX5
zL&ioMbjEBkDX$2V<;tqk=4y?7z<oz5OMWCH1^=8q>CxgYT}13|)!v}WL&2I2le)*;
zXWg06G8)Xbx9qPxplWM~4X|p8V)FL*E0O;u4=h56AtonP%!x^h(<N}bBLm_f`?b5`
zV7Ml>UVr$slDx*AHg{AthzA?nDvqnV+TsHnHI)(OovW3@KyJ<YmudIc+DhsRMPs3{
zx?TAonQ{Ex^P=FN92p#vRDWC~0qpV8x|V%5t-nYE*zWDUX#(L8&Il(iV%+CT&d)J(
z)o{SyG*DkC65>4unx?Z;m#&DN#YIq;T*R0;^cu<<=rfI=2d$j-(TY21Tr?ihHvz#^
z0fPCap$2kscZx5culk&8ATCCbIkC#e@!l>DVIeJ_Ps-(knHt~PH)?%b$5$^fLr%2*
zH&V|MH~UaIsiEHrr&ABd;v6G(SNN+o?T!zO(8NZh?pUpaGriipqbghsY-o$`QXOxr
zIM|@6YA_$cmAOa07bZBKV?ttLlb|M-UR;_ZS%8unrQLagLu7a5M;0cE5$2kd7S(}+
z)o-_J{8)FntmXl7Tu7sMGm!YRKkV)n47o-?_d3Lyl(_m`Dw+n3luY=i>3U;QQ8K*g
zR?l3J{^zQw$>EotY)m%kz4Rt4WF$!%(^i4`CtMf%QcHzF+5HY=ZY&wP!Xy>VV0I-&
zX_GY$>*HbZ!3HIcKz`_T5~HnEk?qp1rPe}Ak;Y^(l&0J0eLMBcH5iR5dqdBRA{&-j
zyij};hfxj@fyka)Boc9w?h?U}o=pAd4`O_3Qf!zcA*o9%EJj?WIM-sb;K}*b6Kyq!
zh*Je+T5_$0m|zx~3rbYv4W_v?E&){?&(m;2F52p1&kzdJ4EjvHV_fepPqYt=yf#Oe
zNsnb|UTK-BS#as!U_z3r%7J_<vumM!pfuR(>_fU&i<Op=iY66V##oE9(!gdMLq{S$
zg9#4Yo<6BZ4<Sky9jgOa6hXaoeM)wUN~iA&>RFR(p9J-60G9Oy^{SHrRl4a}rL&?0
z#cm!*h8oD&ARvsQewlq^oRw>!5j4s`flk)qJ%UDP#_8tFiyFo4r5Xb!Z9~E4jQ9Oi
zBi4@kY~Dj17eOLO6zU>Wm^nll8c2lZq4l#HHNSAJM1y0Kp~y5yeL&%K*{XK75AVJv
z&<C{Mo=k6$pGb$?uA}(^9We$HR)^q*<>uxZ<vA?<i5{XO9?7YUwI@N>G?z6Rjk$6o
zYfqNcPj7j<+!q|uAs)~=dn!36x2Mu`0x)&w$s^ifPa-$uj-+mID@)(73TCOUubRP3
zc))(f;8wf!Od+mNSRyK+cTKLGj$ymk8091bH;cMD9zUL9e@xwawMGW_t4;KF3Bo6%
zp-qVu-9i!_-Tl@Q8yPL{eb)Y*u!9coew8jg3_d4Eg}p_XLkHUbMICp@Ksn9pUI^{O
zsrI3cFUhla<!8I$+>Qz-ZoR%_RAXPZWC4K6i!kAz4>8DB(Xv+&`<{)0mf2W77a60K
zq@NHN78WQzKEnitH67G+dy~Oz^0xF%o0Kr(d+2r`vMb0QvYnW_(z}v7F(o!Iz1}Q6
zWZx%X#xGJO0P=G{S*ipCe>%o1CCJlX1&OedP8UI^?htkc1??2+TxMs`{tgY9&UWnI
z-+{qxE<Ob9)X?c@3E2U5^GWxq45dxE28PNeO@hCJ&n%VQpG2_OB^nCk_wN&#Mv6H$
z*=LbK_}}Ca=I!zX<=^A?6jkqdr`If;SJ*ENJl`k(YVH}1k&<4J+lMny4ABZh$rQPx
ziWy3$$j34E&D{&_n|l|^kr*)dC1XJJ`aLvoCc`a?!haK<M;9mi_awqTsNT8z{qE_)
zXQR(7+9uB4WuWd~?7qI`k1M)wIO1F*l+giWp&B+oC;j7o%OnG`etiGl*N4ap7s%9c
zm2?P4(h%14^?fPr6+ImpOUD}N$g1g<3Ff6255IrS`hkTkJ3#z$$+F-px?B2VE4Xuy
zs|Dl1wsRXx5er)9m};P>$hx>x&y0lfQRSl=#(13@MF#BoE0(O=O@ggt;je$4OCX-j
zzi?!6&s#!aTk+w@{i{Eo);hb6hF+!##WXr<uzpbdhyb6^;X6_$Hqh-y=+-unhb%WN
zIK7P{@+o9U0FjaeQ6!|CEPMD~(+BO=%N-=-FbO#bu>i?kTud?_5atUq?F$0L{+DDi
z`jw6R_63>x1^J!WoV)LLj~9xU&E2?W|B8CU59gY=6D`+vtW<tcE+qd!xvS&0E(_bX
zG=Fiyq_5}QE529!$+x<D9PzQMgrLs1iZ>KdRV@{bR28`?eO+4U_TyVVO23dsWXZ%S
z_n*=WMIW1vb#ZU^CJWK?OUC+arNVqVF^vvs^s!B@-*!Fj6W#TcYlS7AB_<LCF7nd`
zNKp^*I;fA>774EhwFwb)au}T$ik<v7hwqG`s)cl-j*cuJVI9^q<Aa_13@6jxjv%8#
z$cQ*H+I;zD^O+j%M&ljZog(sbIhmP3Ci$O>zo_llP!W|Gk`>93ir=I_Vs|ykaIz~&
zs5Aa7RqJQPEeT%}zBX|4mVhn0)`TvL;b<_K<7j6W6ungzAeII+?e5sqvG;iR8PM6B
z`5^V0>Vxwp8`x+{F4SJx&yh@a?VLFgvsIgSSZV?_5oK}JsSTXIG3(rYrCkI=MutOX
z_XJCo2LVcf_#q=oh`X>}yD5HqDwn!_OQyeS^~NIGcFlH>v4%8+*2gsInmAo^28Lbx
zNKn8{W4p=@*R(brXl^`E)lq%e_HNMy4iCsNRPijPP4on_s9;M`tXLFlORUmy35_l3
z2UO?JR~<SJ?RnNdu_tN}O5IUhI!%qxe@UjU2w6BS9!Jzrd4mU9%O>mkvJEMD$;Em?
zkWfI5S;{tyRGW(nOeT^1Y4<3$<ROXY)h8l|BlR>3g(W$*Gz%rjI!Fp{snYhTVA#wM
z>7NddG<}Yg?MNxKrrR(s;D=D1CD{NiYqJ(3N`?x@5f~7_Vgzw%DGwuUqGfDpR$ZY8
z5O|J0)!{+^@szL(smdSKPtXi@5BjGi&6ZPA=v7i!WVI=AXqUT^@Ue6><guaH&7AIj
zLHJP(_(X<Dba;?I^~K3~DVbg=nd!P@B~G0VLfsg4LuqSrvCf()0wE5r9wh4i@iZY{
zZ<e$^sE33|_Lhgk`i7HFh2@p2-}b-4`CrlA`bv4b_LHL@i1lq9hu=FE`1bCDyX(25
zzVmBms@AVaiCV1DF5aLEQEB}x)vs%+7p$<XS5bD{hA}=GpV2$P(Kt>?UpYx<{!D#D
z>htTbQ~p#PIA*OotEoM6!g@s2c}gF3K@)xPxbC3p?za%__*QfNyCdH;e9k#sy<IFq
zQ7%O2a!ol!IcCa*O<TWi-nx0=Q*%bXQ*h8!SJYn8Uf<sS_S^f9SD)mn-nAU>#0)q?
zQl9LdV{Z}+y>lFA*zP&w<lgcKmtMR(KIopq_UxG^Jl6D%VtO?6Ik{Es@>qKBo!Fz1
z_|<w4u!HT`3Ol<M1IIQD^IM(m`koFUI=-@|rlPW@wjwS%IzBE+W1lfF>dCU&nkUPm
zHNB_l8^TI||5X~tTz2Jg|8wWMj-M0lbJ_R(kFOGYx?+XLqkG3QZ@#<!lK3fOq%m@A
z^!~Zy*oeenfQost)06MF*o*JJdr3nRd)^nWlNSnb719g5((_YuveM^4K)C76g=O<p
z^A;wCy{VZ;JhO^Y^U|}u(z9|v)O?-3;7JG18;ikkva)9)Lne>K;RoFi?ct6@;hcZh
z%2ocGR*Fwr`J@2|ki5IO^PQTQN95ZI`^k@wRTH*4uR5tLecy?i#LDN3Pzwp{)v$*@
z-#4GwyWi3o*zwV~P468nZ#&;!3ky6gwTwJh<6gDogP*&{^mGe*^K!HnBWF#o%&XQI
z*zb}AOM$*RBpJ*Bm4(JwOFl>ca=a=OgA6e<J|hXvu@S41@+vJwHgjoCr8%4HenhFT
z?Ernols%J+O2|n_*0_(9$tJ8@ew->YmvZg{WtU`Gs}lUuRs|dLYs~vO_kOZxW#%T^
z0b{FiUv_0$L3*JsH6c9E@3qL+(-x*KEeh<=*<#{zva>TwQ>`(ayKDj@D-SK(yfeo5
z`(D$Y56}en{@jpHE*F`v2DL;sQ1Or5N8&5B=G2;~6N#TRy$i25D=UucYe&?Ot5eI4
zS@-GBn2zC4K67Q3+n<mT`-?Z;`{Qcvo!paM)VVA2Wcb<JpLNt)n@e&m(hs`$q;`!T
zJ}rA^Zjw2%bo0@7cU>uIDYO*sx3!kERkdN8Y|iOGgDIy<UkK%9L%J?8e12k7dPr(e
z+Sa_Kd1d?j8<UF)sw*lj<_43khRe$@udta5@(rKu0R3@4*`Q|wjrwq7I2WE*8U@g_
zF+N70otPQ(VgA0n*7Em{<hJKFZpxj_C7-XVjuuOkeBro?*7o$al+tOLX{9To(!)~Y
zE#-}tM*S7PtX0{?Bg=Biayr7AGP&>Km#(wE$+<r(E0oz5c2#!LCB?df`U7-O3w>_e
zOV^6ajrE0=_QoH!6X)%>w8x@aQY^>AE=(z1%2mExvMX#NSDt<t+AF<W_f;Vql}n<#
zk4v^JH5bg%8gpKzDGTx0jGUax5{Y&>E-QkwPowLE{G`-`l)RXNjVEgAICsuTCc|yw
z`pINaw~whxDc6@46~uD%brL9K>$CEdIb~_3$XVe~d08eKrm!_Bxslu%1c)+q+WgF%
z*z}CJ)FORxnYqA<t@17UnriTw7pJEcYs~4kjNDAl{((21yU@yoRT*SpxHg}<-H7}m
z?BpgoYAsG5Ef^ITub;hS)?%t?tik2GX}S>iVd}oDBc>+nnU?aFwUv8JJ1=K*?#o?d
zvfr|*e{U_U$*;YX@Jrm^zGV8WZ#Z|IOy;uq+O|vK-$i|za=qXa)4lcNnc3&px6i45
zJ(t@NkdYXwq1+n@6Z3}Ujmf9|tV5GGES#`q)ryrN)OqLVn6-N%vlr*a8aGswUVZN_
z^+bi%CY((Dj_*JuGd6l{`t<f$)$=RnTDMyxiu6)eu&KSn*qYaxT%l{!HD*_vt(?hR
znqQaGlHF#iD%oFEb~xz-w=p4pXOy@r#;+n=8yw#GdiL7vo!d88?s(EID|g(@ycb8F
z55MVuKId+>?Jn`mKWyyC>o9Uhj~a51Y3^kQ`=1MWH{v@>O?7kA?aSv{(C2kBpPPrs
z><{TPxBL7x7yG?G5)iDdBrXW-xp;#v!o~f|9&@{}XV%o%36iMAi|2l%jK%=TwoDO~
zqfK_`%^8$N5TC1lpy?fSqh$q0eeHh<Y1yecso9)?{)2MMh4hB_qk1Ueq-Z<{X-7h4
z%UC2@$(^*{$pcMQKxIIA05^!B(D%z%=vCU3AFnzJZUk>kKbC%LP9bje6~J9Laos-j
zh7e4b4yBXmh>_`scayiKqMU5^0kU*OX%^ReygN?7?9HG789PMF?cdQCg`Dj1bO<%P
zg#6hy5Oq$|+qjaG?-iX^xg#@2#`?YpfB}hg#0hCe8u>1b4&mI_W?HjKGObCiiLHtI
zNy)$dCS&vRexNRA>Cim-5=UIpF#%Xg(tBo0nbJ`}G5e5@x;w~ws9$rj*n<VSnXVm2
zmxJDG`ETTzv-?k86aTT$@d%*YJ9qfhWk1WLS`8iEE=MMUx?IF1!O8mHBCny`Xq69T
zSL|53L>!$>AmXQ*yee|_igU@g<1~Lo%E^$uWcD&TS4sX&gN1v+U#|N|w45-VI;FIG
zfqw0(!)xu@4E+Z2<dPZtCQB5bK&sN+llRRyxkRJikg;~H61t~X=~cVesZ%D)331&=
zUk2Y(x!;j6>wvD2G@7<rf?V%$-{LL5j8f9;*A0Z3e?2GP)A4V|zvW@!yOIJ_p-e_W
zgh^|Iz>Z@yxOBpr65BeIhsxTU8bwTO-Q<mp?PVRRL+uISjlg)&<@O==p|p}z%H3<`
zU#E;={(Q$bj&D5Y&o@x!`h2_e`TBgq8m?Wpe`Ei~b51uSSoFY!sHE78UgNm#xIqg^
zq1=+1j9^Q0G9(7TaYkKZV-0(&V!Z5Q9Ak^qteMtauiQL|KITWY`xXORYBQNFn(mP9
zkg94!YizZb<O6s1-0Um6n1Uh@5B#6v@k+NJ(v2@6zk$5JdA-7}McyAZsYr7?UD^$h
zl(bYTJ)P2RZUXXN%J(=K`cgr5tx%Owr7wfP77Cj$Z1+9!n#$q9yym-M`F23mB$@HV
z3@E_*Dq8oIH&se5#Y1@)^<=2Q@e&H4vUMrIvLa90!1$h8cL@mr=7W<bKhOYa>4$tk
zNwzh^qM-)+OLF4b#Uk|bP##vfFQFA&)s89MooA#eMPF+qia2fGKh)2fyKj;i3K6v$
zN5RuDh4odOK6>=DNdCV3co++OrG3X`#}4U3&#=p=g?qZ1c6R@L1|?eEr6gIPf7pY=
z4(%oU?;m@8_x@K~j;`b4%A2CzQ@z(*TUo9-dh)BI->&*&(O$}j1#tF>i||a;0NT&|
z8zGS!&y(06lGQ)BAM%!;Mm~mKhp@dBfAJ0l`|Ei9_gz{pk`}s8K)o0epL7v1dLj{P
zG?|T-Y>QX61&sdrwCj;4xxiX7!SgRdf+0_zMZ3m%N*kw?hZ<56yyzloq+Jj71^S%S
z46n2dbR8wVz|yWUQk7b^-YZ*ggn9###768!jTvdVx_rG?zP>o!oK3pMcw%E@T#GYA
za|X(A3rN>PG=mx?rT0t=XqO9%K^lVJBVFSxS(ZGr$qVHM7K;+iB3+N<?qT2-LCqT{
z;Rv-zS^X`+99@3J_=ZYmSGssIeM|M<g1)0<CCMgLpKBHk4uDJUu5_^E3_W9K4;{)k
zwSx!bu3)XtY>BOktqLuS&~brtUyYxo%28vd(`5XI67K&m4fLT}bPf7?ZFy)e=a`g8
zKcn9}CMyZJ3{R0Jl!?}p=TI?+{^8db`a)rhBwGvy0!g-b403ZjQJ4r|1BheCqS|FN
z_;E)nE_=&$sITq;AA=+sw;FT|01X#POn*|k^QKfq?1O2}7W+-08?@kFGyHZ1!E3yp
zQFxI73M5Wn^X$FLP-)Qsg;zv`VS7Uak(MqtjG!Kv1O1JZ6GOSWNo|m)+C*ctVbHyZ
z^wQbNGRstw%p2mYOF_|YAf6aQ7mLWDN%;9WpzL!sXuzns4ji0n{2utzcX}SV-t>?>
zhq)M#kXlrZlLg@I8;U9pyyLY102_%zuQs~J(2`d4yf+(K=KhH{o77_z3`s|(0D;<>
zBag~YNJYpqJ~b^$+(_M)4K+Z*hlZ?4i7w^V@3;K~hUML@_r}(VK}Fia8OCA8DY@&x
ziW;%2ET<~_Xlh*$XK2_~Fj2J9ytP3F<&NluZ6nAw&amQ-O^Cjy)g)MP^tjUS0uelC
zO*!(diLnMlVnXR24XBhP?$|=CCy_LXn933MV%avxD`8Q2W$pnnhm5~bXHZ_N{hq%1
zXfbEFx$dl0B<2D+Q5lyK7lSg>y!R7~Fhe;oszDC8CX%eiBc>n|-+7eS$qlHP`Uldz
zC;{6JJsOFJ?lsav)X&=o{Y=(cKP4;e0YvdHBD#~i-P1^+5aRC}<6pC>Ch2rbSM#xp
z)m^dg#FVWL_2(WEH$fk#O^YjvE6%L4R9Y%(iz_oK(@in@IF8I6BwQrQ*D$FW(Lo2d
z)5!~b>9fAn{UYcphWf)tMBVQOGWXMq#2Z4fNS0U8HHh*qrYD0r_d*|fG6b<iWh_~P
zf0*PgCMWlilb~RxN^HoHOIZT(rp%pTtztxRe4*_?^Nsuuf{v~DnoiE5OXhIKmEonw
z#pds`Zf`z1@>g2^B8{feKug`Fv+3+na6?{F(v$XQ=^{I4Pat=IOpOv>>vj=VUVgcL
zc08Krvo}2^sA#WgZ);2|1a8P(1KUyDnbI8898|nDWheYNeNe@o{rdg-0~MD24Yw1%
zTlpbEXh0-GhUo?R+PidWAT|m}i`-QlP#kb5@=!5h>d7;^zZw6{OSRN}7j?#J-LC4L
z?J4ak9n~f6+>h#$WI_;R`4nWJ<b`?5|NIEByWl_p#XT<Q#<Mq&^&7zfG88AVfVyzY
zz!dmgAn{Tmfco6Qi^NIeNb{DP1^W0+=(b*UJgcNDu7aRKPL(350k~%cVAMY;dj`^a
z0q5R=@&EB9(GiFEHuzfKK}Kev2xU~|RpLWd^aUQw_q|<ekAyWm-(7cCqWuFaoPB{n
zKO<%IeR|KJY`xd;Q~7V4!EP8{W=|67w{TRZ)6Wdbyf(v^IVR+r66}i{8^}ZvaDwmb
zp0w7W%+8=MWW0%<k7`-_2FF4=Pi%KAu@`+|P{t1lAH8eR0u<FQC0iXs;Ob3u3@7c!
z$msKk4d>FWkQPYE<sxEYBO)NPry*@d7(qX<T)u}kMANgx-<~C+Hp52hu9M=M(!V-L
znYi47^?gLz1#dbU{G;-F;8qmvjkK0Q6tY%+4sl|n6XhDaH$mwGBn}mJjSF#^`~cDA
z-tQ1wo;-oNXcpl(?11w~&I`NL7%4}C!s02b72y0s6zA1sIF^I64a7Jg$O7+0T7K#i
zf~*@)39@EMFtWcvjs*smzH&;GDocb?$;nx1$eH*dXA+;HkBL!6TVg9x0OG)cG^;Jk
zrjkswq%--yl~Arz!=KSUttE#8(g}Wt|1hQzfQOuWipd`WwN!#MFN~r_Oz49Sk=OqQ
zIl;6Wnt1YpoW{I@tV$5gRMKmt1l(iSz+)dlrjgV;qJ8CG0$o3n#@aW?YxLUENR?v)
zlYoLHfaE+JGnoMLw)_YE7oR=o!?ny`y6N>;5p%sLqyABEwddY~=7?66J%}M`j&OOj
zr7o!+!)Tv+0fv&kyhC!&Hu5E6J03m%Ci`%|9`w8*B)|SLu+|f4z@mvWro6Z;K<g8_
zzGhTsAej|P<ndP$Ka4(~cr^W3<dGd$R=u&jn2RW|H0~|uir7b2Kl<vI^ZP4LR~=0`
zoN{1w>E`k^W?%EV(n(oV`O$)#v(FOQzOxM{>fS9l&RK|TP1&flv#^A(+&EEu(fn;r
z=bDxP<A&6Y97_MgC4F%NiK1JgrP3xXYGbtSVeW@evyFlmIt36&)H1Heb<L00*b@$`
ze>|)vi%~c?1(jy`9cr@oihO(rpybjAhveZ+VeFm+#p!lWi6Ba<0{>fK$93><JgEam
zC;FQqUQ%azMh%f|scsjwYL`(~gEdmA+W)6MT%s@dzh7ZZKqfR{XvS{5G;F)N_WgAO
zHwH3ckAyaNry>1hPBJ&ybFv|_7iAM<Q)P2yi)FsD7+HcWMV2PZk{M)XnN8LtJ0km3
z_POj|GKZ7GX@Jx7P6M4ZPD7nWIgNLk?DUS)d8f-xpE%ug`q}Asr@x(OKUqI{zp?!$
z_j{$^!hZGr4)*)2U$3)^^Rv!7odcbdoHL#Cor|2yovWSeoi94yb{Xd4=d#viqsw-e
zAeShYWS2UZ8!n%@{NvK&@>o7Vu9gpxkCu;@zbaoKUm{;AUoYP*-!9)RkC7+Jb@HR~
z6Y}%&JMu^JKjoC^$1sd5^DHxlna0d!<}(|aEzC}44->+aGF41H)52V2t}r*5JIpW4
z-^{;RXSP2(fOThIVAbr4?09w#yM$fKZeq8wfov!n&1SMKY%6=3z0Tfe|6m_;E?j@k
zjT^_!<b1gxPRkW>_1ppO2zQnHn!CsS*1v!MXZtJrPwxLl|K<JH^xxD!uzyVdr2e`6
z3;SF9SM`6p|MC7G_rKr2N1;%7C_EKv#W2N4#Y>9$igk*OiZDg0B41%uR4M8eEsEoc
z4;5c2epe91W4<5XpI7li_>uf%{uSPbU%{{D*YSaTI3L4j@g}~Uujlvi2l-C^1b>#l
z#9!ra@}KkH@b~%Ok!56DJzR&nj&+^n`m*aB*VkSBT-Uj7a^3E_+cn%Z#x>DZ@0#gq
za5cM@x;D8UaXsdG%JrP<2d-CKue;uH{m%7Q*MD3eyScdacN^g5?k2hobsOn6-fg<u
ze79w8Z@KxqZFSq}w#O~hphkx6XY#XsmN-Bdtmbpn1&Z0wTOrv6C}+|RRSX=YU{Q)H
zNs&($`Q0HQyZI9J0T@M)Puz@G@bEKHy~Q(Q6uj_cw<m|tfN-o+7^#++LTAyR>CcW?
z4*U|uFI`ly;}rZHg@%7gA@V@v@{`pX9?(?*KQo!1hlNjBh#=MuYs`aMt6Mh$)G)e{
z_!Zddc=3NFF?(6TV@4$MHemV55+M0O6NvCXND3Pi5{f@S!M}=@55|F>(ql;67$VQi
z#!1Xl^UQb!Uzt>TU4d$2UKp<CrMZ`eZ)<63;|Ih0=FQb_|5mBY%gxWu6N^#?eVD1r
z$O#M#-VwUP3v(2sUJu}%>##Dt&#%d<%t1>tQZx{BLkVj9+r!N<VoHKOE-j9W$)Q3L
zkcETTL?R?r=ZM9zmiUqcE;VNB@OWj8F*`p;OfUIoX`#w?>$ftC#*&Md1z0@SVTqPo
zBWx;O2v@`?`l#@DiAzW1rOU!IixfN(7wb;WR7=bS;QYnk>FSdw*Wr9$QuD%8HIGVg
zzX@u7ySi1)6Z1AKl<uks25V_zA)!5`lLbnNX=@4Dt)U*^@N?Ot=HGy7gv^;=eRRud
z{)oB+Q&T^OpH9)^0b%NyfHRJyvwsxJ))%i`qofO8f>$w<4iKpeJb=x<6lFTWd3hn4
z>BzdySVtDCE?qEL-q(D+{Na}0!<*7uGWTVb=aktBD&bTWR3(?_Y&(u^J{<Tp$EzHH
z21lSHP{{+rVSY*(YGxcBs8O}MvK1K>;1(fQ=oL>;jKrJbIpPgPlAuom4vF+5@{#;d
zz-#+g1NZ=j&>m2(4j{vjnvw85e2rlV(mqn}>Ot?4W>35j^f&|+t%-@*8A+PW@uAx`
zf#jdCo0_*Bix)FW%d*Q<`&&wC>NIv=`z*V!W0pL+u0^+BRaTZ~D%BjTXzuJ%ddvH>
zU7?+uD@0RLVt%4(04^K<Kip0Zk3McEqnh_u=yeADas>}+%L345^1<l0_B{rBpnbQJ
ze-U2>Zcy+GQLi~sU8R6r80y~d(z_ECh^H~r)#NT-SeByTpYRC$hc-aOlNgM6$v_1C
zMhpfMHh7+Yq70;+FVyU77W0^)vA!dyt12YiR1~ApL5DB3U<Z|ZmSmP@mS=LQjfc{^
zR3KXK2MFsX4X**97IccBrvhk}$x<v$RE%+WJOU$^^y!$QkJlD74md5X8kAM?NHrd4
z5hIu68}P%WyXLV>%}-a`1MC6lh<(#wSQuWY2FSV7-l;<ssKbwySSevk;6imW=?V$!
z6OD}#3Bf$IkG80lOoiQj8}yD0Z38cLJZ;l}zDPQ6iAC>U-+ucq*b%7`6Ib9V%j16n
zY5$>qi$%+r6gAJzQ}gH4-80IiyI-#8o}pLp=M=p&%5f6q>fRZt3SQ#9;z3tGT1^M@
zE)x__?9h{m^%GmvCsD!UQhV@_geZa}1WyNp+E2ifH3j$2TQ1|*t7{ef1jS=NW}|xP
z>wu>$hmnE89A70}BVXD8I__hjJ_cL3l>wJ{zNOq;ZYuH0FM?QL5tq;8rRS%c((AXP
zbe$H;iIL&ADPV1%FkZnq;o1E3f(%ne_xj%To(&Pz<~AkIW*23dGEBYeyVrY0G{mH$
zKY<Jptiv$373D=GCHQH$lTwe5Rq%c6X1riPt3frS#pQ01D(|(d7B#}K^IFXJ`PJ&j
zH<{7!Sy!w1XhSr*Ez$PCf#uDJGyrx4K*1A^c=#>g-}2xM#R?ugS*Ljf<2-n8R>liY
z_&A=@abPFUlw?CpNiz2YqFG#=FLAvhASTCMEW;0&e@ByGk9Y_Tn9)BbsX+rmBw7aI
z$&k)gv@DyvtK{Po(o<6SM73o1cyE`}*y;a$@e(WmXN+?8!qo~XD9N)hiSb3y`QQ;1
z7o{6%=AkgczpgM)KVDu}S9;(eu3ScjX5G4pI2$yLc;nZpStp&L6lg*01%LihKV>D4
zh|tGj;8<{e#Wg&Jn-zRSgW*U+1Mg%oDV%w5k{F0Tk1$TJxfM<%Q7HhS=X11Ku^CAb
zi3{Tw$L-LCCnQC2sjM+vo?%Qi>O^auk*`+_cuMC>eP%qB`5~_<`0a{bK%Sth(hbCu
z#Fxw;+tm*KIdIUm@9N{A6M5;;kp?k|W{Yo3sMA`+NK0sDTx{Q?qUHxrojMq&&hjM|
zV_vbw1mQAOHmgg`%!H(DiWiUzjq-{1FIjYAQ1hS*z&k}Bme*--sI`%r*L>uXykxhA
z@2G9;d<W(aVr{i4HL)dPtW|4_(WhtWGLq9dJ{7zB{GcAdHZ5CRvRFl(=&rSRS_YF{
zYcVZ$BD+qJ!5DIbe+eEn3V-&E+V{&Z`{a1&93B!pn4hRPxo*0emppp2{RMIa^tG;-
z4R+E1;)YszZ<l5s^y>ymaO?^9D1S};U;^KbX8`sUQm9@$?LT%b1vU0qK#)oQpPu~1
z5|RI~k7R)Vk29nh(3~XK9zUIOL4~x(n6D0A)&LciUy#S~li2FGlJGF4gu$0CZBt2O
zb)|Tp<%O{dJ{kSlmdqbaI-sq`Bc{RA)W>?v$HrxbTC`q~mb!#S6(6GB1M2{jQC<XH
zne>zr4UfpzM)b|+6>&*Cd1m<U^cl%I0LtY|R^(_7pRBojSy`G}mQyP7?3-%7$Y9DX
z`Y+Fp&1S2pR1PRAE;gIYUUs>?Nq*mP8KhpWgynDUB>cpQJ1IYRZU^PR|GtCpH>Dv$
z%T7=LWV%r;aRl&Ar@fhh2n;Ueg}G>lN1>WG6%`blVN2i@*+8U|Y~zJhemj<o^}lrE
z(_`nZeev;>>7wHweybP}6B8X5@%KxfR$ED_&C1m@@Wa#tpo&|RQIu{=FH0?@8WQVC
zTX=%e;B}(;gs9JwK`4%x+xMj6p5dl?@N8R*IU+*D!_s5MyqE_eGrk~6!?Wlp!@x5&
zAFpBHOyvb>)LF3N{Iy2)weu?$i2VCl>wSsNR6!>rnm)I4sfMR)Zfv%E=N7|y)ol7)
zeqI^|hcOZ*ecz83!uH+39kYQOD4G8+3EES}GhBSFlffum38#9=2qCl5XH$5xSkb~e
zcTrY2r{En(-OxDNPGA0uzDQmkhYgn~c%ynid>SMYP|b2v0!n9O!}{Xcf%%0X_k;)h
za-TDrl6Z5RHJ;NI6=(3iC)a%-B@Qq&RrGOX-(-2<H*aF(<>^frK3@0;=11^m`Z>X(
zY&S$i@C`655t3m+StJj!Bv5Sgabg3;UJz9>u$}N#se$_QtTb>x;gy6moZ={ko&aAu
z(1$8idxA~2aB0Zwd(DJR!(AvgS&KB*3Ug&i1$XNVFX0-x5o_inB%hC$cdB`b)}#>T
zY#tySuseKd1;|HOiK^Qqn)n_dq<QTX^&X!1EkvttegT*KDSsdCCZ1j%mJf~drv@H@
ztKB?oI))3-7`W)LD>0iDd}CdCJ>1^9%;wli9$KqfXst#{TC2RU9ge!hU<Ats_=~`6
zfb@iKDDd%#iSa!0BI?V$m=7p!DtNAW0m&ww8kv+-11@fPQk)UzP+4h#cqlJOeiM%f
zy9ZyPswn3{{eu>{l5u$GDu^pCUj~|^P&2$(s({I(A>l|zdeA5g^DojrwX{*0oX5K$
oMn#JeYGU{pbzL1U=O}QI<Cx4B=r}K`2S}bYAE*l4!^>p<2eKYy*8l(j
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.eot b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.eot
deleted file mode 100644
index 7c79c6a6bc9a128a2a8eaffbe49a4338625fdbc2..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 38205
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
E0QL`y%>V!Z
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.svg b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.svg
deleted file mode 100644
index 45fdf33830..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.svg
+++ /dev/null
@@ -1,414 +0,0 @@
-<?xml version="1.0" standalone="no"?>
-<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
-<svg xmlns="http://www.w3.org/2000/svg">
-<metadata></metadata>
-<defs>
-<font id="fontawesomeregular" horiz-adv-x="1536" >
-<font-face units-per-em="1792" ascent="1536" descent="-256" />
-<missing-glyph horiz-adv-x="448" />
-<glyph unicode=" " horiz-adv-x="448" />
-<glyph unicode="	" horiz-adv-x="448" />
-<glyph unicode=" " horiz-adv-x="448" />
-<glyph unicode="¨" horiz-adv-x="1792" />
-<glyph unicode="©" horiz-adv-x="1792" />
-<glyph unicode="®" horiz-adv-x="1792" />
-<glyph unicode="´" horiz-adv-x="1792" />
-<glyph unicode="Æ" horiz-adv-x="1792" />
-<glyph unicode=" " horiz-adv-x="768" />
-<glyph unicode=" " />
-<glyph unicode=" " horiz-adv-x="768" />
-<glyph unicode=" " />
-<glyph unicode=" " horiz-adv-x="512" />
-<glyph unicode=" " horiz-adv-x="384" />
-<glyph unicode=" " horiz-adv-x="256" />
-<glyph unicode=" " horiz-adv-x="256" />
-<glyph unicode=" " horiz-adv-x="192" />
-<glyph unicode=" " horiz-adv-x="307" />
-<glyph unicode=" " horiz-adv-x="85" />
-<glyph unicode=" " horiz-adv-x="307" />
-<glyph unicode=" " horiz-adv-x="384" />
-<glyph unicode="™" horiz-adv-x="1792" />
-<glyph unicode="∞" horiz-adv-x="1792" />
-<glyph unicode="≠" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="500" d="M0 0z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1699 1350q0 -35 -43 -78l-632 -632v-768h320q26 0 45 -19t19 -45t-19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45t45 19h320v768l-632 632q-43 43 -43 78q0 23 18 36.5t38 17.5t43 4h1408q23 0 43 -4t38 -17.5t18 -36.5z" />
-<glyph unicode="" d="M1536 1312v-1120q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v537l-768 -237v-709q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89 t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v967q0 31 19 56.5t49 35.5l832 256q12 4 28 4q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -52 -38 -90t-90 -38q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5 t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1664 32v768q-32 -36 -69 -66q-268 -206 -426 -338q-51 -43 -83 -67t-86.5 -48.5t-102.5 -24.5h-1h-1q-48 0 -102.5 24.5t-86.5 48.5t-83 67q-158 132 -426 338q-37 30 -69 66v-768q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1664 1083v11v13.5t-0.5 13 t-3 12.5t-5.5 9t-9 7.5t-14 2.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5q0 -168 147 -284q193 -152 401 -317q6 -5 35 -29.5t46 -37.5t44.5 -31.5t50.5 -27.5t43 -9h1h1q20 0 43 9t50.5 27.5t44.5 31.5t46 37.5t35 29.5q208 165 401 317q54 43 100.5 115.5t46.5 131.5z M1792 1120v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47t47 -113z" />
-<glyph unicode="" horiz-adv-x="1792" d="M896 -128q-26 0 -44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5q224 0 351 -124t127 -344q0 -221 -229 -450l-623 -600 q-18 -18 -44 -18z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -21 -10.5 -35.5t-30.5 -14.5q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455 l502 -73q56 -9 56 -46z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1137 532l306 297l-422 62l-189 382l-189 -382l-422 -62l306 -297l-73 -421l378 199l377 -199zM1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -50 -41 -50q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500 l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455l502 -73q56 -9 56 -46z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1408 131q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5t43 97.5t62 81t85.5 53.5t111.5 20q9 0 42 -21.5t74.5 -48t108 -48t133.5 -21.5t133.5 21.5t108 48t74.5 48t42 21.5q61 0 111.5 -20t85.5 -53.5t62 -81 t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M384 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 320v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 704v128q0 26 -19 45t-45 19h-128 q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 -64v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM384 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45 t45 -19h128q26 0 45 19t19 45zM1792 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 704v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1792 320v128 q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 704v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19 t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1920 1248v-1344q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1344q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" />
-<glyph unicode="" horiz-adv-x="1664" d="M768 512v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM768 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 512v-384q0 -52 -38 -90t-90 -38 h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" />
-<glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 288v-192q0 -40 -28 -68t-68 -28h-320 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-960 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h960q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1671 970q0 -40 -28 -68l-724 -724l-136 -136q-28 -28 -68 -28t-68 28l-136 136l-362 362q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -295l656 657q28 28 68 28t68 -28l136 -136q28 -28 28 -68z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1298 214q0 -40 -28 -68l-136 -136q-28 -28 -68 -28t-68 28l-294 294l-294 -294q-28 -28 -68 -28t-68 28l-136 136q-28 28 -28 68t28 68l294 294l-294 294q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -294l294 294q28 28 68 28t68 -28l136 -136q28 -28 28 -68 t-28 -68l-294 -294l294 -294q28 -28 28 -68z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-224q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v224h-224q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h224v224q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5v-224h224 q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5 t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-576q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h576q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5z M1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z " />
-<glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61t-298 61t-245 164t-164 245t-61 298q0 182 80.5 343t226.5 270q43 32 95.5 25t83.5 -50q32 -42 24.5 -94.5t-49.5 -84.5q-98 -74 -151.5 -181t-53.5 -228q0 -104 40.5 -198.5t109.5 -163.5t163.5 -109.5 t198.5 -40.5t198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5q0 121 -53.5 228t-151.5 181q-42 32 -49.5 84.5t24.5 94.5q31 43 84 50t95 -25q146 -109 226.5 -270t80.5 -343zM896 1408v-640q0 -52 -38 -90t-90 -38t-90 38t-38 90v640q0 52 38 90t90 38t90 -38t38 -90z" />
-<glyph unicode="" horiz-adv-x="1792" d="M256 96v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM640 224v-320q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1024 480v-576q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23 v576q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1408 864v-960q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v960q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1792 1376v-1472q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1472q0 14 9 23t23 9h192q14 0 23 -9t9 -23z" />
-<glyph unicode="" d="M1024 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1536 749v-222q0 -12 -8 -23t-20 -13l-185 -28q-19 -54 -39 -91q35 -50 107 -138q10 -12 10 -25t-9 -23q-27 -37 -99 -108t-94 -71q-12 0 -26 9l-138 108q-44 -23 -91 -38 q-16 -136 -29 -186q-7 -28 -36 -28h-222q-14 0 -24.5 8.5t-11.5 21.5l-28 184q-49 16 -90 37l-141 -107q-10 -9 -25 -9q-14 0 -25 11q-126 114 -165 168q-7 10 -7 23q0 12 8 23q15 21 51 66.5t54 70.5q-27 50 -41 99l-183 27q-13 2 -21 12.5t-8 23.5v222q0 12 8 23t19 13 l186 28q14 46 39 92q-40 57 -107 138q-10 12 -10 24q0 10 9 23q26 36 98.5 107.5t94.5 71.5q13 0 26 -10l138 -107q44 23 91 38q16 136 29 186q7 28 36 28h222q14 0 24.5 -8.5t11.5 -21.5l28 -184q49 -16 90 -37l142 107q9 9 24 9q13 0 25 -10q129 -119 165 -170q7 -8 7 -22 q0 -12 -8 -23q-15 -21 -51 -66.5t-54 -70.5q26 -50 41 -98l183 -28q13 -2 21 -12.5t8 -23.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M512 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM768 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1024 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1152 76v948h-896v-948q0 -22 7 -40.5t14.5 -27t10.5 -8.5h832q3 0 10.5 8.5t14.5 27t7 40.5zM480 1152h448l-48 117q-7 9 -17 11h-317q-10 -2 -17 -11zM1408 1120v-64q0 -14 -9 -23t-23 -9h-96v-948q0 -83 -47 -143.5t-113 -60.5h-832 q-66 0 -113 58.5t-47 141.5v952h-96q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h309l70 167q15 37 54 63t79 26h320q40 0 79 -26t54 -63l70 -167h309q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1408 544v-480q0 -26 -19 -45t-45 -19h-384v384h-256v-384h-384q-26 0 -45 19t-19 45v480q0 1 0.5 3t0.5 3l575 474l575 -474q1 -2 1 -6zM1631 613l-62 -74q-8 -9 -21 -11h-3q-13 0 -21 7l-692 577l-692 -577q-12 -8 -24 -7q-13 2 -21 11l-62 74q-8 10 -7 23.5t11 21.5 l719 599q32 26 76 26t76 -26l244 -204v195q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-408l219 -182q10 -8 11 -21.5t-7 -23.5z" />
-<glyph unicode="" horiz-adv-x="1280" d="M128 0h1024v768h-416q-40 0 -68 28t-28 68v416h-512v-1280zM768 896h376q-10 29 -22 41l-313 313q-12 12 -41 22v-376zM1280 864v-896q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h640q40 0 88 -20t76 -48l312 -312q28 -28 48 -76t20 -88z " />
-<glyph unicode="" d="M896 992v-448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h224v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1111 540v4l-24 320q-1 13 -11 22.5t-23 9.5h-186q-13 0 -23 -9.5t-11 -22.5l-24 -320v-4q-1 -12 8 -20t21 -8h244q12 0 21 8t8 20zM1870 73q0 -73 -46 -73h-704q13 0 22 9.5t8 22.5l-20 256q-1 13 -11 22.5t-23 9.5h-272q-13 0 -23 -9.5t-11 -22.5l-20 -256 q-1 -13 8 -22.5t22 -9.5h-704q-46 0 -46 73q0 54 26 116l417 1044q8 19 26 33t38 14h339q-13 0 -23 -9.5t-11 -22.5l-15 -192q-1 -14 8 -23t22 -9h166q13 0 22 9t8 23l-15 192q-1 13 -11 22.5t-23 9.5h339q20 0 38 -14t26 -33l417 -1044q26 -62 26 -116z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1280 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 416v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h465l135 -136 q58 -56 136 -56t136 56l136 136h464q40 0 68 -28t28 -68zM1339 985q17 -41 -14 -70l-448 -448q-18 -19 -45 -19t-45 19l-448 448q-31 29 -14 70q17 39 59 39h256v448q0 26 19 45t45 19h256q26 0 45 -19t19 -45v-448h256q42 0 59 -39z" />
-<glyph unicode="" d="M1120 608q0 -12 -10 -24l-319 -319q-11 -9 -23 -9t-23 9l-320 320q-15 16 -7 35q8 20 30 20h192v352q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-352h192q14 0 23 -9t9 -23zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273 t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1118 660q-8 -20 -30 -20h-192v-352q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v352h-192q-14 0 -23 9t-9 23q0 12 10 24l319 319q11 9 23 9t23 -9l320 -320q15 -16 7 -35zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198 t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1023 576h316q-1 3 -2.5 8t-2.5 8l-212 496h-708l-212 -496q-1 -2 -2.5 -8t-2.5 -8h316l95 -192h320zM1536 546v-482q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v482q0 62 25 123l238 552q10 25 36.5 42t52.5 17h832q26 0 52.5 -17t36.5 -42l238 -552 q25 -61 25 -123z" />
-<glyph unicode="" d="M1184 640q0 -37 -32 -55l-544 -320q-15 -9 -32 -9q-16 0 -32 8q-32 19 -32 56v640q0 37 32 56q33 18 64 -1l544 -320q32 -18 32 -55zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l138 138q-148 137 -349 137q-104 0 -198.5 -40.5t-163.5 -109.5t-109.5 -163.5t-40.5 -198.5t40.5 -198.5t109.5 -163.5t163.5 -109.5t198.5 -40.5q119 0 225 52t179 147q7 10 23 12q14 0 25 -9 l137 -138q9 -8 9.5 -20.5t-7.5 -22.5q-109 -132 -264 -204.5t-327 -72.5q-156 0 -298 61t-245 164t-164 245t-61 298t61 298t164 245t245 164t298 61q147 0 284.5 -55.5t244.5 -156.5l130 129q29 31 70 14q39 -17 39 -59z" />
-<glyph unicode="" d="M1511 480q0 -5 -1 -7q-64 -268 -268 -434.5t-478 -166.5q-146 0 -282.5 55t-243.5 157l-129 -129q-19 -19 -45 -19t-45 19t-19 45v448q0 26 19 45t45 19h448q26 0 45 -19t19 -45t-19 -45l-137 -137q71 -66 161 -102t187 -36q134 0 250 65t186 179q11 17 53 117 q8 23 30 23h192q13 0 22.5 -9.5t9.5 -22.5zM1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-26 0 -45 19t-19 45t19 45l138 138q-148 137 -349 137q-134 0 -250 -65t-186 -179q-11 -17 -53 -117q-8 -23 -30 -23h-199q-13 0 -22.5 9.5t-9.5 22.5v7q65 268 270 434.5t480 166.5 q146 0 284 -55.5t245 -156.5l130 129q19 19 45 19t45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M384 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M384 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1536 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5z M1536 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5zM1536 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5 t9.5 -22.5zM1664 160v832q0 13 -9.5 22.5t-22.5 9.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1792 1248v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47 t47 -113z" />
-<glyph unicode="" horiz-adv-x="1152" d="M320 768h512v192q0 106 -75 181t-181 75t-181 -75t-75 -181v-192zM1152 672v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v192q0 184 132 316t316 132t316 -132t132 -316v-192h32q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1792" d="M320 1280q0 -72 -64 -110v-1266q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v1266q-64 38 -64 110q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -25 -12.5 -38.5t-39.5 -27.5q-215 -116 -369 -116q-61 0 -123.5 22t-108.5 48 t-115.5 48t-142.5 22q-192 0 -464 -146q-17 -9 -33 -9q-26 0 -45 19t-19 45v742q0 32 31 55q21 14 79 43q236 120 421 120q107 0 200 -29t219 -88q38 -19 88 -19q54 0 117.5 21t110 47t88 47t54.5 21q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1664 650q0 -166 -60 -314l-20 -49l-185 -33q-22 -83 -90.5 -136.5t-156.5 -53.5v-32q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-32q71 0 130 -35.5t93 -95.5l68 12q29 95 29 193q0 148 -88 279t-236.5 209t-315.5 78 t-315.5 -78t-236.5 -209t-88 -279q0 -98 29 -193l68 -12q34 60 93 95.5t130 35.5v32q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v32q-88 0 -156.5 53.5t-90.5 136.5l-185 33l-20 49q-60 148 -60 314q0 151 67 291t179 242.5 t266 163.5t320 61t320 -61t266 -163.5t179 -242.5t67 -291z" />
-<glyph unicode="" horiz-adv-x="768" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1152" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142z" />
-<glyph unicode="" horiz-adv-x="1664" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142zM1408 640q0 -153 -85 -282.5t-225 -188.5q-13 -5 -25 -5q-27 0 -46 19t-19 45q0 39 39 59q56 29 76 44q74 54 115.5 135.5t41.5 173.5t-41.5 173.5 t-115.5 135.5q-20 15 -76 44q-39 20 -39 59q0 26 19 45t45 19q13 0 26 -5q140 -59 225 -188.5t85 -282.5zM1664 640q0 -230 -127 -422.5t-338 -283.5q-13 -5 -26 -5q-26 0 -45 19t-19 45q0 36 39 59q7 4 22.5 10.5t22.5 10.5q46 25 82 51q123 91 192 227t69 289t-69 289 t-192 227q-36 26 -82 51q-7 4 -22.5 10.5t-22.5 10.5q-39 23 -39 59q0 26 19 45t45 19q13 0 26 -5q211 -91 338 -283.5t127 -422.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 384v-128h-128v128h128zM384 1152v-128h-128v128h128zM1152 1152v-128h-128v128h128zM128 129h384v383h-384v-383zM128 896h384v384h-384v-384zM896 896h384v384h-384v-384zM640 640v-640h-640v640h640zM1152 128v-128h-128v128h128zM1408 128v-128h-128v128h128z M1408 640v-384h-384v128h-128v-384h-128v640h384v-128h128v128h128zM640 1408v-640h-640v640h640zM1408 1408v-640h-640v640h640z" />
-<glyph unicode="" horiz-adv-x="1792" d="M63 0h-63v1408h63v-1408zM126 1h-32v1407h32v-1407zM220 1h-31v1407h31v-1407zM377 1h-31v1407h31v-1407zM534 1h-62v1407h62v-1407zM660 1h-31v1407h31v-1407zM723 1h-31v1407h31v-1407zM786 1h-31v1407h31v-1407zM943 1h-63v1407h63v-1407zM1100 1h-63v1407h63v-1407z M1226 1h-63v1407h63v-1407zM1352 1h-63v1407h63v-1407zM1446 1h-63v1407h63v-1407zM1635 1h-94v1407h94v-1407zM1698 1h-32v1407h32v-1407zM1792 0h-63v1408h63v-1408z" />
-<glyph unicode="" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1920" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91zM1899 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-36 0 -59 14t-53 45l470 470q37 37 37 90q0 52 -37 91l-715 714q-38 38 -102 64.5t-117 26.5h224q53 0 117 -26.5t102 -64.5l715 -714q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1639 1058q40 -57 18 -129l-275 -906q-19 -64 -76.5 -107.5t-122.5 -43.5h-923q-77 0 -148.5 53.5t-99.5 131.5q-24 67 -2 127q0 4 3 27t4 37q1 8 -3 21.5t-3 19.5q2 11 8 21t16.5 23.5t16.5 23.5q23 38 45 91.5t30 91.5q3 10 0.5 30t-0.5 28q3 11 17 28t17 23 q21 36 42 92t25 90q1 9 -2.5 32t0.5 28q4 13 22 30.5t22 22.5q19 26 42.5 84.5t27.5 96.5q1 8 -3 25.5t-2 26.5q2 8 9 18t18 23t17 21q8 12 16.5 30.5t15 35t16 36t19.5 32t26.5 23.5t36 11.5t47.5 -5.5l-1 -3q38 9 51 9h761q74 0 114 -56t18 -130l-274 -906 q-36 -119 -71.5 -153.5t-128.5 -34.5h-869q-27 0 -38 -15q-11 -16 -1 -43q24 -70 144 -70h923q29 0 56 15.5t35 41.5l300 987q7 22 5 57q38 -15 59 -43zM575 1056q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5 t-16.5 -22.5zM492 800q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5t-16.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289q0 34 19.5 62t52.5 41q21 9 44 9h1048z" />
-<glyph unicode="" horiz-adv-x="1664" d="M384 0h896v256h-896v-256zM384 640h896v384h-160q-40 0 -68 28t-28 68v160h-640v-640zM1536 576q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 576v-416q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-160q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68 v160h-224q-13 0 -22.5 9.5t-9.5 22.5v416q0 79 56.5 135.5t135.5 56.5h64v544q0 40 28 68t68 28h672q40 0 88 -20t76 -48l152 -152q28 -28 48 -76t20 -88v-256h64q79 0 135.5 -56.5t56.5 -135.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M960 864q119 0 203.5 -84.5t84.5 -203.5t-84.5 -203.5t-203.5 -84.5t-203.5 84.5t-84.5 203.5t84.5 203.5t203.5 84.5zM1664 1280q106 0 181 -75t75 -181v-896q0 -106 -75 -181t-181 -75h-1408q-106 0 -181 75t-75 181v896q0 106 75 181t181 75h224l51 136 q19 49 69.5 84.5t103.5 35.5h512q53 0 103.5 -35.5t69.5 -84.5l51 -136h224zM960 128q185 0 316.5 131.5t131.5 316.5t-131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M725 977l-170 -450q73 -1 153.5 -2t119 -1.5t52.5 -0.5l29 2q-32 95 -92 241q-53 132 -92 211zM21 -128h-21l2 79q22 7 80 18q89 16 110 31q20 16 48 68l237 616l280 724h75h53l11 -21l205 -480q103 -242 124 -297q39 -102 96 -235q26 -58 65 -164q24 -67 65 -149 q22 -49 35 -57q22 -19 69 -23q47 -6 103 -27q6 -39 6 -57q0 -14 -1 -26q-80 0 -192 8q-93 8 -189 8q-79 0 -135 -2l-200 -11l-58 -2q0 45 4 78l131 28q56 13 68 23q12 12 12 27t-6 32l-47 114l-92 228l-450 2q-29 -65 -104 -274q-23 -64 -23 -84q0 -31 17 -43 q26 -21 103 -32q3 0 13.5 -2t30 -5t40.5 -6q1 -28 1 -58q0 -17 -2 -27q-66 0 -349 20l-48 -8q-81 -14 -167 -14z" />
-<glyph unicode="" horiz-adv-x="1408" d="M555 15q76 -32 140 -32q131 0 216 41t122 113q38 70 38 181q0 114 -41 180q-58 94 -141 126q-80 32 -247 32q-74 0 -101 -10v-144l-1 -173l3 -270q0 -15 12 -44zM541 761q43 -7 109 -7q175 0 264 65t89 224q0 112 -85 187q-84 75 -255 75q-52 0 -130 -13q0 -44 2 -77 q7 -122 6 -279l-1 -98q0 -43 1 -77zM0 -128l2 94q45 9 68 12q77 12 123 31q17 27 21 51q9 66 9 194l-2 497q-5 256 -9 404q-1 87 -11 109q-1 4 -12 12q-18 12 -69 15q-30 2 -114 13l-4 83l260 6l380 13l45 1q5 0 14 0.5t14 0.5q1 0 21.5 -0.5t40.5 -0.5h74q88 0 191 -27 q43 -13 96 -39q57 -29 102 -76q44 -47 65 -104t21 -122q0 -70 -32 -128t-95 -105q-26 -20 -150 -77q177 -41 267 -146q92 -106 92 -236q0 -76 -29 -161q-21 -62 -71 -117q-66 -72 -140 -108q-73 -36 -203 -60q-82 -15 -198 -11l-197 4q-84 2 -298 -11q-33 -3 -272 -11z" />
-<glyph unicode="" horiz-adv-x="1024" d="M0 -126l17 85q4 1 77 20q76 19 116 39q29 37 41 101l27 139l56 268l12 64q8 44 17 84.5t16 67t12.5 46.5t9 30.5t3.5 11.5l29 157l16 63l22 135l8 50v38q-41 22 -144 28q-28 2 -38 4l19 103l317 -14q39 -2 73 -2q66 0 214 9q33 2 68 4.5t36 2.5q-2 -19 -6 -38 q-7 -29 -13 -51q-55 -19 -109 -31q-64 -16 -101 -31q-12 -31 -24 -88q-9 -44 -13 -82q-44 -199 -66 -306l-61 -311l-38 -158l-43 -235l-12 -45q-2 -7 1 -27q64 -15 119 -21q36 -5 66 -10q-1 -29 -7 -58q-7 -31 -9 -41q-18 0 -23 -1q-24 -2 -42 -2q-9 0 -28 3q-19 4 -145 17 l-198 2q-41 1 -174 -11q-74 -7 -98 -9z" />
-<glyph unicode="" horiz-adv-x="1792" d="M81 1407l54 -27q20 -5 211 -5h130l19 3l115 1l215 -1h293l34 -2q14 -1 28 7t21 16l7 8l42 1q15 0 28 -1v-104.5t1 -131.5l1 -100l-1 -58q0 -32 -4 -51q-39 -15 -68 -18q-25 43 -54 128q-8 24 -15.5 62.5t-11.5 65.5t-6 29q-13 15 -27 19q-7 2 -42.5 2t-103.5 -1t-111 -1 q-34 0 -67 -5q-10 -97 -8 -136l1 -152v-332l3 -359l-1 -147q-1 -46 11 -85q49 -25 89 -32q2 0 18 -5t44 -13t43 -12q30 -8 50 -18q5 -45 5 -50q0 -10 -3 -29q-14 -1 -34 -1q-110 0 -187 10q-72 8 -238 8q-88 0 -233 -14q-48 -4 -70 -4q-2 22 -2 26l-1 26v9q21 33 79 49 q139 38 159 50q9 21 12 56q8 192 6 433l-5 428q-1 62 -0.5 118.5t0.5 102.5t-2 57t-6 15q-6 5 -14 6q-38 6 -148 6q-43 0 -100 -13.5t-73 -24.5q-13 -9 -22 -33t-22 -75t-24 -84q-6 -19 -19.5 -32t-20.5 -13q-44 27 -56 44v297v86zM1744 128q33 0 42 -18.5t-11 -44.5 l-126 -162q-20 -26 -49 -26t-49 26l-126 162q-20 26 -11 44.5t42 18.5h80v1024h-80q-33 0 -42 18.5t11 44.5l126 162q20 26 49 26t49 -26l126 -162q20 -26 11 -44.5t-42 -18.5h-80v-1024h80z" />
-<glyph unicode="" d="M81 1407l54 -27q20 -5 211 -5h130l19 3l115 1l446 -1h318l34 -2q14 -1 28 7t21 16l7 8l42 1q15 0 28 -1v-104.5t1 -131.5l1 -100l-1 -58q0 -32 -4 -51q-39 -15 -68 -18q-25 43 -54 128q-8 24 -15.5 62.5t-11.5 65.5t-6 29q-13 15 -27 19q-7 2 -58.5 2t-138.5 -1t-128 -1 q-94 0 -127 -5q-10 -97 -8 -136l1 -152v52l3 -359l-1 -147q-1 -46 11 -85q49 -25 89 -32q2 0 18 -5t44 -13t43 -12q30 -8 50 -18q5 -45 5 -50q0 -10 -3 -29q-14 -1 -34 -1q-110 0 -187 10q-72 8 -238 8q-82 0 -233 -13q-45 -5 -70 -5q-2 22 -2 26l-1 26v9q21 33 79 49 q139 38 159 50q9 21 12 56q6 137 6 433l-5 44q0 265 -2 278q-2 11 -6 15q-6 5 -14 6q-38 6 -148 6q-50 0 -168.5 -14t-132.5 -24q-13 -9 -22 -33t-22 -75t-24 -84q-6 -19 -19.5 -32t-20.5 -13q-44 27 -56 44v297v86zM1505 113q26 -20 26 -49t-26 -49l-162 -126 q-26 -20 -44.5 -11t-18.5 42v80h-1024v-80q0 -33 -18.5 -42t-44.5 11l-162 126q-26 20 -26 49t26 49l162 126q26 20 44.5 11t18.5 -42v-80h1024v80q0 33 18.5 42t44.5 -11z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h896q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45t-45 -19 h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h640q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M256 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM256 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5 t9.5 -22.5zM256 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344 q13 0 22.5 -9.5t9.5 -22.5zM256 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192 q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M384 992v-576q0 -13 -9.5 -22.5t-22.5 -9.5q-14 0 -23 9l-288 288q-9 9 -9 23t9 23l288 288q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M352 704q0 -14 -9 -23l-288 -288q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v576q0 13 9.5 22.5t22.5 9.5q14 0 23 -9l288 -288q9 -9 9 -23zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 1184v-1088q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-403 403v-166q0 -119 -84.5 -203.5t-203.5 -84.5h-704q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h704q119 0 203.5 -84.5t84.5 -203.5v-165l403 402q18 19 45 19q12 0 25 -5 q39 -17 39 -59z" />
-<glyph unicode="" horiz-adv-x="1920" d="M640 960q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1664 576v-448h-1408v192l320 320l160 -160l512 512zM1760 1280h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-1216q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5v1216 q0 13 -9.5 22.5t-22.5 9.5zM1920 1248v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" />
-<glyph unicode="" d="M363 0l91 91l-235 235l-91 -91v-107h128v-128h107zM886 928q0 22 -22 22q-10 0 -17 -7l-542 -542q-7 -7 -7 -17q0 -22 22 -22q10 0 17 7l542 542q7 7 7 17zM832 1120l416 -416l-832 -832h-416v416zM1515 1024q0 -53 -37 -90l-166 -166l-416 416l166 165q36 38 90 38 q53 0 91 -38l235 -234q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1024" d="M768 896q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1024 896q0 -109 -33 -179l-364 -774q-16 -33 -47.5 -52t-67.5 -19t-67.5 19t-46.5 52l-365 774q-33 70 -33 179q0 212 150 362t362 150t362 -150t150 -362z" />
-<glyph unicode="" d="M768 96v1088q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M512 384q0 36 -20 69q-1 1 -15.5 22.5t-25.5 38t-25 44t-21 50.5q-4 16 -21 16t-21 -16q-7 -23 -21 -50.5t-25 -44t-25.5 -38t-15.5 -22.5q-20 -33 -20 -69q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 512q0 -212 -150 -362t-362 -150t-362 150t-150 362 q0 145 81 275q6 9 62.5 90.5t101 151t99.5 178t83 201.5q9 30 34 47t51 17t51.5 -17t33.5 -47q28 -93 83 -201.5t99.5 -178t101 -151t62.5 -90.5q81 -127 81 -275z" />
-<glyph unicode="" horiz-adv-x="1792" d="M888 352l116 116l-152 152l-116 -116v-56h96v-96h56zM1328 1072q-16 16 -33 -1l-350 -350q-17 -17 -1 -33t33 1l350 350q17 17 1 33zM1408 478v-190q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-14 -14 -32 -8q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v126q0 13 9 22l64 64q15 15 35 7t20 -29zM1312 1216l288 -288l-672 -672h-288v288zM1756 1084l-92 -92 l-288 288l92 92q28 28 68 28t68 -28l152 -152q28 -28 28 -68t-28 -68z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1408 547v-259q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h255v0q13 0 22.5 -9.5t9.5 -22.5q0 -27 -26 -32q-77 -26 -133 -60q-10 -4 -16 -4h-112q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832 q66 0 113 47t47 113v214q0 19 18 29q28 13 54 37q16 16 35 8q21 -9 21 -29zM1645 1043l-384 -384q-18 -19 -45 -19q-12 0 -25 5q-39 17 -39 59v192h-160q-323 0 -438 -131q-119 -137 -74 -473q3 -23 -20 -34q-8 -2 -12 -2q-16 0 -26 13q-10 14 -21 31t-39.5 68.5t-49.5 99.5 t-38.5 114t-17.5 122q0 49 3.5 91t14 90t28 88t47 81.5t68.5 74t94.5 61.5t124.5 48.5t159.5 30.5t196.5 11h160v192q0 42 39 59q13 5 25 5q26 0 45 -19l384 -384q19 -19 19 -45t-19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1408 606v-318q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-10 -10 -23 -10q-3 0 -9 2q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832 q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v254q0 13 9 22l64 64q10 10 23 10q6 0 12 -3q20 -8 20 -29zM1639 1095l-814 -814q-24 -24 -57 -24t-57 24l-430 430q-24 24 -24 57t24 57l110 110q24 24 57 24t57 -24l263 -263l647 647q24 24 57 24t57 -24l110 -110 q24 -24 24 -57t-24 -57z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-384v-384h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v384h-384v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45 t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h384v384h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45t-19 -45t-45 -19h-128v-384h384v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1024" d="M979 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1747 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19l710 710 q19 19 32 13t13 -32v-710q4 11 13 19z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1619 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-8 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-19 19 -19 45t19 45l710 710q19 19 32 13t13 -32v-710q5 11 13 19z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1384 609l-1328 -738q-23 -13 -39.5 -3t-16.5 36v1472q0 26 16.5 36t39.5 -3l1328 -738q23 -13 23 -31t-23 -31z" />
-<glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45zM640 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45z" />
-<glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q19 -19 19 -45t-19 -45l-710 -710q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" />
-<glyph unicode="" horiz-adv-x="1792" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19l-710 -710 q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" />
-<glyph unicode="" horiz-adv-x="1024" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19z" />
-<glyph unicode="" horiz-adv-x="1538" d="M14 557l710 710q19 19 45 19t45 -19l710 -710q19 -19 13 -32t-32 -13h-1472q-26 0 -32 13t13 32zM1473 0h-1408q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1408q26 0 45 -19t19 -45v-256q0 -26 -19 -45t-45 -19z" />
-<glyph unicode="" horiz-adv-x="1152" d="M742 -37l-652 651q-37 37 -37 90.5t37 90.5l652 651q37 37 90.5 37t90.5 -37l75 -75q37 -37 37 -90.5t-37 -90.5l-486 -486l486 -485q37 -38 37 -91t-37 -90l-75 -75q-37 -37 -90.5 -37t-90.5 37z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1099 704q0 -52 -37 -91l-652 -651q-37 -37 -90 -37t-90 37l-76 75q-37 39 -37 91q0 53 37 90l486 486l-486 485q-37 39 -37 91q0 53 37 90l76 75q36 38 90 38t90 -38l652 -651q37 -37 37 -90z" />
-<glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-256v256q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-256h-256q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h256v-256q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v256h256q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5 t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5 t103 -385.5z" />
-<glyph unicode="" d="M1149 414q0 26 -19 45l-181 181l181 181q19 19 19 45q0 27 -19 46l-90 90q-19 19 -46 19q-26 0 -45 -19l-181 -181l-181 181q-19 19 -45 19q-27 0 -46 -19l-90 -90q-19 -19 -19 -46q0 -26 19 -45l181 -181l-181 -181q-19 -19 -19 -45q0 -27 19 -46l90 -90q19 -19 46 -19 q26 0 45 19l181 181l181 -181q19 -19 45 -19q27 0 46 19l90 90q19 19 19 46zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1284 802q0 28 -18 46l-91 90q-19 19 -45 19t-45 -19l-408 -407l-226 226q-19 19 -45 19t-45 -19l-91 -90q-18 -18 -18 -46q0 -27 18 -45l362 -362q19 -19 45 -19q27 0 46 19l543 543q18 18 18 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M896 160v192q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h192q14 0 23 9t9 23zM1152 832q0 88 -55.5 163t-138.5 116t-170 41q-243 0 -371 -213q-15 -24 8 -42l132 -100q7 -6 19 -6q16 0 25 12q53 68 86 92q34 24 86 24q48 0 85.5 -26t37.5 -59 q0 -38 -20 -61t-68 -45q-63 -28 -115.5 -86.5t-52.5 -125.5v-36q0 -14 9 -23t23 -9h192q14 0 23 9t9 23q0 19 21.5 49.5t54.5 49.5q32 18 49 28.5t46 35t44.5 48t28 60.5t12.5 81zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1024 160v160q0 14 -9 23t-23 9h-96v512q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h96v-320h-96q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h448q14 0 23 9t9 23zM896 1056v160q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23 t23 -9h192q14 0 23 9t9 23zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1197 512h-109q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h109q-32 108 -112.5 188.5t-188.5 112.5v-109q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v109q-108 -32 -188.5 -112.5t-112.5 -188.5h109q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-109 q32 -108 112.5 -188.5t188.5 -112.5v109q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-109q108 32 188.5 112.5t112.5 188.5zM1536 704v-128q0 -26 -19 -45t-45 -19h-143q-37 -161 -154.5 -278.5t-278.5 -154.5v-143q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v143 q-161 37 -278.5 154.5t-154.5 278.5h-143q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h143q37 161 154.5 278.5t278.5 154.5v143q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-143q161 -37 278.5 -154.5t154.5 -278.5h143q26 0 45 -19t19 -45z" />
-<glyph unicode="" d="M1097 457l-146 -146q-10 -10 -23 -10t-23 10l-137 137l-137 -137q-10 -10 -23 -10t-23 10l-146 146q-10 10 -10 23t10 23l137 137l-137 137q-10 10 -10 23t10 23l146 146q10 10 23 10t23 -10l137 -137l137 137q10 10 23 10t23 -10l146 -146q10 -10 10 -23t-10 -23 l-137 -137l137 -137q10 -10 10 -23t-10 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5 t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1171 723l-422 -422q-19 -19 -45 -19t-45 19l-294 294q-19 19 -19 45t19 45l102 102q19 19 45 19t45 -19l147 -147l275 275q19 19 45 19t45 -19l102 -102q19 -19 19 -45t-19 -45zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198 t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1312 643q0 161 -87 295l-754 -753q137 -89 297 -89q111 0 211.5 43.5t173.5 116.5t116 174.5t43 212.5zM313 344l755 754q-135 91 -300 91q-148 0 -273 -73t-198 -199t-73 -274q0 -162 89 -299zM1536 643q0 -157 -61 -300t-163.5 -246t-245 -164t-298.5 -61t-298.5 61 t-245 164t-163.5 246t-61 300t61 299.5t163.5 245.5t245 164t298.5 61t298.5 -61t245 -164t163.5 -245.5t61 -299.5z" />
-<glyph unicode="" d="M1536 640v-128q0 -53 -32.5 -90.5t-84.5 -37.5h-704l293 -294q38 -36 38 -90t-38 -90l-75 -76q-37 -37 -90 -37q-52 0 -91 37l-651 652q-37 37 -37 90q0 52 37 91l651 650q38 38 91 38q52 0 90 -38l75 -74q38 -38 38 -91t-38 -91l-293 -293h704q52 0 84.5 -37.5 t32.5 -90.5z" />
-<glyph unicode="" d="M1472 576q0 -54 -37 -91l-651 -651q-39 -37 -91 -37q-51 0 -90 37l-75 75q-38 38 -38 91t38 91l293 293h-704q-52 0 -84.5 37.5t-32.5 90.5v128q0 53 32.5 90.5t84.5 37.5h704l-293 294q-38 36 -38 90t38 90l75 75q38 38 90 38q53 0 91 -38l651 -651q37 -35 37 -90z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1611 565q0 -51 -37 -90l-75 -75q-38 -38 -91 -38q-54 0 -90 38l-294 293v-704q0 -52 -37.5 -84.5t-90.5 -32.5h-128q-53 0 -90.5 32.5t-37.5 84.5v704l-294 -293q-36 -38 -90 -38t-90 38l-75 75q-38 38 -38 90q0 53 38 91l651 651q35 37 90 37q54 0 91 -37l651 -651 q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1611 704q0 -53 -37 -90l-651 -652q-39 -37 -91 -37q-53 0 -90 37l-651 652q-38 36 -38 90q0 53 38 91l74 75q39 37 91 37q53 0 90 -37l294 -294v704q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-704l294 294q37 37 90 37q52 0 91 -37l75 -75q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 896q0 -26 -19 -45l-512 -512q-19 -19 -45 -19t-45 19t-19 45v256h-224q-98 0 -175.5 -6t-154 -21.5t-133 -42.5t-105.5 -69.5t-80 -101t-48.5 -138.5t-17.5 -181q0 -55 5 -123q0 -6 2.5 -23.5t2.5 -26.5q0 -15 -8.5 -25t-23.5 -10q-16 0 -28 17q-7 9 -13 22 t-13.5 30t-10.5 24q-127 285 -127 451q0 199 53 333q162 403 875 403h224v256q0 26 19 45t45 19t45 -19l512 -512q19 -19 19 -45z" />
-<glyph unicode="" d="M755 480q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23zM1536 1344v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332 q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45z" />
-<glyph unicode="" d="M768 576v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45zM1523 1248q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45 t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-416v-416q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v416h-416q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h416v416q0 40 28 68t68 28h192q40 0 68 -28t28 -68v-416h416q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-1216q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h1216q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1482 486q46 -26 59.5 -77.5t-12.5 -97.5l-64 -110q-26 -46 -77.5 -59.5t-97.5 12.5l-266 153v-307q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v307l-266 -153q-46 -26 -97.5 -12.5t-77.5 59.5l-64 110q-26 46 -12.5 97.5t59.5 77.5l266 154l-266 154 q-46 26 -59.5 77.5t12.5 97.5l64 110q26 46 77.5 59.5t97.5 -12.5l266 -153v307q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-307l266 153q46 26 97.5 12.5t77.5 -59.5l64 -110q26 -46 12.5 -97.5t-59.5 -77.5l-266 -154z" />
-<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM896 161v190q0 14 -9 23.5t-22 9.5h-192q-13 0 -23 -10t-10 -23v-190q0 -13 10 -23t23 -10h192 q13 0 22 9.5t9 23.5zM894 505l18 621q0 12 -10 18q-10 8 -24 8h-220q-14 0 -24 -8q-10 -6 -10 -18l17 -621q0 -10 10 -17.5t24 -7.5h185q14 0 23.5 7.5t10.5 17.5z" />
-<glyph unicode="" d="M928 180v56v468v192h-320v-192v-468v-56q0 -25 18 -38.5t46 -13.5h192q28 0 46 13.5t18 38.5zM472 1024h195l-126 161q-26 31 -69 31q-40 0 -68 -28t-28 -68t28 -68t68 -28zM1160 1120q0 40 -28 68t-68 28q-43 0 -69 -31l-125 -161h194q40 0 68 28t28 68zM1536 864v-320 q0 -14 -9 -23t-23 -9h-96v-416q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v416h-96q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h440q-93 0 -158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5q107 0 168 -77l128 -165l128 165q61 77 168 77q93 0 158.5 -65.5t65.5 -158.5 t-65.5 -158.5t-158.5 -65.5h440q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1280 832q0 26 -19 45t-45 19q-172 0 -318 -49.5t-259.5 -134t-235.5 -219.5q-19 -21 -19 -45q0 -26 19 -45t45 -19q24 0 45 19q27 24 74 71t67 66q137 124 268.5 176t313.5 52q26 0 45 19t19 45zM1792 1030q0 -95 -20 -193q-46 -224 -184.5 -383t-357.5 -268 q-214 -108 -438 -108q-148 0 -286 47q-15 5 -88 42t-96 37q-16 0 -39.5 -32t-45 -70t-52.5 -70t-60 -32q-30 0 -51 11t-31 24t-27 42q-2 4 -6 11t-5.5 10t-3 9.5t-1.5 13.5q0 35 31 73.5t68 65.5t68 56t31 48q0 4 -14 38t-16 44q-9 51 -9 104q0 115 43.5 220t119 184.5 t170.5 139t204 95.5q55 18 145 25.5t179.5 9t178.5 6t163.5 24t113.5 56.5l29.5 29.5t29.5 28t27 20t36.5 16t43.5 4.5q39 0 70.5 -46t47.5 -112t24 -124t8 -96z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1408 -160v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1152 896q0 -78 -24.5 -144t-64 -112.5t-87.5 -88t-96 -77.5t-87.5 -72t-64 -81.5t-24.5 -96.5q0 -96 67 -224l-4 1l1 -1 q-90 41 -160 83t-138.5 100t-113.5 122.5t-72.5 150.5t-27.5 184q0 78 24.5 144t64 112.5t87.5 88t96 77.5t87.5 72t64 81.5t24.5 96.5q0 94 -66 224l3 -1l-1 1q90 -41 160 -83t138.5 -100t113.5 -122.5t72.5 -150.5t27.5 -184z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1664 576q-152 236 -381 353q61 -104 61 -225q0 -185 -131.5 -316.5t-316.5 -131.5t-316.5 131.5t-131.5 316.5q0 121 61 225q-229 -117 -381 -353q133 -205 333.5 -326.5t434.5 -121.5t434.5 121.5t333.5 326.5zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5 t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1792 576q0 -34 -20 -69q-140 -230 -376.5 -368.5t-499.5 -138.5t-499.5 139t-376.5 368q-20 35 -20 69t20 69q140 229 376.5 368t499.5 139t499.5 -139t376.5 -368q20 -35 20 -69z" />
-<glyph unicode="" horiz-adv-x="1792" d="M555 201l78 141q-87 63 -136 159t-49 203q0 121 61 225q-229 -117 -381 -353q167 -258 427 -375zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1307 1151q0 -7 -1 -9 q-105 -188 -315 -566t-316 -567l-49 -89q-10 -16 -28 -16q-12 0 -134 70q-16 10 -16 28q0 12 44 87q-143 65 -263.5 173t-208.5 245q-20 31 -20 69t20 69q153 235 380 371t496 136q89 0 180 -17l54 97q10 16 28 16q5 0 18 -6t31 -15.5t33 -18.5t31.5 -18.5t19.5 -11.5 q16 -10 16 -27zM1344 704q0 -139 -79 -253.5t-209 -164.5l280 502q8 -45 8 -84zM1792 576q0 -35 -20 -69q-39 -64 -109 -145q-150 -172 -347.5 -267t-419.5 -95l74 132q212 18 392.5 137t301.5 307q-115 179 -282 294l63 112q95 -64 182.5 -153t144.5 -184q20 -34 20 -69z " />
-<glyph unicode="" horiz-adv-x="1792" d="M1024 161v190q0 14 -9.5 23.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -23.5v-190q0 -14 9.5 -23.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 23.5zM1022 535l18 459q0 12 -10 19q-13 11 -24 11h-220q-11 0 -24 -11q-10 -7 -10 -21l17 -457q0 -10 10 -16.5t24 -6.5h185 q14 0 23.5 6.5t10.5 16.5zM1008 1469l768 -1408q35 -63 -2 -126q-17 -29 -46.5 -46t-63.5 -17h-1536q-34 0 -63.5 17t-46.5 46q-37 63 -2 126l768 1408q17 31 47 49t65 18t65 -18t47 -49z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1376 1376q44 -52 12 -148t-108 -172l-161 -161l160 -696q5 -19 -12 -33l-128 -96q-7 -6 -19 -6q-4 0 -7 1q-15 3 -21 16l-279 508l-259 -259l53 -194q5 -17 -8 -31l-96 -96q-9 -9 -23 -9h-2q-15 2 -24 13l-189 252l-252 189q-11 7 -13 23q-1 13 9 25l96 97q9 9 23 9 q6 0 8 -1l194 -53l259 259l-508 279q-14 8 -17 24q-2 16 9 27l128 128q14 13 30 8l665 -159l160 160q76 76 172 108t148 -12z" />
-<glyph unicode="" horiz-adv-x="1664" d="M128 -128h288v288h-288v-288zM480 -128h320v288h-320v-288zM128 224h288v320h-288v-320zM480 224h320v320h-320v-320zM128 608h288v288h-288v-288zM864 -128h320v288h-320v-288zM480 608h320v288h-320v-288zM1248 -128h288v288h-288v-288zM864 224h320v320h-320v-320z M512 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1248 224h288v320h-288v-320zM864 608h320v288h-320v-288zM1248 608h288v288h-288v-288zM1280 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64 q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1664 1152v-1280q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47 h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" />
-<glyph unicode="" horiz-adv-x="1792" d="M666 1055q-60 -92 -137 -273q-22 45 -37 72.5t-40.5 63.5t-51 56.5t-63 35t-81.5 14.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q250 0 410 -225zM1792 256q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v192q-32 0 -85 -0.5t-81 -1t-73 1 t-71 5t-64 10.5t-63 18.5t-58 28.5t-59 40t-55 53.5t-56 69.5q59 93 136 273q22 -45 37 -72.5t40.5 -63.5t51 -56.5t63 -35t81.5 -14.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23zM1792 1152q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5 v192h-256q-48 0 -87 -15t-69 -45t-51 -61.5t-45 -77.5q-32 -62 -78 -171q-29 -66 -49.5 -111t-54 -105t-64 -100t-74 -83t-90 -68.5t-106.5 -42t-128 -16.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q48 0 87 15t69 45t51 61.5t45 77.5q32 62 78 171q29 66 49.5 111 t54 105t64 100t74 83t90 68.5t106.5 42t128 16.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22q-17 -2 -30.5 9t-17.5 29v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281 q0 130 71 248.5t191 204.5t286 136.5t348 50.5q244 0 450 -85.5t326 -233t120 -321.5z" />
-<glyph unicode="" d="M1536 704v-128q0 -201 -98.5 -362t-274 -251.5t-395.5 -90.5t-395.5 90.5t-274 251.5t-98.5 362v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-128q0 -52 23.5 -90t53.5 -57t71 -30t64 -13t44 -2t44 2t64 13t71 30t53.5 57t23.5 90v128q0 26 19 45t45 19h384 q26 0 45 -19t19 -45zM512 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45zM1536 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1611 320q0 -53 -37 -90l-75 -75q-38 -38 -91 -38q-54 0 -90 38l-486 485l-486 -485q-36 -38 -90 -38t-90 38l-75 75q-38 36 -38 90q0 53 38 91l651 651q37 37 90 37q52 0 91 -37l650 -651q38 -38 38 -91z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1611 832q0 -53 -37 -90l-651 -651q-38 -38 -91 -38q-54 0 -90 38l-651 651q-38 36 -38 90q0 53 38 91l74 75q39 37 91 37q53 0 90 -37l486 -486l486 486q37 37 90 37q52 0 91 -37l75 -75q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1280 32q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-8 0 -13.5 2t-9 7t-5.5 8t-3 11.5t-1 11.5v13v11v160v416h-192q-26 0 -45 19t-19 45q0 24 15 41l320 384q19 22 49 22t49 -22l320 -384q15 -17 15 -41q0 -26 -19 -45t-45 -19h-192v-384h576q16 0 25 -11l160 -192q7 -11 7 -21 zM1920 448q0 -24 -15 -41l-320 -384q-20 -23 -49 -23t-49 23l-320 384q-15 17 -15 41q0 26 19 45t45 19h192v384h-576q-16 0 -25 12l-160 192q-7 9 -7 20q0 13 9.5 22.5t22.5 9.5h960q8 0 13.5 -2t9 -7t5.5 -8t3 -11.5t1 -11.5v-13v-11v-160v-416h192q26 0 45 -19t19 -45z " />
-<glyph unicode="" horiz-adv-x="1664" d="M640 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1536 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1664 1088v-512q0 -24 -16 -42.5t-41 -21.5 l-1044 -122q1 -7 4.5 -21.5t6 -26.5t2.5 -22q0 -16 -24 -64h920q26 0 45 -19t19 -45t-19 -45t-45 -19h-1024q-26 0 -45 19t-19 45q0 14 11 39.5t29.5 59.5t20.5 38l-177 823h-204q-26 0 -45 19t-19 45t19 45t45 19h256q16 0 28.5 -6.5t20 -15.5t13 -24.5t7.5 -26.5 t5.5 -29.5t4.5 -25.5h1201q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1879 584q0 -31 -31 -66l-336 -396q-43 -51 -120.5 -86.5t-143.5 -35.5h-1088q-34 0 -60.5 13t-26.5 43q0 31 31 66l336 396q43 51 120.5 86.5t143.5 35.5h1088q34 0 60.5 -13t26.5 -43zM1536 928v-160h-832q-94 0 -197 -47.5t-164 -119.5l-337 -396l-5 -6q0 4 -0.5 12.5 t-0.5 12.5v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158z" />
-<glyph unicode="" horiz-adv-x="768" d="M704 1216q0 -26 -19 -45t-45 -19h-128v-1024h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v1024h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-1024v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h1024v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1920" d="M512 512v-384h-256v384h256zM896 1024v-896h-256v896h256zM1280 768v-640h-256v640h256zM1664 1152v-1024h-256v1024h256zM1792 32v1216q0 13 -9.5 22.5t-22.5 9.5h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-1216q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5z M1920 1248v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" />
-<glyph unicode="" d="M1280 926q-56 -25 -121 -34q68 40 93 117q-65 -38 -134 -51q-61 66 -153 66q-87 0 -148.5 -61.5t-61.5 -148.5q0 -29 5 -48q-129 7 -242 65t-192 155q-29 -50 -29 -106q0 -114 91 -175q-47 1 -100 26v-2q0 -75 50 -133.5t123 -72.5q-29 -8 -51 -8q-13 0 -39 4 q21 -63 74.5 -104t121.5 -42q-116 -90 -261 -90q-26 0 -50 3q148 -94 322 -94q112 0 210 35.5t168 95t120.5 137t75 162t24.5 168.5q0 18 -1 27q63 45 105 109zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5 t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1307 618l23 219h-198v109q0 49 15.5 68.5t71.5 19.5h110v219h-175q-152 0 -218 -72t-66 -213v-131h-131v-219h131v-635h262v635h175zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960 q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M928 704q0 14 -9 23t-23 9q-66 0 -113 -47t-47 -113q0 -14 9 -23t23 -9t23 9t9 23q0 40 28 68t68 28q14 0 23 9t9 23zM1152 574q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM128 0h1536v128h-1536v-128zM1280 574q0 159 -112.5 271.5 t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM256 1216h384v128h-384v-128zM128 1024h1536v118v138h-828l-64 -128h-644v-128zM1792 1280v-1280q0 -53 -37.5 -90.5t-90.5 -37.5h-1536q-53 0 -90.5 37.5t-37.5 90.5v1280 q0 53 37.5 90.5t90.5 37.5h1536q53 0 90.5 -37.5t37.5 -90.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M832 1024q0 80 -56 136t-136 56t-136 -56t-56 -136q0 -42 19 -83q-41 19 -83 19q-80 0 -136 -56t-56 -136t56 -136t136 -56t136 56t56 136q0 42 -19 83q41 -19 83 -19q80 0 136 56t56 136zM1683 320q0 -17 -49 -66t-66 -49q-9 0 -28.5 16t-36.5 33t-38.5 40t-24.5 26 l-96 -96l220 -220q28 -28 28 -68q0 -42 -39 -81t-81 -39q-40 0 -68 28l-671 671q-176 -131 -365 -131q-163 0 -265.5 102.5t-102.5 265.5q0 160 95 313t248 248t313 95q163 0 265.5 -102.5t102.5 -265.5q0 -189 -131 -365l355 -355l96 96q-3 3 -26 24.5t-40 38.5t-33 36.5 t-16 28.5q0 17 49 66t66 49q13 0 23 -10q6 -6 46 -44.5t82 -79.5t86.5 -86t73 -78t28.5 -41z" />
-<glyph unicode="" horiz-adv-x="1920" d="M896 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1664 128q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 1152q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1280 731v-185q0 -10 -7 -19.5t-16 -10.5l-155 -24q-11 -35 -32 -76q34 -48 90 -115q7 -10 7 -20q0 -12 -7 -19q-23 -30 -82.5 -89.5t-78.5 -59.5q-11 0 -21 7l-115 90q-37 -19 -77 -31q-11 -108 -23 -155q-7 -24 -30 -24h-186q-11 0 -20 7.5t-10 17.5 l-23 153q-34 10 -75 31l-118 -89q-7 -7 -20 -7q-11 0 -21 8q-144 133 -144 160q0 9 7 19q10 14 41 53t47 61q-23 44 -35 82l-152 24q-10 1 -17 9.5t-7 19.5v185q0 10 7 19.5t16 10.5l155 24q11 35 32 76q-34 48 -90 115q-7 11 -7 20q0 12 7 20q22 30 82 89t79 59q11 0 21 -7 l115 -90q34 18 77 32q11 108 23 154q7 24 30 24h186q11 0 20 -7.5t10 -17.5l23 -153q34 -10 75 -31l118 89q8 7 20 7q11 0 21 -8q144 -133 144 -160q0 -9 -7 -19q-12 -16 -42 -54t-45 -60q23 -48 34 -82l152 -23q10 -2 17 -10.5t7 -19.5zM1920 198v-140q0 -16 -149 -31 q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20 t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31zM1920 1222v-140q0 -16 -149 -31q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68 q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70 q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1408 768q0 -139 -94 -257t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224 q0 139 94 257t256.5 186.5t353.5 68.5t353.5 -68.5t256.5 -186.5t94 -257zM1792 512q0 -120 -71 -224.5t-195 -176.5q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7 q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230z" />
-<glyph unicode="" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 768q0 51 -39 89.5t-89 38.5h-352q0 58 48 159.5t48 160.5q0 98 -32 145t-128 47q-26 -26 -38 -85t-30.5 -125.5t-59.5 -109.5q-22 -23 -77 -91q-4 -5 -23 -30t-31.5 -41t-34.5 -42.5 t-40 -44t-38.5 -35.5t-40 -27t-35.5 -9h-32v-640h32q13 0 31.5 -3t33 -6.5t38 -11t35 -11.5t35.5 -12.5t29 -10.5q211 -73 342 -73h121q192 0 192 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5q32 1 53.5 47t21.5 81zM1536 769 q0 -89 -49 -163q9 -33 9 -69q0 -77 -38 -144q3 -21 3 -43q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5h-36h-93q-96 0 -189.5 22.5t-216.5 65.5q-116 40 -138 40h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h274q36 24 137 155q58 75 107 128 q24 25 35.5 85.5t30.5 126.5t62 108q39 37 90 37q84 0 151 -32.5t102 -101.5t35 -186q0 -93 -48 -192h176q104 0 180 -76t76 -179z" />
-<glyph unicode="" d="M256 1088q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 512q0 35 -21.5 81t-53.5 47q15 17 25 47.5t10 55.5q0 69 -53 119q18 32 18 69t-17.5 73.5t-47.5 52.5q5 30 5 56q0 85 -49 126t-136 41h-128q-131 0 -342 -73q-5 -2 -29 -10.5 t-35.5 -12.5t-35 -11.5t-38 -11t-33 -6.5t-31.5 -3h-32v-640h32q16 0 35.5 -9t40 -27t38.5 -35.5t40 -44t34.5 -42.5t31.5 -41t23 -30q55 -68 77 -91q41 -43 59.5 -109.5t30.5 -125.5t38 -85q96 0 128 47t32 145q0 59 -48 160.5t-48 159.5h352q50 0 89 38.5t39 89.5z M1536 511q0 -103 -76 -179t-180 -76h-176q48 -99 48 -192q0 -118 -35 -186q-35 -69 -102 -101.5t-151 -32.5q-51 0 -90 37q-34 33 -54 82t-25.5 90.5t-17.5 84.5t-31 64q-48 50 -107 127q-101 131 -137 155h-274q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5 h288q22 0 138 40q128 44 223 66t200 22h112q140 0 226.5 -79t85.5 -216v-5q60 -77 60 -178q0 -22 -3 -43q38 -67 38 -144q0 -36 -9 -69q49 -74 49 -163z" />
-<glyph unicode="" horiz-adv-x="896" d="M832 1504v-1339l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1664 940q0 81 -21.5 143t-55 98.5t-81.5 59.5t-94 31t-98 8t-112 -25.5t-110.5 -64t-86.5 -72t-60 -61.5q-18 -22 -49 -22t-49 22q-24 28 -60 61.5t-86.5 72t-110.5 64t-112 25.5t-98 -8t-94 -31t-81.5 -59.5t-55 -98.5t-21.5 -143q0 -168 187 -355l581 -560l580 559 q188 188 188 356zM1792 940q0 -221 -229 -450l-623 -600q-18 -18 -44 -18t-44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5 q224 0 351 -124t127 -344z" />
-<glyph unicode="" horiz-adv-x="1664" d="M640 96q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h320q13 0 22.5 -9.5t9.5 -22.5q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-66 0 -113 -47t-47 -113v-704 q0 -66 47 -113t113 -47h288h11h13t11.5 -1t11.5 -3t8 -5.5t7 -9t2 -13.5zM1568 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45z" />
-<glyph unicode="" d="M237 122h231v694h-231v-694zM483 1030q-1 52 -36 86t-93 34t-94.5 -34t-36.5 -86q0 -51 35.5 -85.5t92.5 -34.5h1q59 0 95 34.5t36 85.5zM1068 122h231v398q0 154 -73 233t-193 79q-136 0 -209 -117h2v101h-231q3 -66 0 -694h231v388q0 38 7 56q15 35 45 59.5t74 24.5 q116 0 116 -157v-371zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1152" d="M480 672v448q0 14 -9 23t-23 9t-23 -9t-9 -23v-448q0 -14 9 -23t23 -9t23 9t9 23zM1152 320q0 -26 -19 -45t-45 -19h-429l-51 -483q-2 -12 -10.5 -20.5t-20.5 -8.5h-1q-27 0 -32 27l-76 485h-404q-26 0 -45 19t-19 45q0 123 78.5 221.5t177.5 98.5v512q-52 0 -90 38 t-38 90t38 90t90 38h640q52 0 90 -38t38 -90t-38 -90t-90 -38v-512q99 0 177.5 -98.5t78.5 -221.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1408 608v-320q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h704q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v320 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1792 1472v-512q0 -26 -19 -45t-45 -19t-45 19l-176 176l-652 -652q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l652 652l-176 176q-19 19 -19 45t19 45t45 19h512q26 0 45 -19t19 -45z" />
-<glyph unicode="" d="M1184 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45zM1536 992v-704q0 -119 -84.5 -203.5t-203.5 -84.5h-320q-13 0 -22.5 9.5t-9.5 22.5 q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q66 0 113 47t47 113v704q0 66 -47 113t-113 47h-288h-11h-13t-11.5 1t-11.5 3t-8 5.5t-7 9t-2 13.5q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M458 653q-74 162 -74 371h-256v-96q0 -78 94.5 -162t235.5 -113zM1536 928v96h-256q0 -209 -74 -371q141 29 235.5 113t94.5 162zM1664 1056v-128q0 -71 -41.5 -143t-112 -130t-173 -97.5t-215.5 -44.5q-42 -54 -95 -95q-38 -34 -52.5 -72.5t-14.5 -89.5q0 -54 30.5 -91 t97.5 -37q75 0 133.5 -45.5t58.5 -114.5v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 69 58.5 114.5t133.5 45.5q67 0 97.5 37t30.5 91q0 51 -14.5 89.5t-52.5 72.5q-53 41 -95 95q-113 5 -215.5 44.5t-173 97.5t-112 130t-41.5 143v128q0 40 28 68t68 28h288v96 q0 66 47 113t113 47h576q66 0 113 -47t47 -113v-96h288q40 0 68 -28t28 -68z" />
-<glyph unicode="" d="M394 184q-8 -9 -20 3q-13 11 -4 19q8 9 20 -3q12 -11 4 -19zM352 245q9 -12 0 -19q-8 -6 -17 7t0 18q9 7 17 -6zM291 305q-5 -7 -13 -2q-10 5 -7 12q3 5 13 2q10 -5 7 -12zM322 271q-6 -7 -16 3q-9 11 -2 16q6 6 16 -3q9 -11 2 -16zM451 159q-4 -12 -19 -6q-17 4 -13 15 t19 7q16 -5 13 -16zM514 154q0 -11 -16 -11q-17 -2 -17 11q0 11 16 11q17 2 17 -11zM572 164q2 -10 -14 -14t-18 8t14 15q16 2 18 -9zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-224q-16 0 -24.5 1t-19.5 5t-16 14.5t-5 27.5v239q0 97 -52 142q57 6 102.5 18t94 39 t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103 q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -103t0.5 -68q0 -22 -11 -33.5t-22 -13t-33 -1.5 h-224q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1280 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 288v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h427q21 -56 70.5 -92 t110.5 -36h256q61 0 110.5 36t70.5 92h427q40 0 68 -28t28 -68zM1339 936q-17 -40 -59 -40h-256v-448q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v448h-256q-42 0 -59 40q-17 39 14 69l448 448q18 19 45 19t45 -19l448 -448q31 -30 14 -69z" />
-<glyph unicode="" d="M1407 710q0 44 -7 113.5t-18 96.5q-12 30 -17 44t-9 36.5t-4 48.5q0 23 5 68.5t5 67.5q0 37 -10 55q-4 1 -13 1q-19 0 -58 -4.5t-59 -4.5q-60 0 -176 24t-175 24q-43 0 -94.5 -11.5t-85 -23.5t-89.5 -34q-137 -54 -202 -103q-96 -73 -159.5 -189.5t-88 -236t-24.5 -248.5 q0 -40 12.5 -120t12.5 -121q0 -23 -11 -66.5t-11 -65.5t12 -36.5t34 -14.5q24 0 72.5 11t73.5 11q57 0 169.5 -15.5t169.5 -15.5q181 0 284 36q129 45 235.5 152.5t166 245.5t59.5 275zM1535 712q0 -165 -70 -327.5t-196 -288t-281 -180.5q-124 -44 -326 -44 q-57 0 -170 14.5t-169 14.5q-24 0 -72.5 -14.5t-73.5 -14.5q-73 0 -123.5 55.5t-50.5 128.5q0 24 11 68t11 67q0 40 -12.5 120.5t-12.5 121.5q0 111 18 217.5t54.5 209.5t100.5 194t150 156q78 59 232 120q194 78 316 78q60 0 175.5 -24t173.5 -24q19 0 57 5t58 5 q81 0 118 -50.5t37 -134.5q0 -23 -5 -68t-5 -68q0 -10 1 -18.5t3 -17t4 -13.5t6.5 -16t6.5 -17q16 -40 25 -118.5t9 -136.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1408 296q0 -27 -10 -70.5t-21 -68.5q-21 -50 -122 -106q-94 -51 -186 -51q-27 0 -52.5 3.5t-57.5 12.5t-47.5 14.5t-55.5 20.5t-49 18q-98 35 -175 83q-128 79 -264.5 215.5t-215.5 264.5q-48 77 -83 175q-3 9 -18 49t-20.5 55.5t-14.5 47.5t-12.5 57.5t-3.5 52.5 q0 92 51 186q56 101 106 122q25 11 68.5 21t70.5 10q14 0 21 -3q18 -6 53 -76q11 -19 30 -54t35 -63.5t31 -53.5q3 -4 17.5 -25t21.5 -35.5t7 -28.5q0 -20 -28.5 -50t-62 -55t-62 -53t-28.5 -46q0 -9 5 -22.5t8.5 -20.5t14 -24t11.5 -19q76 -137 174 -235t235 -174 q2 -1 19 -11.5t24 -14t20.5 -8.5t22.5 -5q18 0 46 28.5t53 62t55 62t50 28.5q14 0 28.5 -7t35.5 -21.5t25 -17.5q25 -15 53.5 -31t63.5 -35t54 -30q70 -35 76 -53q3 -7 3 -21z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1120 1280h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v832q0 66 -47 113t-113 47zM1408 1120v-832q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1152 1280h-1024v-1242l423 406l89 85l89 -85l423 -406v1242zM1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289 q0 34 19.5 62t52.5 41q21 9 44 9h1048z" />
-<glyph unicode="" d="M1280 343q0 11 -2 16q-3 8 -38.5 29.5t-88.5 49.5l-53 29q-5 3 -19 13t-25 15t-21 5q-18 0 -47 -32.5t-57 -65.5t-44 -33q-7 0 -16.5 3.5t-15.5 6.5t-17 9.5t-14 8.5q-99 55 -170.5 126.5t-126.5 170.5q-2 3 -8.5 14t-9.5 17t-6.5 15.5t-3.5 16.5q0 13 20.5 33.5t45 38.5 t45 39.5t20.5 36.5q0 10 -5 21t-15 25t-13 19q-3 6 -15 28.5t-25 45.5t-26.5 47.5t-25 40.5t-16.5 18t-16 2q-48 0 -101 -22q-46 -21 -80 -94.5t-34 -130.5q0 -16 2.5 -34t5 -30.5t9 -33t10 -29.5t12.5 -33t11 -30q60 -164 216.5 -320.5t320.5 -216.5q6 -2 30 -11t33 -12.5 t29.5 -10t33 -9t30.5 -5t34 -2.5q57 0 130.5 34t94.5 80q22 53 22 101zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1620 1128q-67 -98 -162 -167q1 -14 1 -42q0 -130 -38 -259.5t-115.5 -248.5t-184.5 -210.5t-258 -146t-323 -54.5q-271 0 -496 145q35 -4 78 -4q225 0 401 138q-105 2 -188 64.5t-114 159.5q33 -5 61 -5q43 0 85 11q-112 23 -185.5 111.5t-73.5 205.5v4q68 -38 146 -41 q-66 44 -105 115t-39 154q0 88 44 163q121 -149 294.5 -238.5t371.5 -99.5q-8 38 -8 74q0 134 94.5 228.5t228.5 94.5q140 0 236 -102q109 21 205 78q-37 -115 -142 -178q93 10 186 50z" />
-<glyph unicode="" horiz-adv-x="768" d="M511 980h257l-30 -284h-227v-824h-341v824h-170v284h170v171q0 182 86 275.5t283 93.5h227v-284h-142q-39 0 -62.5 -6.5t-34 -23.5t-13.5 -34.5t-3 -49.5v-142z" />
-<glyph unicode="" d="M1536 640q0 -251 -146.5 -451.5t-378.5 -277.5q-27 -5 -39.5 7t-12.5 30v211q0 97 -52 142q57 6 102.5 18t94 39t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5 q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23 q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -89t0.5 -54q0 -18 -13 -30t-40 -7q-232 77 -378.5 277.5t-146.5 451.5q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1664 960v-256q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45v256q0 106 -75 181t-181 75t-181 -75t-75 -181v-192h96q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h672v192q0 185 131.5 316.5t316.5 131.5 t316.5 -131.5t131.5 -316.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1760 1408q66 0 113 -47t47 -113v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600zM160 1280q-13 0 -22.5 -9.5t-9.5 -22.5v-224h1664v224q0 13 -9.5 22.5t-22.5 9.5h-1600zM1760 0q13 0 22.5 9.5t9.5 22.5v608h-1664v-608 q0 -13 9.5 -22.5t22.5 -9.5h1600zM256 128v128h256v-128h-256zM640 128v128h384v-128h-384z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM896 69q2 -28 -17 -48q-18 -21 -47 -21h-135q-25 0 -43 16.5t-20 41.5q-22 229 -184.5 391.5t-391.5 184.5q-25 2 -41.5 20t-16.5 43v135q0 29 21 47q17 17 43 17h5q160 -13 306 -80.5 t259 -181.5q114 -113 181.5 -259t80.5 -306zM1408 67q2 -27 -18 -47q-18 -20 -46 -20h-143q-26 0 -44.5 17.5t-19.5 42.5q-12 215 -101 408.5t-231.5 336t-336 231.5t-408.5 102q-25 1 -42.5 19.5t-17.5 43.5v143q0 28 20 46q18 18 44 18h3q262 -13 501.5 -120t425.5 -294 q187 -186 294 -425.5t120 -501.5z" />
-<glyph unicode="" d="M1040 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1296 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1408 160v320q0 13 -9.5 22.5t-22.5 9.5 h-1216q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h1216q13 0 22.5 9.5t9.5 22.5zM178 640h1180l-157 482q-4 13 -16 21.5t-26 8.5h-782q-14 0 -26 -8.5t-16 -21.5zM1536 480v-320q0 -66 -47 -113t-113 -47h-1216q-66 0 -113 47t-47 113v320q0 25 16 75 l197 606q17 53 63 86t101 33h782q55 0 101 -33t63 -86l197 -606q16 -50 16 -75z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1664 896q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5v-384q0 -52 -38 -90t-90 -38q-417 347 -812 380q-58 -19 -91 -66t-31 -100.5t40 -92.5q-20 -33 -23 -65.5t6 -58t33.5 -55t48 -50t61.5 -50.5q-29 -58 -111.5 -83t-168.5 -11.5t-132 55.5q-7 23 -29.5 87.5 t-32 94.5t-23 89t-15 101t3.5 98.5t22 110.5h-122q-66 0 -113 47t-47 113v192q0 66 47 113t113 47h480q435 0 896 384q52 0 90 -38t38 -90v-384zM1536 292v954q-394 -302 -768 -343v-270q377 -42 768 -341z" />
-<glyph unicode="" horiz-adv-x="1664" d="M848 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM183 128h1298q-164 181 -246.5 411.5t-82.5 484.5q0 256 -320 256t-320 -256q0 -254 -82.5 -484.5t-246.5 -411.5zM1664 128q0 -52 -38 -90t-90 -38 h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q190 161 287 397.5t97 498.5q0 165 96 262t264 117q-8 18 -8 37q0 40 28 68t68 28t68 -28t28 -68q0 -19 -8 -37q168 -20 264 -117t96 -262q0 -262 97 -498.5t287 -397.5z" />
-<glyph unicode="" d="M1376 640l138 -135q30 -28 20 -70q-12 -41 -52 -51l-188 -48l53 -186q12 -41 -19 -70q-29 -31 -70 -19l-186 53l-48 -188q-10 -40 -51 -52q-12 -2 -19 -2q-31 0 -51 22l-135 138l-135 -138q-28 -30 -70 -20q-41 11 -51 52l-48 188l-186 -53q-41 -12 -70 19q-31 29 -19 70 l53 186l-188 48q-40 10 -52 51q-10 42 20 70l138 135l-138 135q-30 28 -20 70q12 41 52 51l188 48l-53 186q-12 41 19 70q29 31 70 19l186 -53l48 188q10 41 51 51q41 12 70 -19l135 -139l135 139q29 30 70 19q41 -10 51 -51l48 -188l186 53q41 12 70 -19q31 -29 19 -70 l-53 -186l188 -48q40 -10 52 -51q10 -42 -20 -70z" />
-<glyph unicode="" horiz-adv-x="1792" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 768q0 51 -39 89.5t-89 38.5h-576q0 20 15 48.5t33 55t33 68t15 84.5q0 67 -44.5 97.5t-115.5 30.5q-24 0 -90 -139q-24 -44 -37 -65q-40 -64 -112 -145q-71 -81 -101 -106 q-69 -57 -140 -57h-32v-640h32q72 0 167 -32t193.5 -64t179.5 -32q189 0 189 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5h331q52 0 90 38t38 90zM1792 769q0 -105 -75.5 -181t-180.5 -76h-169q-4 -62 -37 -119q3 -21 3 -43 q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5q-133 0 -322 69q-164 59 -223 59h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h288q10 0 21.5 4.5t23.5 14t22.5 18t24 22.5t20.5 21.5t19 21.5t14 17q65 74 100 129q13 21 33 62t37 72t40.5 63t55 49.5 t69.5 17.5q125 0 206.5 -67t81.5 -189q0 -68 -22 -128h374q104 0 180 -76t76 -179z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1376 128h32v640h-32q-35 0 -67.5 12t-62.5 37t-50 46t-49 54q-2 3 -3.5 4.5t-4 4.5t-4.5 5q-72 81 -112 145q-14 22 -38 68q-1 3 -10.5 22.5t-18.5 36t-20 35.5t-21.5 30.5t-18.5 11.5q-71 0 -115.5 -30.5t-44.5 -97.5q0 -43 15 -84.5t33 -68t33 -55t15 -48.5h-576 q-50 0 -89 -38.5t-39 -89.5q0 -52 38 -90t90 -38h331q-15 -17 -25 -47.5t-10 -55.5q0 -69 53 -119q-18 -32 -18 -69t17.5 -73.5t47.5 -52.5q-4 -24 -4 -56q0 -85 48.5 -126t135.5 -41q84 0 183 32t194 64t167 32zM1664 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45 t45 -19t45 19t19 45zM1792 768v-640q0 -53 -37.5 -90.5t-90.5 -37.5h-288q-59 0 -223 -59q-190 -69 -317 -69q-142 0 -230 77.5t-87 217.5l1 5q-61 76 -61 178q0 22 3 43q-33 57 -37 119h-169q-105 0 -180.5 76t-75.5 181q0 103 76 179t180 76h374q-22 60 -22 128 q0 122 81.5 189t206.5 67q38 0 69.5 -17.5t55 -49.5t40.5 -63t37 -72t33 -62q35 -55 100 -129q2 -3 14 -17t19 -21.5t20.5 -21.5t24 -22.5t22.5 -18t23.5 -14t21.5 -4.5h288q53 0 90.5 -37.5t37.5 -90.5z" />
-<glyph unicode="" d="M1280 -64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 700q0 189 -167 189q-26 0 -56 -5q-16 30 -52.5 47.5t-73.5 17.5t-69 -18q-50 53 -119 53q-25 0 -55.5 -10t-47.5 -25v331q0 52 -38 90t-90 38q-51 0 -89.5 -39t-38.5 -89v-576 q-20 0 -48.5 15t-55 33t-68 33t-84.5 15q-67 0 -97.5 -44.5t-30.5 -115.5q0 -24 139 -90q44 -24 65 -37q64 -40 145 -112q81 -71 106 -101q57 -69 57 -140v-32h640v32q0 72 32 167t64 193.5t32 179.5zM1536 705q0 -133 -69 -322q-59 -164 -59 -223v-288q0 -53 -37.5 -90.5 t-90.5 -37.5h-640q-53 0 -90.5 37.5t-37.5 90.5v288q0 10 -4.5 21.5t-14 23.5t-18 22.5t-22.5 24t-21.5 20.5t-21.5 19t-17 14q-74 65 -129 100q-21 13 -62 33t-72 37t-63 40.5t-49.5 55t-17.5 69.5q0 125 67 206.5t189 81.5q68 0 128 -22v374q0 104 76 180t179 76 q105 0 181 -75.5t76 -180.5v-169q62 -4 119 -37q21 3 43 3q101 0 178 -60q139 1 219.5 -85t80.5 -227z" />
-<glyph unicode="" d="M1408 576q0 84 -32 183t-64 194t-32 167v32h-640v-32q0 -35 -12 -67.5t-37 -62.5t-46 -50t-54 -49q-9 -8 -14 -12q-81 -72 -145 -112q-22 -14 -68 -38q-3 -1 -22.5 -10.5t-36 -18.5t-35.5 -20t-30.5 -21.5t-11.5 -18.5q0 -71 30.5 -115.5t97.5 -44.5q43 0 84.5 15t68 33 t55 33t48.5 15v-576q0 -50 38.5 -89t89.5 -39q52 0 90 38t38 90v331q46 -35 103 -35q69 0 119 53q32 -18 69 -18t73.5 17.5t52.5 47.5q24 -4 56 -4q85 0 126 48.5t41 135.5zM1280 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 580 q0 -142 -77.5 -230t-217.5 -87l-5 1q-76 -61 -178 -61q-22 0 -43 3q-54 -30 -119 -37v-169q0 -105 -76 -180.5t-181 -75.5q-103 0 -179 76t-76 180v374q-54 -22 -128 -22q-121 0 -188.5 81.5t-67.5 206.5q0 38 17.5 69.5t49.5 55t63 40.5t72 37t62 33q55 35 129 100 q3 2 17 14t21.5 19t21.5 20.5t22.5 24t18 22.5t14 23.5t4.5 21.5v288q0 53 37.5 90.5t90.5 37.5h640q53 0 90.5 -37.5t37.5 -90.5v-288q0 -59 59 -223q69 -190 69 -317z" />
-<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-502l189 189q19 19 19 45t-19 45l-91 91q-18 18 -45 18t-45 -18l-362 -362l-91 -91q-18 -18 -18 -45t18 -45l91 -91l362 -362q18 -18 45 -18t45 18l91 91q18 18 18 45t-18 45l-189 189h502q26 0 45 19t19 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1285 640q0 27 -18 45l-91 91l-362 362q-18 18 -45 18t-45 -18l-91 -91q-18 -18 -18 -45t18 -45l189 -189h-502q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h502l-189 -189q-19 -19 -19 -45t19 -45l91 -91q18 -18 45 -18t45 18l362 362l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1284 641q0 27 -18 45l-362 362l-91 91q-18 18 -45 18t-45 -18l-91 -91l-362 -362q-18 -18 -18 -45t18 -45l91 -91q18 -18 45 -18t45 18l189 189v-502q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v502l189 -189q19 -19 45 -19t45 19l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1284 639q0 27 -18 45l-91 91q-18 18 -45 18t-45 -18l-189 -189v502q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-502l-189 189q-19 19 -45 19t-45 -19l-91 -91q-18 -18 -18 -45t18 -45l362 -362l91 -91q18 -18 45 -18t45 18l91 91l362 362q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1042 887q-2 -1 -9.5 -9.5t-13.5 -9.5q2 0 4.5 5t5 11t3.5 7q6 7 22 15q14 6 52 12q34 8 51 -11 q-2 2 9.5 13t14.5 12q3 2 15 4.5t15 7.5l2 22q-12 -1 -17.5 7t-6.5 21q0 -2 -6 -8q0 7 -4.5 8t-11.5 -1t-9 -1q-10 3 -15 7.5t-8 16.5t-4 15q-2 5 -9.5 10.5t-9.5 10.5q-1 2 -2.5 5.5t-3 6.5t-4 5.5t-5.5 2.5t-7 -5t-7.5 -10t-4.5 -5q-3 2 -6 1.5t-4.5 -1t-4.5 -3t-5 -3.5 q-3 -2 -8.5 -3t-8.5 -2q15 5 -1 11q-10 4 -16 3q9 4 7.5 12t-8.5 14h5q-1 4 -8.5 8.5t-17.5 8.5t-13 6q-8 5 -34 9.5t-33 0.5q-5 -6 -4.5 -10.5t4 -14t3.5 -12.5q1 -6 -5.5 -13t-6.5 -12q0 -7 14 -15.5t10 -21.5q-3 -8 -16 -16t-16 -12q-5 -8 -1.5 -18.5t10.5 -16.5 q2 -2 1.5 -4t-3.5 -4.5t-5.5 -4t-6.5 -3.5l-3 -2q-11 -5 -20.5 6t-13.5 26q-7 25 -16 30q-23 8 -29 -1q-5 13 -41 26q-25 9 -58 4q6 1 0 15q-7 15 -19 12q3 6 4 17.5t1 13.5q3 13 12 23q1 1 7 8.5t9.5 13.5t0.5 6q35 -4 50 11q5 5 11.5 17t10.5 17q9 6 14 5.5t14.5 -5.5 t14.5 -5q14 -1 15.5 11t-7.5 20q12 -1 3 17q-5 7 -8 9q-12 4 -27 -5q-8 -4 2 -8q-1 1 -9.5 -10.5t-16.5 -17.5t-16 5q-1 1 -5.5 13.5t-9.5 13.5q-8 0 -16 -15q3 8 -11 15t-24 8q19 12 -8 27q-7 4 -20.5 5t-19.5 -4q-5 -7 -5.5 -11.5t5 -8t10.5 -5.5t11.5 -4t8.5 -3 q14 -10 8 -14q-2 -1 -8.5 -3.5t-11.5 -4.5t-6 -4q-3 -4 0 -14t-2 -14q-5 5 -9 17.5t-7 16.5q7 -9 -25 -6l-10 1q-4 0 -16 -2t-20.5 -1t-13.5 8q-4 8 0 20q1 4 4 2q-4 3 -11 9.5t-10 8.5q-46 -15 -94 -41q6 -1 12 1q5 2 13 6.5t10 5.5q34 14 42 7l5 5q14 -16 20 -25 q-7 4 -30 1q-20 -6 -22 -12q7 -12 5 -18q-4 3 -11.5 10t-14.5 11t-15 5q-16 0 -22 -1q-146 -80 -235 -222q7 -7 12 -8q4 -1 5 -9t2.5 -11t11.5 3q9 -8 3 -19q1 1 44 -27q19 -17 21 -21q3 -11 -10 -18q-1 2 -9 9t-9 4q-3 -5 0.5 -18.5t10.5 -12.5q-7 0 -9.5 -16t-2.5 -35.5 t-1 -23.5l2 -1q-3 -12 5.5 -34.5t21.5 -19.5q-13 -3 20 -43q6 -8 8 -9q3 -2 12 -7.5t15 -10t10 -10.5q4 -5 10 -22.5t14 -23.5q-2 -6 9.5 -20t10.5 -23q-1 0 -2.5 -1t-2.5 -1q3 -7 15.5 -14t15.5 -13q1 -3 2 -10t3 -11t8 -2q2 20 -24 62q-15 25 -17 29q-3 5 -5.5 15.5 t-4.5 14.5q2 0 6 -1.5t8.5 -3.5t7.5 -4t2 -3q-3 -7 2 -17.5t12 -18.5t17 -19t12 -13q6 -6 14 -19.5t0 -13.5q9 0 20 -10t17 -20q5 -8 8 -26t5 -24q2 -7 8.5 -13.5t12.5 -9.5l16 -8t13 -7q5 -2 18.5 -10.5t21.5 -11.5q10 -4 16 -4t14.5 2.5t13.5 3.5q15 2 29 -15t21 -21 q36 -19 55 -11q-2 -1 0.5 -7.5t8 -15.5t9 -14.5t5.5 -8.5q5 -6 18 -15t18 -15q6 4 7 9q-3 -8 7 -20t18 -10q14 3 14 32q-31 -15 -49 18q0 1 -2.5 5.5t-4 8.5t-2.5 8.5t0 7.5t5 3q9 0 10 3.5t-2 12.5t-4 13q-1 8 -11 20t-12 15q-5 -9 -16 -8t-16 9q0 -1 -1.5 -5.5t-1.5 -6.5 q-13 0 -15 1q1 3 2.5 17.5t3.5 22.5q1 4 5.5 12t7.5 14.5t4 12.5t-4.5 9.5t-17.5 2.5q-19 -1 -26 -20q-1 -3 -3 -10.5t-5 -11.5t-9 -7q-7 -3 -24 -2t-24 5q-13 8 -22.5 29t-9.5 37q0 10 2.5 26.5t3 25t-5.5 24.5q3 2 9 9.5t10 10.5q2 1 4.5 1.5t4.5 0t4 1.5t3 6q-1 1 -4 3 q-3 3 -4 3q7 -3 28.5 1.5t27.5 -1.5q15 -11 22 2q0 1 -2.5 9.5t-0.5 13.5q5 -27 29 -9q3 -3 15.5 -5t17.5 -5q3 -2 7 -5.5t5.5 -4.5t5 0.5t8.5 6.5q10 -14 12 -24q11 -40 19 -44q7 -3 11 -2t4.5 9.5t0 14t-1.5 12.5l-1 8v18l-1 8q-15 3 -18.5 12t1.5 18.5t15 18.5q1 1 8 3.5 t15.5 6.5t12.5 8q21 19 15 35q7 0 11 9q-1 0 -5 3t-7.5 5t-4.5 2q9 5 2 16q5 3 7.5 11t7.5 10q9 -12 21 -2q7 8 1 16q5 7 20.5 10.5t18.5 9.5q7 -2 8 2t1 12t3 12q4 5 15 9t13 5l17 11q3 4 0 4q18 -2 31 11q10 11 -6 20q3 6 -3 9.5t-15 5.5q3 1 11.5 0.5t10.5 1.5 q15 10 -7 16q-17 5 -43 -12zM879 10q206 36 351 189q-3 3 -12.5 4.5t-12.5 3.5q-18 7 -24 8q1 7 -2.5 13t-8 9t-12.5 8t-11 7q-2 2 -7 6t-7 5.5t-7.5 4.5t-8.5 2t-10 -1l-3 -1q-3 -1 -5.5 -2.5t-5.5 -3t-4 -3t0 -2.5q-21 17 -36 22q-5 1 -11 5.5t-10.5 7t-10 1.5t-11.5 -7 q-5 -5 -6 -15t-2 -13q-7 5 0 17.5t2 18.5q-3 6 -10.5 4.5t-12 -4.5t-11.5 -8.5t-9 -6.5t-8.5 -5.5t-8.5 -7.5q-3 -4 -6 -12t-5 -11q-2 4 -11.5 6.5t-9.5 5.5q2 -10 4 -35t5 -38q7 -31 -12 -48q-27 -25 -29 -40q-4 -22 12 -26q0 -7 -8 -20.5t-7 -21.5q0 -6 2 -16z" />
-<glyph unicode="" horiz-adv-x="1664" d="M384 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1028 484l-682 -682q-37 -37 -90 -37q-52 0 -91 37l-106 108q-38 36 -38 90q0 53 38 91l681 681q39 -98 114.5 -173.5t173.5 -114.5zM1662 919q0 -39 -23 -106q-47 -134 -164.5 -217.5 t-258.5 -83.5q-185 0 -316.5 131.5t-131.5 316.5t131.5 316.5t316.5 131.5q58 0 121.5 -16.5t107.5 -46.5q16 -11 16 -28t-16 -28l-293 -169v-224l193 -107q5 3 79 48.5t135.5 81t70.5 35.5q15 0 23.5 -10t8.5 -25z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1024 128h640v128h-640v-128zM640 640h1024v128h-1024v-128zM1280 1152h384v128h-384v-128zM1792 320v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 832v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19 t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1403 1241q17 -41 -14 -70l-493 -493v-742q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-256 256q-19 19 -19 45v486l-493 493q-31 29 -14 70q17 39 59 39h1280q42 0 59 -39z" />
-<glyph unicode="" horiz-adv-x="1792" d="M640 1280h512v128h-512v-128zM1792 640v-480q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v480h672v-160q0 -26 19 -45t45 -19h320q26 0 45 19t19 45v160h672zM1024 640v-128h-256v128h256zM1792 1120v-384h-1792v384q0 66 47 113t113 47h352v160q0 40 28 68 t68 28h576q40 0 68 -28t28 -68v-160h352q66 0 113 -47t47 -113z" />
-<glyph unicode="" d="M1283 995l-355 -355l355 -355l144 144q29 31 70 14q39 -17 39 -59v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l144 144l-355 355l-355 -355l144 -144q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l144 -144 l355 355l-355 355l-144 -144q-19 -19 -45 -19q-12 0 -24 5q-40 17 -40 59v448q0 26 19 45t45 19h448q42 0 59 -40q17 -39 -14 -69l-144 -144l355 -355l355 355l-144 144q-31 30 -14 69q17 40 59 40h448q26 0 45 -19t19 -45v-448q0 -42 -39 -59q-13 -5 -25 -5q-26 0 -45 19z " />
-<glyph unicode="" horiz-adv-x="1920" d="M593 640q-162 -5 -265 -128h-134q-82 0 -138 40.5t-56 118.5q0 353 124 353q6 0 43.5 -21t97.5 -42.5t119 -21.5q67 0 133 23q-5 -37 -5 -66q0 -139 81 -256zM1664 3q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5 t43 97.5t62 81t85.5 53.5t111.5 20q10 0 43 -21.5t73 -48t107 -48t135 -21.5t135 21.5t107 48t73 48t43 21.5q61 0 111.5 -20t85.5 -53.5t62 -81t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM640 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75 t75 -181zM1344 896q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5zM1920 671q0 -78 -56 -118.5t-138 -40.5h-134q-103 123 -265 128q81 117 81 256q0 29 -5 66q66 -23 133 -23q59 0 119 21.5t97.5 42.5 t43.5 21q124 0 124 -353zM1792 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1456 320q0 40 -28 68l-208 208q-28 28 -68 28q-42 0 -72 -32q3 -3 19 -18.5t21.5 -21.5t15 -19t13 -25.5t3.5 -27.5q0 -40 -28 -68t-68 -28q-15 0 -27.5 3.5t-25.5 13t-19 15t-21.5 21.5t-18.5 19q-33 -31 -33 -73q0 -40 28 -68l206 -207q27 -27 68 -27q40 0 68 26 l147 146q28 28 28 67zM753 1025q0 40 -28 68l-206 207q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l208 -208q27 -27 68 -27q42 0 72 31q-3 3 -19 18.5t-21.5 21.5t-15 19t-13 25.5t-3.5 27.5q0 40 28 68t68 28q15 0 27.5 -3.5t25.5 -13t19 -15 t21.5 -21.5t18.5 -19q33 31 33 73zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-206 207q-83 83 -83 203q0 123 88 209l-88 88q-86 -88 -208 -88q-120 0 -204 84l-208 208q-84 84 -84 204t85 203l147 146q83 83 203 83q121 0 204 -85l206 -207 q83 -83 83 -203q0 -123 -88 -209l88 -88q86 88 208 88q120 0 204 -84l208 -208q84 -84 84 -204z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088q-185 0 -316.5 131.5t-131.5 316.5q0 132 71 241.5t187 163.5q-2 28 -2 43q0 212 150 362t362 150q158 0 286.5 -88t187.5 -230q70 62 166 62q106 0 181 -75t75 -181q0 -75 -41 -138q129 -30 213 -134.5t84 -239.5z " />
-<glyph unicode="" horiz-adv-x="1664" d="M1527 88q56 -89 21.5 -152.5t-140.5 -63.5h-1152q-106 0 -140.5 63.5t21.5 152.5l503 793v399h-64q-26 0 -45 19t-19 45t19 45t45 19h512q26 0 45 -19t19 -45t-19 -45t-45 -19h-64v-399zM748 813l-272 -429h712l-272 429l-20 31v37v399h-128v-399v-37z" />
-<glyph unicode="" horiz-adv-x="1792" d="M960 640q26 0 45 -19t19 -45t-19 -45t-45 -19t-45 19t-19 45t19 45t45 19zM1260 576l507 -398q28 -20 25 -56q-5 -35 -35 -51l-128 -64q-13 -7 -29 -7q-17 0 -31 8l-690 387l-110 -66q-8 -4 -12 -5q14 -49 10 -97q-7 -77 -56 -147.5t-132 -123.5q-132 -84 -277 -84 q-136 0 -222 78q-90 84 -79 207q7 76 56 147t131 124q132 84 278 84q83 0 151 -31q9 13 22 22l122 73l-122 73q-13 9 -22 22q-68 -31 -151 -31q-146 0 -278 84q-82 53 -131 124t-56 147q-5 59 15.5 113t63.5 93q85 79 222 79q145 0 277 -84q83 -52 132 -123t56 -148 q4 -48 -10 -97q4 -1 12 -5l110 -66l690 387q14 8 31 8q16 0 29 -7l128 -64q30 -16 35 -51q3 -36 -25 -56zM579 836q46 42 21 108t-106 117q-92 59 -192 59q-74 0 -113 -36q-46 -42 -21 -108t106 -117q92 -59 192 -59q74 0 113 36zM494 91q81 51 106 117t-21 108 q-39 36 -113 36q-100 0 -192 -59q-81 -51 -106 -117t21 -108q39 -36 113 -36q100 0 192 59zM672 704l96 -58v11q0 36 33 56l14 8l-79 47l-26 -26q-3 -3 -10 -11t-12 -12q-2 -2 -4 -3.5t-3 -2.5zM896 480l96 -32l736 576l-128 64l-768 -431v-113l-160 -96l9 -8q2 -2 7 -6 q4 -4 11 -12t11 -12l26 -26zM1600 64l128 64l-520 408l-177 -138q-2 -3 -13 -7z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1696 1152q40 0 68 -28t28 -68v-1216q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v288h-544q-40 0 -68 28t-28 68v672q0 40 20 88t48 76l408 408q28 28 76 48t88 20h416q40 0 68 -28t28 -68v-328q68 40 128 40h416zM1152 939l-299 -299h299v299zM512 1323l-299 -299 h299v299zM708 676l316 316v416h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h512v256q0 40 20 88t48 76zM1664 -128v1152h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h896z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1404 151q0 -117 -79 -196t-196 -79q-135 0 -235 100l-777 776q-113 115 -113 271q0 159 110 270t269 111q158 0 273 -113l605 -606q10 -10 10 -22q0 -16 -30.5 -46.5t-46.5 -30.5q-13 0 -23 10l-606 607q-79 77 -181 77q-106 0 -179 -75t-73 -181q0 -105 76 -181 l776 -777q63 -63 145 -63q64 0 106 42t42 106q0 82 -63 145l-581 581q-26 24 -60 24q-29 0 -48 -19t-19 -48q0 -32 25 -59l410 -410q10 -10 10 -22q0 -16 -31 -47t-47 -31q-12 0 -22 10l-410 410q-63 61 -63 149q0 82 57 139t139 57q88 0 149 -63l581 -581q100 -98 100 -235 z" />
-<glyph unicode="" d="M384 0h768v384h-768v-384zM1280 0h128v896q0 14 -10 38.5t-20 34.5l-281 281q-10 10 -34 20t-39 10v-416q0 -40 -28 -68t-68 -28h-576q-40 0 -68 28t-28 68v416h-128v-1280h128v416q0 40 28 68t68 28h832q40 0 68 -28t28 -68v-416zM896 928v320q0 13 -9.5 22.5t-22.5 9.5 h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5zM1536 896v-928q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h928q40 0 88 -20t76 -48l280 -280q28 -28 48 -76t20 -88z" />
-<glyph unicode="" d="M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1536 192v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 704v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 1216v-128q0 -26 -19 -45 t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M384 128q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM384 640q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1152q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z M1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M381 -84q0 -80 -54.5 -126t-135.5 -46q-106 0 -172 66l57 88q49 -45 106 -45q29 0 50.5 14.5t21.5 42.5q0 64 -105 56l-26 56q8 10 32.5 43.5t42.5 54t37 38.5v1q-16 0 -48.5 -1t-48.5 -1v-53h-106v152h333v-88l-95 -115q51 -12 81 -49t30 -88zM383 543v-159h-362 q-6 36 -6 54q0 51 23.5 93t56.5 68t66 47.5t56.5 43.5t23.5 45q0 25 -14.5 38.5t-39.5 13.5q-46 0 -81 -58l-85 59q24 51 71.5 79.5t105.5 28.5q73 0 123 -41.5t50 -112.5q0 -50 -34 -91.5t-75 -64.5t-75.5 -50.5t-35.5 -52.5h127v60h105zM1792 224v-192q0 -13 -9.5 -22.5 t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1123v-99h-335v99h107q0 41 0.5 122t0.5 121v12h-2q-8 -17 -50 -54l-71 76l136 127h106v-404h108zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5 t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1760 640q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-1728q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h1728zM483 704q-28 35 -51 80q-48 97 -48 188q0 181 134 309q133 127 393 127q50 0 167 -19q66 -12 177 -48q10 -38 21 -118q14 -123 14 -183q0 -18 -5 -45l-12 -3l-84 6 l-14 2q-50 149 -103 205q-88 91 -210 91q-114 0 -182 -59q-67 -58 -67 -146q0 -73 66 -140t279 -129q69 -20 173 -66q58 -28 95 -52h-743zM990 448h411q7 -39 7 -92q0 -111 -41 -212q-23 -55 -71 -104q-37 -35 -109 -81q-80 -48 -153 -66q-80 -21 -203 -21q-114 0 -195 23 l-140 40q-57 16 -72 28q-8 8 -8 22v13q0 108 -2 156q-1 30 0 68l2 37v44l102 2q15 -34 30 -71t22.5 -56t12.5 -27q35 -57 80 -94q43 -36 105 -57q59 -22 132 -22q64 0 139 27q77 26 122 86q47 61 47 129q0 84 -81 157q-34 29 -137 71z" />
-<glyph unicode="" d="M48 1313q-37 2 -45 4l-3 88q13 1 40 1q60 0 112 -4q132 -7 166 -7q86 0 168 3q116 4 146 5q56 0 86 2l-1 -14l2 -64v-9q-60 -9 -124 -9q-60 0 -79 -25q-13 -14 -13 -132q0 -13 0.5 -32.5t0.5 -25.5l1 -229l14 -280q6 -124 51 -202q35 -59 96 -92q88 -47 177 -47 q104 0 191 28q56 18 99 51q48 36 65 64q36 56 53 114q21 73 21 229q0 79 -3.5 128t-11 122.5t-13.5 159.5l-4 59q-5 67 -24 88q-34 35 -77 34l-100 -2l-14 3l2 86h84l205 -10q76 -3 196 10l18 -2q6 -38 6 -51q0 -7 -4 -31q-45 -12 -84 -13q-73 -11 -79 -17q-15 -15 -15 -41 q0 -7 1.5 -27t1.5 -31q8 -19 22 -396q6 -195 -15 -304q-15 -76 -41 -122q-38 -65 -112 -123q-75 -57 -182 -89q-109 -33 -255 -33q-167 0 -284 46q-119 47 -179 122q-61 76 -83 195q-16 80 -16 237v333q0 188 -17 213q-25 36 -147 39zM1536 -96v64q0 14 -9 23t-23 9h-1472 q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h1472q14 0 23 9t9 23z" />
-<glyph unicode="" horiz-adv-x="1664" d="M512 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23 v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 160v192 q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192 q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1664 1248v-1088q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1344q66 0 113 -47t47 -113 z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1190 955l293 293l-107 107l-293 -293zM1637 1248q0 -27 -18 -45l-1286 -1286q-18 -18 -45 -18t-45 18l-198 198q-18 18 -18 45t18 45l1286 1286q18 18 45 18t45 -18l198 -198q18 -18 18 -45zM286 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM636 1276 l196 -60l-196 -60l-60 -196l-60 196l-196 60l196 60l60 196zM1566 798l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM926 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98z" />
-<glyph unicode="" horiz-adv-x="1792" d="M640 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM256 640h384v256h-158q-13 0 -22 -9l-195 -195q-9 -9 -9 -22v-30zM1536 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM1792 1216v-1024q0 -15 -4 -26.5t-13.5 -18.5 t-16.5 -11.5t-23.5 -6t-22.5 -2t-25.5 0t-22.5 0.5q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-64q-3 0 -22.5 -0.5t-25.5 0t-22.5 2t-23.5 6t-16.5 11.5t-13.5 18.5t-4 26.5q0 26 19 45t45 19v320q0 8 -0.5 35t0 38 t2.5 34.5t6.5 37t14 30.5t22.5 30l198 198q19 19 50.5 32t58.5 13h160v192q0 26 19 45t45 19h1024q26 0 45 -19t19 -45z" />
-<glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103q-111 0 -218 32q59 93 78 164q9 34 54 211q20 -39 73 -67.5t114 -28.5q121 0 216 68.5t147 188.5t52 270q0 114 -59.5 214t-172.5 163t-255 63q-105 0 -196 -29t-154.5 -77t-109 -110.5t-67 -129.5t-21.5 -134 q0 -104 40 -183t117 -111q30 -12 38 20q2 7 8 31t8 30q6 23 -11 43q-51 61 -51 151q0 151 104.5 259.5t273.5 108.5q151 0 235.5 -82t84.5 -213q0 -170 -68.5 -289t-175.5 -119q-61 0 -98 43.5t-23 104.5q8 35 26.5 93.5t30 103t11.5 75.5q0 50 -27 83t-77 33 q-62 0 -105 -57t-43 -142q0 -73 25 -122l-99 -418q-17 -70 -13 -177q-206 91 -333 281t-127 423q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-725q85 122 108 210q9 34 53 209q21 -39 73.5 -67t112.5 -28q181 0 295.5 147.5t114.5 373.5q0 84 -35 162.5t-96.5 139t-152.5 97t-197 36.5q-104 0 -194.5 -28.5t-153 -76.5 t-107.5 -109.5t-66.5 -128t-21.5 -132.5q0 -102 39.5 -180t116.5 -110q13 -5 23.5 0t14.5 19q10 44 15 61q6 23 -11 42q-50 62 -50 150q0 150 103.5 256.5t270.5 106.5q149 0 232.5 -81t83.5 -210q0 -168 -67.5 -286t-173.5 -118q-60 0 -97 43.5t-23 103.5q8 34 26.5 92.5 t29.5 102t11 74.5q0 49 -26.5 81.5t-75.5 32.5q-61 0 -103.5 -56.5t-42.5 -139.5q0 -72 24 -121l-98 -414q-24 -100 -7 -254h-183q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960z" />
-<glyph unicode="" d="M678 -57q0 -38 -10 -71h-380q-95 0 -171.5 56.5t-103.5 147.5q24 45 69 77.5t100 49.5t107 24t107 7q32 0 49 -2q6 -4 30.5 -21t33 -23t31 -23t32 -25.5t27.5 -25.5t26.5 -29.5t21 -30.5t17.5 -34.5t9.5 -36t4.5 -40.5zM385 294q-234 -7 -385 -85v433q103 -118 273 -118 q32 0 70 5q-21 -61 -21 -86q0 -67 63 -149zM558 805q0 -100 -43.5 -160.5t-140.5 -60.5q-51 0 -97 26t-78 67.5t-56 93.5t-35.5 104t-11.5 99q0 96 51.5 165t144.5 69q66 0 119 -41t84 -104t47 -130t16 -128zM1536 896v-736q0 -119 -84.5 -203.5t-203.5 -84.5h-468 q39 73 39 157q0 66 -22 122.5t-55.5 93t-72 71t-72 59.5t-55.5 54.5t-22 59.5q0 36 23 68t56 61.5t65.5 64.5t55.5 93t23 131t-26.5 145.5t-75.5 118.5q-6 6 -14 11t-12.5 7.5t-10 9.5t-10.5 17h135l135 64h-437q-138 0 -244.5 -38.5t-182.5 -133.5q0 126 81 213t207 87h960 q119 0 203.5 -84.5t84.5 -203.5v-96h-256v256h-128v-256h-256v-128h256v-256h128v256h256z" />
-<glyph unicode="" horiz-adv-x="1664" d="M876 71q0 21 -4.5 40.5t-9.5 36t-17.5 34.5t-21 30.5t-26.5 29.5t-27.5 25.5t-32 25.5t-31 23t-33 23t-30.5 21q-17 2 -50 2q-54 0 -106 -7t-108 -25t-98 -46t-69 -75t-27 -107q0 -68 35.5 -121.5t93 -84t120.5 -45.5t127 -15q59 0 112.5 12.5t100.5 39t74.5 73.5 t27.5 110zM756 933q0 60 -16.5 127.5t-47 130.5t-84 104t-119.5 41q-93 0 -144 -69t-51 -165q0 -47 11.5 -99t35.5 -104t56 -93.5t78 -67.5t97 -26q97 0 140.5 60.5t43.5 160.5zM625 1408h437l-135 -79h-135q71 -45 110 -126t39 -169q0 -74 -23 -131.5t-56 -92.5t-66 -64.5 t-56 -61t-23 -67.5q0 -26 16.5 -51t43 -48t58.5 -48t64 -55.5t58.5 -66t43 -85t16.5 -106.5q0 -160 -140 -282q-152 -131 -420 -131q-59 0 -119.5 10t-122 33.5t-108.5 58t-77 89t-30 121.5q0 61 37 135q32 64 96 110.5t145 71t155 36t150 13.5q-64 83 -64 149q0 12 2 23.5 t5 19.5t8 21.5t7 21.5q-40 -5 -70 -5q-149 0 -255.5 98t-106.5 246q0 140 95 250.5t234 141.5q94 20 187 20zM1664 1152v-128h-256v-256h-128v256h-256v128h256v256h128v-256h256z" />
-<glyph unicode="" horiz-adv-x="1920" d="M768 384h384v96h-128v448h-114l-148 -137l77 -80q42 37 55 57h2v-288h-128v-96zM1280 640q0 -70 -21 -142t-59.5 -134t-101.5 -101t-138 -39t-138 39t-101.5 101t-59.5 134t-21 142t21 142t59.5 134t101.5 101t138 39t138 -39t101.5 -101t59.5 -134t21 -142zM1792 384 v512q-106 0 -181 75t-75 181h-1152q0 -106 -75 -181t-181 -75v-512q106 0 181 -75t75 -181h1152q0 106 75 181t181 75zM1920 1216v-1152q0 -26 -19 -45t-45 -19h-1792q-26 0 -45 19t-19 45v1152q0 26 19 45t45 19h1792q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 320q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="640" d="M640 1088v-896q0 -26 -19 -45t-45 -19t-45 19l-448 448q-19 19 -19 45t19 45l448 448q19 19 45 19t45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="640" d="M576 640q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19t-19 45v896q0 26 19 45t45 19t45 -19l448 -448q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M160 0h608v1152h-640v-1120q0 -13 9.5 -22.5t22.5 -9.5zM1536 32v1120h-640v-1152h608q13 0 22.5 9.5t9.5 22.5zM1664 1248v-1216q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1344q66 0 113 -47t47 -113z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45zM1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 826v-794q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v794q44 -49 101 -87q362 -246 497 -345q57 -42 92.5 -65.5t94.5 -48t110 -24.5h1h1q51 0 110 24.5t94.5 48t92.5 65.5q170 123 498 345q57 39 100 87zM1792 1120q0 -79 -49 -151t-122 -123 q-376 -261 -468 -325q-10 -7 -42.5 -30.5t-54 -38t-52 -32.5t-57.5 -27t-50 -9h-1h-1q-23 0 -50 9t-57.5 27t-52 32.5t-54 38t-42.5 30.5q-91 64 -262 182.5t-205 142.5q-62 42 -117 115.5t-55 136.5q0 78 41.5 130t118.5 52h1472q65 0 112.5 -47t47.5 -113z" />
-<glyph unicode="" d="M349 911v-991h-330v991h330zM370 1217q1 -73 -50.5 -122t-135.5 -49h-2q-82 0 -132 49t-50 122q0 74 51.5 122.5t134.5 48.5t133 -48.5t51 -122.5zM1536 488v-568h-329v530q0 105 -40.5 164.5t-126.5 59.5q-63 0 -105.5 -34.5t-63.5 -85.5q-11 -30 -11 -81v-553h-329 q2 399 2 647t-1 296l-1 48h329v-144h-2q20 32 41 56t56.5 52t87 43.5t114.5 15.5q171 0 275 -113.5t104 -332.5z" />
-<glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61q-172 0 -327 72.5t-264 204.5q-7 10 -6.5 22.5t8.5 20.5l137 138q10 9 25 9q16 -2 23 -12q73 -95 179 -147t225 -52q104 0 198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5t-40.5 198.5t-109.5 163.5 t-163.5 109.5t-198.5 40.5q-98 0 -188 -35.5t-160 -101.5l137 -138q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l130 -129q107 101 244.5 156.5t284.5 55.5q156 0 298 -61t245 -164t164 -245t61 -298z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1771 0q0 -53 -37 -90l-107 -108q-39 -37 -91 -37q-53 0 -90 37l-363 364q-38 36 -38 90q0 53 43 96l-256 256l-126 -126q-14 -14 -34 -14t-34 14q2 -2 12.5 -12t12.5 -13t10 -11.5t10 -13.5t6 -13.5t5.5 -16.5t1.5 -18q0 -38 -28 -68q-3 -3 -16.5 -18t-19 -20.5 t-18.5 -16.5t-22 -15.5t-22 -9t-26 -4.5q-40 0 -68 28l-408 408q-28 28 -28 68q0 13 4.5 26t9 22t15.5 22t16.5 18.5t20.5 19t18 16.5q30 28 68 28q10 0 18 -1.5t16.5 -5.5t13.5 -6t13.5 -10t11.5 -10t13 -12.5t12 -12.5q-14 14 -14 34t14 34l348 348q14 14 34 14t34 -14 q-2 2 -12.5 12t-12.5 13t-10 11.5t-10 13.5t-6 13.5t-5.5 16.5t-1.5 18q0 38 28 68q3 3 16.5 18t19 20.5t18.5 16.5t22 15.5t22 9t26 4.5q40 0 68 -28l408 -408q28 -28 28 -68q0 -13 -4.5 -26t-9 -22t-15.5 -22t-16.5 -18.5t-20.5 -19t-18 -16.5q-30 -28 -68 -28 q-10 0 -18 1.5t-16.5 5.5t-13.5 6t-13.5 10t-11.5 10t-13 12.5t-12 12.5q14 -14 14 -34t-14 -34l-126 -126l256 -256q43 43 96 43q52 0 91 -37l363 -363q37 -39 37 -91z" />
-<glyph unicode="" horiz-adv-x="1792" d="M384 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM576 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1004 351l101 382q6 26 -7.5 48.5t-38.5 29.5 t-48 -6.5t-30 -39.5l-101 -382q-60 -5 -107 -43.5t-63 -98.5q-20 -77 20 -146t117 -89t146 20t89 117q16 60 -6 117t-72 91zM1664 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 1024q0 53 -37.5 90.5 t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1472 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1792 384q0 -261 -141 -483q-19 -29 -54 -29h-1402q-35 0 -54 29 q-141 221 -141 483q0 182 71 348t191 286t286 191t348 71t348 -71t286 -191t191 -286t71 -348z" />
-<glyph unicode="" horiz-adv-x="1792" d="M896 1152q-204 0 -381.5 -69.5t-282 -187.5t-104.5 -255q0 -112 71.5 -213.5t201.5 -175.5l87 -50l-27 -96q-24 -91 -70 -172q152 63 275 171l43 38l57 -6q69 -8 130 -8q204 0 381.5 69.5t282 187.5t104.5 255t-104.5 255t-282 187.5t-381.5 69.5zM1792 640 q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22h-5q-15 0 -27 10.5t-16 27.5v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281q0 174 120 321.5 t326 233t450 85.5t450 -85.5t326 -233t120 -321.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M704 1152q-153 0 -286 -52t-211.5 -141t-78.5 -191q0 -82 53 -158t149 -132l97 -56l-35 -84q34 20 62 39l44 31l53 -10q78 -14 153 -14q153 0 286 52t211.5 141t78.5 191t-78.5 191t-211.5 141t-286 52zM704 1280q191 0 353.5 -68.5t256.5 -186.5t94 -257t-94 -257 t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224q0 139 94 257t256.5 186.5 t353.5 68.5zM1526 111q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129 q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230q0 -120 -71 -224.5t-195 -176.5z" />
-<glyph unicode="" horiz-adv-x="896" d="M885 970q18 -20 7 -44l-540 -1157q-13 -25 -42 -25q-4 0 -14 2q-17 5 -25.5 19t-4.5 30l197 808l-406 -101q-4 -1 -12 -1q-18 0 -31 11q-18 15 -13 39l201 825q4 14 16 23t28 9h328q19 0 32 -12.5t13 -29.5q0 -8 -5 -18l-171 -463l396 98q8 2 12 2q19 0 34 -15z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 288v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192q0 52 38 90t90 38h512v192h-96q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h320q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-96v-192h512q52 0 90 -38t38 -90v-192h96q40 0 68 -28t28 -68 z" />
-<glyph unicode="" horiz-adv-x="1664" d="M896 708v-580q0 -104 -76 -180t-180 -76t-180 76t-76 180q0 26 19 45t45 19t45 -19t19 -45q0 -50 39 -89t89 -39t89 39t39 89v580q33 11 64 11t64 -11zM1664 681q0 -13 -9.5 -22.5t-22.5 -9.5q-11 0 -23 10q-49 46 -93 69t-102 23q-68 0 -128 -37t-103 -97 q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -28 -17q-18 0 -29 17q-4 6 -14.5 24t-17.5 28q-43 60 -102.5 97t-127.5 37t-127.5 -37t-102.5 -97q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -29 -17q-17 0 -28 17q-4 6 -14.5 24t-17.5 28q-43 60 -103 97t-128 37q-58 0 -102 -23t-93 -69 q-12 -10 -23 -10q-13 0 -22.5 9.5t-9.5 22.5q0 5 1 7q45 183 172.5 319.5t298 204.5t360.5 68q140 0 274.5 -40t246.5 -113.5t194.5 -187t115.5 -251.5q1 -2 1 -7zM896 1408v-98q-42 2 -64 2t-64 -2v98q0 26 19 45t45 19t45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M768 -128h896v640h-416q-40 0 -68 28t-28 68v416h-384v-1152zM1024 1312v64q0 13 -9.5 22.5t-22.5 9.5h-704q-13 0 -22.5 -9.5t-9.5 -22.5v-64q0 -13 9.5 -22.5t22.5 -9.5h704q13 0 22.5 9.5t9.5 22.5zM1280 640h299l-299 299v-299zM1792 512v-672q0 -40 -28 -68t-68 -28 h-960q-40 0 -68 28t-28 68v160h-544q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h1088q40 0 68 -28t28 -68v-328q21 -13 36 -28l408 -408q28 -28 48 -76t20 -88z" />
-<glyph unicode="" horiz-adv-x="1024" d="M736 960q0 -13 -9.5 -22.5t-22.5 -9.5t-22.5 9.5t-9.5 22.5q0 46 -54 71t-106 25q-13 0 -22.5 9.5t-9.5 22.5t9.5 22.5t22.5 9.5q50 0 99.5 -16t87 -54t37.5 -90zM896 960q0 72 -34.5 134t-90 101.5t-123 62t-136.5 22.5t-136.5 -22.5t-123 -62t-90 -101.5t-34.5 -134 q0 -101 68 -180q10 -11 30.5 -33t30.5 -33q128 -153 141 -298h228q13 145 141 298q10 11 30.5 33t30.5 33q68 79 68 180zM1024 960q0 -155 -103 -268q-45 -49 -74.5 -87t-59.5 -95.5t-34 -107.5q47 -28 47 -82q0 -37 -25 -64q25 -27 25 -64q0 -52 -45 -81q13 -23 13 -47 q0 -46 -31.5 -71t-77.5 -25q-20 -44 -60 -70t-87 -26t-87 26t-60 70q-46 0 -77.5 25t-31.5 71q0 24 13 47q-45 29 -45 81q0 37 25 64q-25 27 -25 64q0 54 47 82q-4 50 -34 107.5t-59.5 95.5t-74.5 87q-103 113 -103 268q0 99 44.5 184.5t117 142t164 89t186.5 32.5 t186.5 -32.5t164 -89t117 -142t44.5 -184.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 352v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5q-12 0 -24 10l-319 320q-9 9 -9 22q0 14 9 23l320 320q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5v-192h1376q13 0 22.5 -9.5t9.5 -22.5zM1792 896q0 -14 -9 -23l-320 -320q-9 -9 -23 -9 q-13 0 -22.5 9.5t-9.5 22.5v192h-1376q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1376v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1280 608q0 14 -9 23t-23 9h-224v352q0 13 -9.5 22.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-352h-224q-13 0 -22.5 -9.5t-9.5 -22.5q0 -14 9 -23l352 -352q9 -9 23 -9t23 9l351 351q10 12 10 24zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1280 672q0 14 -9 23l-352 352q-9 9 -23 9t-23 -9l-351 -351q-10 -12 -10 -24q0 -14 9 -23t23 -9h224v-352q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5v352h224q13 0 22.5 9.5t9.5 22.5zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -26 -19 -45t-45 -19t-45 19t-19 45t19 45t45 19t45 -19t19 -45zM1408 131q0 -121 -73 -190t-194 -69h-874q-121 0 -194 69t-73 190q0 68 5.5 131t24 138t47.5 132.5t81 103t120 60.5q-22 -52 -22 -120v-203q-58 -20 -93 -70t-35 -111q0 -80 56 -136t136 -56 t136 56t56 136q0 61 -35.5 111t-92.5 70v203q0 62 25 93q132 -104 295 -104t295 104q25 -31 25 -93v-64q-106 0 -181 -75t-75 -181v-89q-32 -29 -32 -71q0 -40 28 -68t68 -28t68 28t28 68q0 42 -32 71v89q0 52 38 90t90 38t90 -38t38 -90v-89q-32 -29 -32 -71q0 -40 28 -68 t68 -28t68 28t28 68q0 42 -32 71v89q0 68 -34.5 127.5t-93.5 93.5q0 10 0.5 42.5t0 48t-2.5 41.5t-7 47t-13 40q68 -15 120 -60.5t81 -103t47.5 -132.5t24 -138t5.5 -131zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5 t271.5 -112.5t112.5 -271.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1280 832q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 832q0 -62 -35.5 -111t-92.5 -70v-395q0 -159 -131.5 -271.5t-316.5 -112.5t-316.5 112.5t-131.5 271.5v132q-164 20 -274 128t-110 252v512q0 26 19 45t45 19q6 0 16 -2q17 30 47 48 t65 18q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5q-33 0 -64 18v-402q0 -106 94 -181t226 -75t226 75t94 181v402q-31 -18 -64 -18q-53 0 -90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5q35 0 65 -18t47 -48q10 2 16 2q26 0 45 -19t19 -45v-512q0 -144 -110 -252 t-274 -128v-132q0 -106 94 -181t226 -75t226 75t94 181v395q-57 21 -92.5 70t-35.5 111q0 80 56 136t136 56t136 -56t56 -136z" />
-<glyph unicode="" horiz-adv-x="1792" d="M640 1152h512v128h-512v-128zM288 1152v-1280h-64q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h64zM1408 1152v-1280h-1024v1280h128v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h128zM1792 928v-832q0 -92 -66 -158t-158 -66h-64v1280h64q92 0 158 -66 t66 -158z" />
-<glyph unicode="" horiz-adv-x="1664" d="M848 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM1664 128q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q190 161 287 397.5t97 498.5 q0 165 96 262t264 117q-8 18 -8 37q0 40 28 68t68 28t68 -28t28 -68q0 -19 -8 -37q168 -20 264 -117t96 -262q0 -262 97 -498.5t287 -397.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1664 896q0 80 -56 136t-136 56h-64v-384h64q80 0 136 56t56 136zM0 128h1792q0 -106 -75 -181t-181 -75h-1280q-106 0 -181 75t-75 181zM1856 896q0 -159 -112.5 -271.5t-271.5 -112.5h-64v-32q0 -92 -66 -158t-158 -66h-704q-92 0 -158 66t-66 158v736q0 26 19 45 t45 19h1152q159 0 271.5 -112.5t112.5 -271.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M640 1472v-640q0 -61 -35.5 -111t-92.5 -70v-779q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v779q-57 20 -92.5 70t-35.5 111v640q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45 t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45zM1408 1472v-1600q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v512h-224q-13 0 -22.5 9.5t-9.5 22.5v800q0 132 94 226t226 94h256q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1024 352v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704q14 0 23 -9t9 -23zM1024 608v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704q14 0 23 -9t9 -23zM128 0h1024v768h-416q-40 0 -68 28t-28 68v416h-512v-1280z M768 896h376q-10 29 -22 41l-313 313q-12 12 -41 22v-376zM1280 864v-896q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h640q40 0 88 -20t76 -48l312 -312q28 -28 48 -76t20 -88z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1536h-1152v-1536h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM1408 1472v-1664q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1664q0 26 19 45t45 19h1280q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1152h-256v-32q0 -40 -28 -68t-68 -28h-448q-40 0 -68 28t-28 68v32h-256v-1152h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM896 1056v320q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-96h-128v96q0 13 -9.5 22.5 t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5v96h128v-96q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1408 1088v-1280q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1280q0 26 19 45t45 19h320 v288q0 40 28 68t68 28h448q40 0 68 -28t28 -68v-288h320q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1920" d="M640 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM256 640h384v256h-158q-14 -2 -22 -9l-195 -195q-7 -12 -9 -22v-30zM1536 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1664 800v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM1920 1344v-1152 q0 -26 -19 -45t-45 -19h-192q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-128q-26 0 -45 19t-19 45t19 45t45 19v416q0 26 13 58t32 51l198 198q19 19 51 32t58 13h160v320q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1280 416v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM640 1152h512v128h-512v-128zM256 1152v-1280h-32 q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h32zM1440 1152v-1280h-1088v1280h160v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h160zM1792 928v-832q0 -92 -66 -158t-158 -66h-32v1280h32q92 0 158 -66t66 -158z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1920 576q-1 -32 -288 -96l-352 -32l-224 -64h-64l-293 -352h69q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-96h-160h-64v32h64v416h-160l-192 -224h-96l-32 32v192h32v32h128v8l-192 24v128l192 24v8h-128v32h-32v192l32 32h96l192 -224h160v416h-64v32h64h160h96 q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-69l293 -352h64l224 -64l352 -32q261 -58 287 -93z" />
-<glyph unicode="" horiz-adv-x="1664" d="M640 640v384h-256v-256q0 -53 37.5 -90.5t90.5 -37.5h128zM1664 192v-192h-1152v192l128 192h-128q-159 0 -271.5 112.5t-112.5 271.5v320l-64 64l32 128h480l32 128h960l32 -192l-64 -32v-800z" />
-<glyph unicode="" d="M1280 192v896q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-512v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-896q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h512v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-320v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-320q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h320v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h320q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M627 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23zM1011 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1024" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM979 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23 l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1075 224q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM1075 608q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393 q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1075 672q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23zM1075 1056q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="640" d="M627 992q0 -13 -10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="640" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1075 352q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1075 800q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1792 544v832q0 13 -9.5 22.5t-22.5 9.5h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5zM1920 1376v-1088q0 -66 -47 -113t-113 -47h-544q0 -37 16 -77.5t32 -71t16 -43.5q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19 t-19 45q0 14 16 44t32 70t16 78h-544q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" />
-<glyph unicode="" horiz-adv-x="1920" d="M416 256q-66 0 -113 47t-47 113v704q0 66 47 113t113 47h1088q66 0 113 -47t47 -113v-704q0 -66 -47 -113t-113 -47h-1088zM384 1120v-704q0 -13 9.5 -22.5t22.5 -9.5h1088q13 0 22.5 9.5t9.5 22.5v704q0 13 -9.5 22.5t-22.5 9.5h-1088q-13 0 -22.5 -9.5t-9.5 -22.5z M1760 192h160v-96q0 -40 -47 -68t-113 -28h-1600q-66 0 -113 28t-47 68v96h160h1600zM1040 96q16 0 16 16t-16 16h-160q-16 0 -16 -16t16 -16h160z" />
-<glyph unicode="" horiz-adv-x="1152" d="M640 128q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1024 288v960q0 13 -9.5 22.5t-22.5 9.5h-832q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h832q13 0 22.5 9.5t9.5 22.5zM1152 1248v-1088q0 -66 -47 -113t-113 -47h-832 q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h832q66 0 113 -47t47 -113z" />
-<glyph unicode="" horiz-adv-x="768" d="M464 128q0 33 -23.5 56.5t-56.5 23.5t-56.5 -23.5t-23.5 -56.5t23.5 -56.5t56.5 -23.5t56.5 23.5t23.5 56.5zM672 288v704q0 13 -9.5 22.5t-22.5 9.5h-512q-13 0 -22.5 -9.5t-9.5 -22.5v-704q0 -13 9.5 -22.5t22.5 -9.5h512q13 0 22.5 9.5t9.5 22.5zM480 1136 q0 16 -16 16h-160q-16 0 -16 -16t16 -16h160q16 0 16 16zM768 1152v-1024q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v1024q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" />
-<glyph unicode="" d="M768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103 t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M768 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z M1664 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z" />
-<glyph unicode="" horiz-adv-x="1664" d="M768 1216v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136zM1664 1216 v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136z" />
-<glyph unicode="" horiz-adv-x="1568" d="M496 192q0 -60 -42.5 -102t-101.5 -42q-60 0 -102 42t-42 102t42 102t102 42q59 0 101.5 -42t42.5 -102zM928 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM320 640q0 -66 -47 -113t-113 -47t-113 47t-47 113 t47 113t113 47t113 -47t47 -113zM1360 192q0 -46 -33 -79t-79 -33t-79 33t-33 79t33 79t79 33t79 -33t33 -79zM528 1088q0 -73 -51.5 -124.5t-124.5 -51.5t-124.5 51.5t-51.5 124.5t51.5 124.5t124.5 51.5t124.5 -51.5t51.5 -124.5zM992 1280q0 -80 -56 -136t-136 -56 t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1536 640q0 -40 -28 -68t-68 -28t-68 28t-28 68t28 68t68 28t68 -28t28 -68zM1328 1088q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5z" />
-<glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 416q0 -166 -127 -451q-3 -7 -10.5 -24t-13.5 -30t-13 -22q-12 -17 -28 -17q-15 0 -23.5 10t-8.5 25q0 9 2.5 26.5t2.5 23.5q5 68 5 123q0 101 -17.5 181t-48.5 138.5t-80 101t-105.5 69.5t-133 42.5t-154 21.5t-175.5 6h-224v-256q0 -26 -19 -45t-45 -19t-45 19 l-512 512q-19 19 -19 45t19 45l512 512q19 19 45 19t45 -19t19 -45v-256h224q713 0 875 -403q53 -134 53 -333z" />
-<glyph unicode="" horiz-adv-x="1664" d="M640 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1280 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1440 320 q0 120 -69 204t-187 84q-41 0 -195 -21q-71 -11 -157 -11t-157 11q-152 21 -195 21q-118 0 -187 -84t-69 -204q0 -88 32 -153.5t81 -103t122 -60t140 -29.5t149 -7h168q82 0 149 7t140 29.5t122 60t81 103t32 153.5zM1664 496q0 -207 -61 -331q-38 -77 -105.5 -133t-141 -86 t-170 -47.5t-171.5 -22t-167 -4.5q-78 0 -142 3t-147.5 12.5t-152.5 30t-137 51.5t-121 81t-86 115q-62 123 -62 331q0 237 136 396q-27 82 -27 170q0 116 51 218q108 0 190 -39.5t189 -123.5q147 35 309 35q148 0 280 -32q105 82 187 121t189 39q51 -102 51 -218 q0 -87 -27 -168q136 -160 136 -398z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1536 224v704q0 40 -28 68t-68 28h-704q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68v-960q0 -40 28 -68t68 -28h1216q40 0 68 28t28 68zM1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320 q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1781 605q0 35 -53 35h-1088q-40 0 -85.5 -21.5t-71.5 -52.5l-294 -363q-18 -24 -18 -40q0 -35 53 -35h1088q40 0 86 22t71 53l294 363q18 22 18 39zM640 768h768v160q0 40 -28 68t-68 28h-576q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68 v-853l256 315q44 53 116 87.5t140 34.5zM1909 605q0 -62 -46 -120l-295 -363q-43 -53 -116 -87.5t-140 -34.5h-1088q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158v-160h192q54 0 99 -24.5t67 -70.5q15 -32 15 -68z " />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" d="M1134 461q-37 -121 -138 -195t-228 -74t-228 74t-138 195q-8 25 4 48.5t38 31.5q25 8 48.5 -4t31.5 -38q25 -80 92.5 -129.5t151.5 -49.5t151.5 49.5t92.5 129.5q8 26 32 38t49 4t37 -31.5t4 -48.5zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5 t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5 t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1134 307q8 -25 -4 -48.5t-37 -31.5t-49 4t-32 38q-25 80 -92.5 129.5t-151.5 49.5t-151.5 -49.5t-92.5 -129.5q-8 -26 -31.5 -38t-48.5 -4q-26 8 -38 31.5t-4 48.5q37 121 138 195t228 74t228 -74t138 -195zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204 t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1152 448q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h640q26 0 45 -19t19 -45zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M832 448v128q0 14 -9 23t-23 9h-192v192q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-192h-192q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h192v-192q0 -14 9 -23t23 -9h128q14 0 23 9t9 23v192h192q14 0 23 9t9 23zM1408 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5 t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 640q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1920 512q0 -212 -150 -362t-362 -150q-192 0 -338 128h-220q-146 -128 -338 -128q-212 0 -362 150 t-150 362t150 362t362 150h896q212 0 362 -150t150 -362z" />
-<glyph unicode="" horiz-adv-x="1920" d="M384 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM512 624v-96q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h224q16 0 16 -16zM384 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 368v-96q0 -16 -16 -16 h-864q-16 0 -16 16v96q0 16 16 16h864q16 0 16 -16zM768 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM640 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1024 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16 h96q16 0 16 -16zM896 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1280 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1152 880v-96 q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 880v-352q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h112v240q0 16 16 16h96q16 0 16 -16zM1792 128v896h-1664v-896 h1664zM1920 1024v-896q0 -53 -37.5 -90.5t-90.5 -37.5h-1664q-53 0 -90.5 37.5t-37.5 90.5v896q0 53 37.5 90.5t90.5 37.5h1664q53 0 90.5 -37.5t37.5 -90.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1664 491v616q-169 -91 -306 -91q-82 0 -145 32q-100 49 -184 76.5t-178 27.5q-173 0 -403 -127v-599q245 113 433 113q55 0 103.5 -7.5t98 -26t77 -31t82.5 -39.5l28 -14q44 -22 101 -22q120 0 293 92zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9 h-64q-14 0 -23 9t-9 23v1266q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102 q-15 -9 -33 -9q-16 0 -32 8q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" />
-<glyph unicode="" horiz-adv-x="1792" d="M832 536v192q-181 -16 -384 -117v-185q205 96 384 110zM832 954v197q-172 -8 -384 -126v-189q215 111 384 118zM1664 491v184q-235 -116 -384 -71v224q-20 6 -39 15q-5 3 -33 17t-34.5 17t-31.5 15t-34.5 15.5t-32.5 13t-36 12.5t-35 8.5t-39.5 7.5t-39.5 4t-44 2 q-23 0 -49 -3v-222h19q102 0 192.5 -29t197.5 -82q19 -9 39 -15v-188q42 -17 91 -17q120 0 293 92zM1664 918v189q-169 -91 -306 -91q-45 0 -78 8v-196q148 -42 384 90zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v1266 q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102q-15 -9 -33 -9q-16 0 -32 8 q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" />
-<glyph unicode="" horiz-adv-x="1664" d="M585 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23zM1664 96v-64q0 -14 -9 -23t-23 -9h-960q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h960q14 0 23 -9 t9 -23z" />
-<glyph unicode="" horiz-adv-x="1920" d="M617 137l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23zM1208 1204l-373 -1291q-4 -13 -15.5 -19.5t-23.5 -2.5l-62 17q-13 4 -19.5 15.5t-2.5 24.5 l373 1291q4 13 15.5 19.5t23.5 2.5l62 -17q13 -4 19.5 -15.5t2.5 -24.5zM1865 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23z" />
-<glyph unicode="" horiz-adv-x="1792" d="M640 454v-70q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-69l-397 -398q-19 -19 -19 -45t19 -45zM1792 416q0 -58 -17 -133.5t-38.5 -138t-48 -125t-40.5 -90.5l-20 -40q-8 -17 -28 -17q-6 0 -9 1 q-25 8 -23 34q43 400 -106 565q-64 71 -170.5 110.5t-267.5 52.5v-251q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-262q411 -28 599 -221q169 -173 169 -509z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1186 579l257 250l-356 52l-66 10l-30 60l-159 322v-963l59 -31l318 -168l-60 355l-12 66zM1638 841l-363 -354l86 -500q5 -33 -6 -51.5t-34 -18.5q-17 0 -40 12l-449 236l-449 -236q-23 -12 -40 -12q-23 0 -34 18.5t-6 51.5l86 500l-364 354q-32 32 -23 59.5t54 34.5 l502 73l225 455q20 41 49 41q28 0 49 -41l225 -455l502 -73q45 -7 54 -34.5t-24 -59.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1401 1187l-640 -1280q-17 -35 -57 -35q-5 0 -15 2q-22 5 -35.5 22.5t-13.5 39.5v576h-576q-22 0 -39.5 13.5t-22.5 35.5t4 42t29 30l1280 640q13 7 29 7q27 0 45 -19q15 -14 18.5 -34.5t-6.5 -39.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M557 256h595v595zM512 301l595 595h-595v-595zM1664 224v-192q0 -14 -9 -23t-23 -9h-224v-224q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v224h-864q-14 0 -23 9t-9 23v864h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224v224q0 14 9 23t23 9h192q14 0 23 -9t9 -23 v-224h851l246 247q10 9 23 9t23 -9q9 -10 9 -23t-9 -23l-247 -246v-851h224q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1024" d="M288 64q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM288 1216q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM928 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1024 1088q0 -52 -26 -96.5t-70 -69.5 q-2 -287 -226 -414q-68 -38 -203 -81q-128 -40 -169.5 -71t-41.5 -100v-26q44 -25 70 -69.5t26 -96.5q0 -80 -56 -136t-136 -56t-136 56t-56 136q0 52 26 96.5t70 69.5v820q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136q0 -52 -26 -96.5t-70 -69.5v-497 q54 26 154 57q55 17 87.5 29.5t70.5 31t59 39.5t40.5 51t28 69.5t8.5 91.5q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136z" />
-<glyph unicode="" horiz-adv-x="1664" d="M439 265l-256 -256q-10 -9 -23 -9q-12 0 -23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23zM608 224v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM384 448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23t9 23t23 9h320 q14 0 23 -9t9 -23zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-334 335q-21 21 -42 56l239 18l273 -274q27 -27 68 -27.5t68 26.5l147 146q28 28 28 67q0 40 -28 68l-274 275l18 239q35 -21 56 -42l336 -336q84 -86 84 -204zM1031 1044l-239 -18 l-273 274q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l274 -274l-18 -240q-35 21 -56 42l-336 336q-84 86 -84 204q0 120 85 203l147 146q83 83 203 83q121 0 204 -85l334 -335q21 -21 42 -56zM1664 960q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9 t-9 23t9 23t23 9h320q14 0 23 -9t9 -23zM1120 1504v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM1527 1353l-256 -256q-11 -9 -23 -9t-23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23z" />
-<glyph unicode="" horiz-adv-x="1024" d="M704 280v-240q0 -16 -12 -28t-28 -12h-240q-16 0 -28 12t-12 28v240q0 16 12 28t28 12h240q16 0 28 -12t12 -28zM1020 880q0 -54 -15.5 -101t-35 -76.5t-55 -59.5t-57.5 -43.5t-61 -35.5q-41 -23 -68.5 -65t-27.5 -67q0 -17 -12 -32.5t-28 -15.5h-240q-15 0 -25.5 18.5 t-10.5 37.5v45q0 83 65 156.5t143 108.5q59 27 84 56t25 76q0 42 -46.5 74t-107.5 32q-65 0 -108 -29q-35 -25 -107 -115q-13 -16 -31 -16q-12 0 -25 8l-164 125q-13 10 -15.5 25t5.5 28q160 266 464 266q80 0 161 -31t146 -83t106 -127.5t41 -158.5z" />
-<glyph unicode="" horiz-adv-x="640" d="M640 192v-128q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64v384h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-576h64q26 0 45 -19t19 -45zM512 1344v-192q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v192 q0 26 19 45t45 19h256q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="640" d="M512 288v-224q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v224q0 26 19 45t45 19h256q26 0 45 -19t19 -45zM542 1344l-28 -768q-1 -26 -20.5 -45t-45.5 -19h-256q-26 0 -45.5 19t-20.5 45l-28 768q-1 26 17.5 45t44.5 19h320q26 0 44.5 -19t17.5 -45z" />
-<glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1534 846v-206h-514l-3 27 q-4 28 -4 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q83 65 188 65q110 0 178 -59.5t68 -158.5q0 -56 -24.5 -103t-62 -76.5t-81.5 -58.5t-82 -50.5t-65.5 -51.5t-30.5 -63h232v80 h126z" />
-<glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1536 -50v-206h-514l-4 27 q-3 45 -3 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q80 65 188 65q110 0 178 -59.5t68 -158.5q0 -66 -34.5 -118.5t-84 -86t-99.5 -62.5t-87 -63t-41 -73h232v80h126z" />
-<glyph unicode="" horiz-adv-x="1920" d="M896 128l336 384h-768l-336 -384h768zM1909 1205q15 -34 9.5 -71.5t-30.5 -65.5l-896 -1024q-38 -44 -96 -44h-768q-38 0 -69.5 20.5t-47.5 54.5q-15 34 -9.5 71.5t30.5 65.5l896 1024q38 44 96 44h768q38 0 69.5 -20.5t47.5 -54.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1664 438q0 -81 -44.5 -135t-123.5 -54q-41 0 -77.5 17.5t-59 38t-56.5 38t-71 17.5q-110 0 -110 -124q0 -39 16 -115t15 -115v-5q-22 0 -33 -1q-34 -3 -97.5 -11.5t-115.5 -13.5t-98 -5q-61 0 -103 26.5t-42 83.5q0 37 17.5 71t38 56.5t38 59t17.5 77.5q0 79 -54 123.5 t-135 44.5q-84 0 -143 -45.5t-59 -127.5q0 -43 15 -83t33.5 -64.5t33.5 -53t15 -50.5q0 -45 -46 -89q-37 -35 -117 -35q-95 0 -245 24q-9 2 -27.5 4t-27.5 4l-13 2q-1 0 -3 1q-2 0 -2 1v1024q2 -1 17.5 -3.5t34 -5t21.5 -3.5q150 -24 245 -24q80 0 117 35q46 44 46 89 q0 22 -15 50.5t-33.5 53t-33.5 64.5t-15 83q0 82 59 127.5t144 45.5q80 0 134 -44.5t54 -123.5q0 -41 -17.5 -77.5t-38 -59t-38 -56.5t-17.5 -71q0 -57 42 -83.5t103 -26.5q64 0 180 15t163 17v-2q-1 -2 -3.5 -17.5t-5 -34t-3.5 -21.5q-24 -150 -24 -245q0 -80 35 -117 q44 -46 89 -46q22 0 50.5 15t53 33.5t64.5 33.5t83 15q82 0 127.5 -59t45.5 -143z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1152 832v-128q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-217 24 -364.5 187.5t-147.5 384.5v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -185 131.5 -316.5t316.5 -131.5 t316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45zM896 1216v-512q0 -132 -94 -226t-226 -94t-226 94t-94 226v512q0 132 94 226t226 94t226 -94t94 -226z" />
-<glyph unicode="" horiz-adv-x="1408" d="M271 591l-101 -101q-42 103 -42 214v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -53 15 -113zM1385 1193l-361 -361v-128q0 -132 -94 -226t-226 -94q-55 0 -109 19l-96 -96q97 -51 205 -51q185 0 316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45v-128 q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-125 13 -235 81l-254 -254q-10 -10 -23 -10t-23 10l-82 82q-10 10 -10 23t10 23l1234 1234q10 10 23 10t23 -10l82 -82q10 -10 10 -23 t-10 -23zM1005 1325l-621 -621v512q0 132 94 226t226 94q102 0 184.5 -59t116.5 -152z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1088 576v640h-448v-1137q119 63 213 137q235 184 235 360zM1280 1344v-768q0 -86 -33.5 -170.5t-83 -150t-118 -127.5t-126.5 -103t-121 -77.5t-89.5 -49.5t-42.5 -20q-12 -6 -26 -6t-26 6q-16 7 -42.5 20t-89.5 49.5t-121 77.5t-126.5 103t-118 127.5t-83 150 t-33.5 170.5v768q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M128 -128h1408v1024h-1408v-1024zM512 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1280 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1664 1152v-1280 q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" />
-<glyph unicode="" horiz-adv-x="1408" d="M512 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 1376v-320q0 -16 -12 -25q-8 -7 -20 -7q-4 0 -7 1l-448 96q-11 2 -18 11t-7 20h-256v-102q111 -23 183.5 -111t72.5 -203v-800q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v800 q0 106 62.5 190.5t161.5 114.5v111h-32q-59 0 -115 -23.5t-91.5 -53t-66 -66.5t-40.5 -53.5t-14 -24.5q-17 -35 -57 -35q-16 0 -29 7q-23 12 -31.5 37t3.5 49q5 10 14.5 26t37.5 53.5t60.5 70t85 67t108.5 52.5q-25 42 -25 86q0 66 47 113t113 47t113 -47t47 -113 q0 -33 -14 -64h302q0 11 7 20t18 11l448 96q3 1 7 1q12 0 20 -7q12 -9 12 -25z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1440 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1664 1376q0 -249 -75.5 -430.5t-253.5 -360.5q-81 -80 -195 -176l-20 -379q-2 -16 -16 -26l-384 -224q-7 -4 -16 -4q-12 0 -23 9l-64 64q-13 14 -8 32l85 276l-281 281l-276 -85q-3 -1 -9 -1 q-14 0 -23 9l-64 64q-17 19 -5 39l224 384q10 14 26 16l379 20q96 114 176 195q188 187 358 258t431 71q14 0 24 -9.5t10 -22.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1745 763l-164 -763h-334l178 832q13 56 -15 88q-27 33 -83 33h-169l-204 -953h-334l204 953h-286l-204 -953h-334l204 953l-153 327h1276q101 0 189.5 -40.5t147.5 -113.5q60 -73 81 -168.5t0 -194.5z" />
-<glyph unicode="" d="M909 141l102 102q19 19 19 45t-19 45l-307 307l307 307q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M717 141l454 454q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l307 -307l-307 -307q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1165 397l102 102q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l307 307l307 -307q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M813 237l454 454q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-307 -307l-307 307q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1130 939l16 175h-884l47 -534h612l-22 -228l-197 -53l-196 53l-13 140h-175l22 -278l362 -100h4v1l359 99l50 544h-644l-15 181h674zM0 1408h1408l-128 -1438l-578 -162l-574 162z" />
-<glyph unicode="" horiz-adv-x="1792" d="M275 1408h1505l-266 -1333l-804 -267l-698 267l71 356h297l-29 -147l422 -161l486 161l68 339h-1208l58 297h1209l38 191h-1208z" />
-<glyph unicode="" horiz-adv-x="1792" d="M960 1280q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1792 352v-352q0 -22 -20 -30q-8 -2 -12 -2q-13 0 -23 9l-93 93q-119 -143 -318.5 -226.5t-429.5 -83.5t-429.5 83.5t-318.5 226.5l-93 -93q-9 -9 -23 -9q-4 0 -12 2q-20 8 -20 30v352 q0 14 9 23t23 9h352q22 0 30 -20q8 -19 -7 -35l-100 -100q67 -91 189.5 -153.5t271.5 -82.5v647h-192q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h192v163q-58 34 -93 92.5t-35 128.5q0 106 75 181t181 75t181 -75t75 -181q0 -70 -35 -128.5t-93 -92.5v-163h192q26 0 45 -19 t19 -45v-128q0 -26 -19 -45t-45 -19h-192v-647q149 20 271.5 82.5t189.5 153.5l-100 100q-15 16 -7 35q8 20 30 20h352q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1056 768q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v320q0 185 131.5 316.5t316.5 131.5t316.5 -131.5t131.5 -316.5q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45q0 106 -75 181t-181 75t-181 -75t-75 -181 v-320h736z" />
-<glyph unicode="" d="M1024 640q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM1152 640q0 159 -112.5 271.5t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM1280 640q0 -212 -150 -362t-362 -150t-362 150 t-150 362t150 362t362 150t362 -150t150 -362zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM896 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM1408 800v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" />
-<glyph unicode="" horiz-adv-x="384" d="M384 288v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 1312v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" />
-<glyph unicode="" d="M512 256q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM863 162q-13 232 -177 396t-396 177q-14 1 -24 -9t-10 -23v-128q0 -13 8.5 -22t21.5 -10q154 -11 264 -121t121 -264q1 -13 10 -21.5t22 -8.5h128q13 0 23 10 t9 24zM1247 161q-5 154 -56 297.5t-139.5 260t-205 205t-260 139.5t-297.5 56q-14 1 -23 -9q-10 -10 -10 -23v-128q0 -13 9 -22t22 -10q204 -7 378 -111.5t278.5 -278.5t111.5 -378q1 -13 10 -22t22 -9h128q13 0 23 10q11 9 9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1152 585q32 18 32 55t-32 55l-544 320q-31 19 -64 1q-32 -19 -32 -56v-640q0 -37 32 -56 q16 -8 32 -8q17 0 32 9z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1024 1084l316 -316l-572 -572l-316 316zM813 105l618 618q19 19 19 45t-19 45l-362 362q-18 18 -45 18t-45 -18l-618 -618q-19 -19 -19 -45t19 -45l362 -362q18 -18 45 -18t45 18zM1702 742l-907 -908q-37 -37 -90.5 -37t-90.5 37l-126 126q56 56 56 136t-56 136 t-136 56t-136 -56l-125 126q-37 37 -37 90.5t37 90.5l907 906q37 37 90.5 37t90.5 -37l125 -125q-56 -56 -56 -136t56 -136t136 -56t136 56l126 -125q37 -37 37 -90.5t-37 -90.5z" />
-<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-896q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h896q26 0 45 19t19 45zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5 t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1152 736v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h832q14 0 23 -9t9 -23zM1280 288v832q0 66 -47 113t-113 47h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113zM1408 1120v-832q0 -119 -84.5 -203.5 t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1018 933q-18 -37 -58 -37h-192v-864q0 -14 -9 -23t-23 -9h-704q-21 0 -29 18q-8 20 4 35l160 192q9 11 25 11h320v640h-192q-40 0 -58 37q-17 37 9 68l320 384q18 22 49 22t49 -22l320 -384q27 -32 9 -68z" />
-<glyph unicode="" horiz-adv-x="1024" d="M32 1280h704q13 0 22.5 -9.5t9.5 -23.5v-863h192q40 0 58 -37t-9 -69l-320 -384q-18 -22 -49 -22t-49 22l-320 384q-26 31 -9 69q18 37 58 37h192v640h-320q-14 0 -25 11l-160 192q-13 14 -4 34q9 19 29 19z" />
-<glyph unicode="" d="M685 237l614 614q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-467 -467l-211 211q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l358 -358q19 -19 45 -19t45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5 t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M404 428l152 -152l-52 -52h-56v96h-96v56zM818 818q14 -13 -3 -30l-291 -291q-17 -17 -30 -3q-14 13 3 30l291 291q17 17 30 3zM544 128l544 544l-288 288l-544 -544v-288h288zM1152 736l92 92q28 28 28 68t-28 68l-152 152q-28 28 -68 28t-68 -28l-92 -92zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1280 608v480q0 26 -19 45t-45 19h-480q-42 0 -59 -39q-17 -41 14 -70l144 -144l-534 -534q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l534 534l144 -144q18 -19 45 -19q12 0 25 5q39 17 39 59zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1005 435l352 352q19 19 19 45t-19 45l-352 352q-30 31 -69 14q-40 -17 -40 -59v-160q-119 0 -216 -19.5t-162.5 -51t-114 -79t-76.5 -95.5t-44.5 -109t-21.5 -111.5t-5 -110.5q0 -181 167 -404q10 -12 25 -12q7 0 13 3q22 9 19 33q-44 354 62 473q46 52 130 75.5 t224 23.5v-160q0 -42 40 -59q12 -5 24 -5q26 0 45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M640 448l256 128l-256 128v-256zM1024 1039v-542l-512 -256v542zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1145 861q18 -35 -5 -66l-320 -448q-19 -27 -52 -27t-52 27l-320 448q-23 31 -5 66q17 35 57 35h640q40 0 57 -35zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1145 419q-17 -35 -57 -35h-640q-40 0 -57 35q-18 35 5 66l320 448q19 27 52 27t52 -27l320 -448q23 -31 5 -66zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1088 640q0 -33 -27 -52l-448 -320q-31 -23 -66 -5q-35 17 -35 57v640q0 40 35 57q35 18 66 -5l448 -320q27 -19 27 -52zM1280 160v960q0 14 -9 23t-23 9h-960q-14 0 -23 -9t-9 -23v-960q0 -14 9 -23t23 -9h960q14 0 23 9t9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M976 229l35 -159q3 -12 -3 -22.5t-17 -14.5l-5 -1q-4 -2 -10.5 -3.5t-16 -4.5t-21.5 -5.5t-25.5 -5t-30 -5t-33.5 -4.5t-36.5 -3t-38.5 -1q-234 0 -409 130.5t-238 351.5h-95q-13 0 -22.5 9.5t-9.5 22.5v113q0 13 9.5 22.5t22.5 9.5h66q-2 57 1 105h-67q-14 0 -23 9 t-9 23v114q0 14 9 23t23 9h98q67 210 243.5 338t400.5 128q102 0 194 -23q11 -3 20 -15q6 -11 3 -24l-43 -159q-3 -13 -14 -19.5t-24 -2.5l-4 1q-4 1 -11.5 2.5l-17.5 3.5t-22.5 3.5t-26 3t-29 2.5t-29.5 1q-126 0 -226 -64t-150 -176h468q16 0 25 -12q10 -12 7 -26 l-24 -114q-5 -26 -32 -26h-488q-3 -37 0 -105h459q15 0 25 -12q9 -12 6 -27l-24 -112q-2 -11 -11 -18.5t-20 -7.5h-387q48 -117 149.5 -185.5t228.5 -68.5q18 0 36 1.5t33.5 3.5t29.5 4.5t24.5 5t18.5 4.5l12 3l5 2q13 5 26 -2q12 -7 15 -21z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1020 399v-367q0 -14 -9 -23t-23 -9h-956q-14 0 -23 9t-9 23v150q0 13 9.5 22.5t22.5 9.5h97v383h-95q-14 0 -23 9.5t-9 22.5v131q0 14 9 23t23 9h95v223q0 171 123.5 282t314.5 111q185 0 335 -125q9 -8 10 -20.5t-7 -22.5l-103 -127q-9 -11 -22 -12q-13 -2 -23 7 q-5 5 -26 19t-69 32t-93 18q-85 0 -137 -47t-52 -123v-215h305q13 0 22.5 -9t9.5 -23v-131q0 -13 -9.5 -22.5t-22.5 -9.5h-305v-379h414v181q0 13 9 22.5t23 9.5h162q14 0 23 -9.5t9 -22.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M978 351q0 -153 -99.5 -263.5t-258.5 -136.5v-175q0 -14 -9 -23t-23 -9h-135q-13 0 -22.5 9.5t-9.5 22.5v175q-66 9 -127.5 31t-101.5 44.5t-74 48t-46.5 37.5t-17.5 18q-17 21 -2 41l103 135q7 10 23 12q15 2 24 -9l2 -2q113 -99 243 -125q37 -8 74 -8q81 0 142.5 43 t61.5 122q0 28 -15 53t-33.5 42t-58.5 37.5t-66 32t-80 32.5q-39 16 -61.5 25t-61.5 26.5t-62.5 31t-56.5 35.5t-53.5 42.5t-43.5 49t-35.5 58t-21 66.5t-8.5 78q0 138 98 242t255 134v180q0 13 9.5 22.5t22.5 9.5h135q14 0 23 -9t9 -23v-176q57 -6 110.5 -23t87 -33.5 t63.5 -37.5t39 -29t15 -14q17 -18 5 -38l-81 -146q-8 -15 -23 -16q-14 -3 -27 7q-3 3 -14.5 12t-39 26.5t-58.5 32t-74.5 26t-85.5 11.5q-95 0 -155 -43t-60 -111q0 -26 8.5 -48t29.5 -41.5t39.5 -33t56 -31t60.5 -27t70 -27.5q53 -20 81 -31.5t76 -35t75.5 -42.5t62 -50 t53 -63.5t31.5 -76.5t13 -94z" />
-<glyph unicode="" horiz-adv-x="898" d="M898 1066v-102q0 -14 -9 -23t-23 -9h-168q-23 -144 -129 -234t-276 -110q167 -178 459 -536q14 -16 4 -34q-8 -18 -29 -18h-195q-16 0 -25 12q-306 367 -498 571q-9 9 -9 22v127q0 13 9.5 22.5t22.5 9.5h112q132 0 212.5 43t102.5 125h-427q-14 0 -23 9t-9 23v102 q0 14 9 23t23 9h413q-57 113 -268 113h-145q-13 0 -22.5 9.5t-9.5 22.5v133q0 14 9 23t23 9h832q14 0 23 -9t9 -23v-102q0 -14 -9 -23t-23 -9h-233q47 -61 64 -144h171q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1027" d="M603 0h-172q-13 0 -22.5 9t-9.5 23v330h-288q-13 0 -22.5 9t-9.5 23v103q0 13 9.5 22.5t22.5 9.5h288v85h-288q-13 0 -22.5 9t-9.5 23v104q0 13 9.5 22.5t22.5 9.5h214l-321 578q-8 16 0 32q10 16 28 16h194q19 0 29 -18l215 -425q19 -38 56 -125q10 24 30.5 68t27.5 61 l191 420q8 19 29 19h191q17 0 27 -16q9 -14 1 -31l-313 -579h215q13 0 22.5 -9.5t9.5 -22.5v-104q0 -14 -9.5 -23t-22.5 -9h-290v-85h290q13 0 22.5 -9.5t9.5 -22.5v-103q0 -14 -9.5 -23t-22.5 -9h-290v-330q0 -13 -9.5 -22.5t-22.5 -9.5z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1043 971q0 100 -65 162t-171 62h-320v-448h320q106 0 171 62t65 162zM1280 971q0 -193 -126.5 -315t-326.5 -122h-340v-118h505q14 0 23 -9t9 -23v-128q0 -14 -9 -23t-23 -9h-505v-192q0 -14 -9.5 -23t-22.5 -9h-167q-14 0 -23 9t-9 23v192h-224q-14 0 -23 9t-9 23v128 q0 14 9 23t23 9h224v118h-224q-14 0 -23 9t-9 23v149q0 13 9 22.5t23 9.5h224v629q0 14 9 23t23 9h539q200 0 326.5 -122t126.5 -315z" />
-<glyph unicode="" horiz-adv-x="1792" d="M514 341l81 299h-159l75 -300q1 -1 1 -3t1 -3q0 1 0.5 3.5t0.5 3.5zM630 768l35 128h-292l32 -128h225zM822 768h139l-35 128h-70zM1271 340l78 300h-162l81 -299q0 -1 0.5 -3.5t1.5 -3.5q0 1 0.5 3t0.5 3zM1382 768l33 128h-297l34 -128h230zM1792 736v-64q0 -14 -9 -23 t-23 -9h-213l-164 -616q-7 -24 -31 -24h-159q-24 0 -31 24l-166 616h-209l-167 -616q-7 -24 -31 -24h-159q-11 0 -19.5 7t-10.5 17l-160 616h-208q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h175l-33 128h-142q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h109l-89 344q-5 15 5 28 q10 12 26 12h137q26 0 31 -24l90 -360h359l97 360q7 24 31 24h126q24 0 31 -24l98 -360h365l93 360q5 24 31 24h137q16 0 26 -12q10 -13 5 -28l-91 -344h111q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-145l-34 -128h179q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1167 896q18 -182 -131 -258q117 -28 175 -103t45 -214q-7 -71 -32.5 -125t-64.5 -89t-97 -58.5t-121.5 -34.5t-145.5 -15v-255h-154v251q-80 0 -122 1v-252h-154v255q-18 0 -54 0.5t-55 0.5h-200l31 183h111q50 0 58 51v402h16q-6 1 -16 1v287q-13 68 -89 68h-111v164 l212 -1q64 0 97 1v252h154v-247q82 2 122 2v245h154v-252q79 -7 140 -22.5t113 -45t82.5 -78t36.5 -114.5zM952 351q0 36 -15 64t-37 46t-57.5 30.5t-65.5 18.5t-74 9t-69 3t-64.5 -1t-47.5 -1v-338q8 0 37 -0.5t48 -0.5t53 1.5t58.5 4t57 8.5t55.5 14t47.5 21t39.5 30 t24.5 40t9.5 51zM881 827q0 33 -12.5 58.5t-30.5 42t-48 28t-55 16.5t-61.5 8t-58 2.5t-54 -1t-39.5 -0.5v-307q5 0 34.5 -0.5t46.5 0t50 2t55 5.5t51.5 11t48.5 18.5t37 27t27 38.5t9 51z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1280 768v-800q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h544v-544q0 -40 28 -68t68 -28h544zM1277 896h-509v509q82 -15 132 -65l312 -312q50 -50 65 -132z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1024 160v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23zM1024 416v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23zM1280 768v-800q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28 t-28 68v1344q0 40 28 68t68 28h544v-544q0 -40 28 -68t68 -28h544zM1277 896h-509v509q82 -15 132 -65l312 -312q50 -50 65 -132z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1191 1128h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1572 -23 v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -11v-2l14 2q9 2 30 2h248v119h121zM1661 874v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162 l230 -662h70z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1191 104h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1661 -150 v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162l230 -662h70zM1572 1001v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -10v-3l14 3q9 1 30 1h248 v119h121z" />
-<glyph unicode="" horiz-adv-x="1792" d="M736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1792 -32v-192q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832 q14 0 23 -9t9 -23zM1600 480v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1408 992v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1216 1504v-192q0 -14 -9 -23t-23 -9h-256 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1216 -32v-192q0 -14 -9 -23t-23 -9h-256q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192 q14 0 23 -9t9 -23zM1408 480v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1600 992v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1792 1504v-192q0 -14 -9 -23t-23 -9h-832 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832q14 0 23 -9t9 -23z" />
-<glyph unicode="" d="M1346 223q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23 zM1486 165q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5 t82 -252.5zM1456 882v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165z" />
-<glyph unicode="" d="M1346 1247q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9 t9 -23zM1456 -142v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165zM1486 1189q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13 q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5t82 -252.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M256 192q0 26 -19 45t-45 19q-27 0 -45.5 -19t-18.5 -45q0 -27 18.5 -45.5t45.5 -18.5q26 0 45 18.5t19 45.5zM416 704v-640q0 -26 -19 -45t-45 -19h-288q-26 0 -45 19t-19 45v640q0 26 19 45t45 19h288q26 0 45 -19t19 -45zM1600 704q0 -86 -55 -149q15 -44 15 -76 q3 -76 -43 -137q17 -56 0 -117q-15 -57 -54 -94q9 -112 -49 -181q-64 -76 -197 -78h-36h-76h-17q-66 0 -144 15.5t-121.5 29t-120.5 39.5q-123 43 -158 44q-26 1 -45 19.5t-19 44.5v641q0 25 18 43.5t43 20.5q24 2 76 59t101 121q68 87 101 120q18 18 31 48t17.5 48.5 t13.5 60.5q7 39 12.5 61t19.5 52t34 50q19 19 45 19q46 0 82.5 -10.5t60 -26t40 -40.5t24 -45t12 -50t5 -45t0.5 -39q0 -38 -9.5 -76t-19 -60t-27.5 -56q-3 -6 -10 -18t-11 -22t-8 -24h277q78 0 135 -57t57 -135z" />
-<glyph unicode="" horiz-adv-x="1664" d="M256 960q0 -26 -19 -45t-45 -19q-27 0 -45.5 19t-18.5 45q0 27 18.5 45.5t45.5 18.5q26 0 45 -18.5t19 -45.5zM416 448v640q0 26 -19 45t-45 19h-288q-26 0 -45 -19t-19 -45v-640q0 -26 19 -45t45 -19h288q26 0 45 19t19 45zM1545 597q55 -61 55 -149q-1 -78 -57.5 -135 t-134.5 -57h-277q4 -14 8 -24t11 -22t10 -18q18 -37 27 -57t19 -58.5t10 -76.5q0 -24 -0.5 -39t-5 -45t-12 -50t-24 -45t-40 -40.5t-60 -26t-82.5 -10.5q-26 0 -45 19q-20 20 -34 50t-19.5 52t-12.5 61q-9 42 -13.5 60.5t-17.5 48.5t-31 48q-33 33 -101 120q-49 64 -101 121 t-76 59q-25 2 -43 20.5t-18 43.5v641q0 26 19 44.5t45 19.5q35 1 158 44q77 26 120.5 39.5t121.5 29t144 15.5h17h76h36q133 -2 197 -78q58 -69 49 -181q39 -37 54 -94q17 -61 0 -117q46 -61 43 -137q0 -32 -15 -76z" />
-<glyph unicode="" d="M919 233v157q0 50 -29 50q-17 0 -33 -16v-224q16 -16 33 -16q29 0 29 49zM1103 355h66v34q0 51 -33 51t-33 -51v-34zM532 621v-70h-80v-423h-74v423h-78v70h232zM733 495v-367h-67v40q-39 -45 -76 -45q-33 0 -42 28q-6 16 -6 54v290h66v-270q0 -24 1 -26q1 -15 15 -15 q20 0 42 31v280h67zM985 384v-146q0 -52 -7 -73q-12 -42 -53 -42q-35 0 -68 41v-36h-67v493h67v-161q32 40 68 40q41 0 53 -42q7 -21 7 -74zM1236 255v-9q0 -29 -2 -43q-3 -22 -15 -40q-27 -40 -80 -40q-52 0 -81 38q-21 27 -21 86v129q0 59 20 86q29 38 80 38t78 -38 q21 -28 21 -86v-76h-133v-65q0 -51 34 -51q24 0 30 26q0 1 0.5 7t0.5 16.5v21.5h68zM785 1079v-156q0 -51 -32 -51t-32 51v156q0 52 32 52t32 -52zM1318 366q0 177 -19 260q-10 44 -43 73.5t-76 34.5q-136 15 -412 15q-275 0 -411 -15q-44 -5 -76.5 -34.5t-42.5 -73.5 q-20 -87 -20 -260q0 -176 20 -260q10 -43 42.5 -73t75.5 -35q137 -15 412 -15t412 15q43 5 75.5 35t42.5 73q20 84 20 260zM563 1017l90 296h-75l-51 -195l-53 195h-78l24 -69t23 -69q35 -103 46 -158v-201h74v201zM852 936v130q0 58 -21 87q-29 38 -78 38q-51 0 -78 -38 q-21 -29 -21 -87v-130q0 -58 21 -87q27 -38 78 -38q49 0 78 38q21 27 21 87zM1033 816h67v370h-67v-283q-22 -31 -42 -31q-15 0 -16 16q-1 2 -1 26v272h-67v-293q0 -37 6 -55q11 -27 43 -27q36 0 77 45v-40zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M971 292v-211q0 -67 -39 -67q-23 0 -45 22v301q22 22 45 22q39 0 39 -67zM1309 291v-46h-90v46q0 68 45 68t45 -68zM343 509h107v94h-312v-94h105v-569h100v569zM631 -60h89v494h-89v-378q-30 -42 -57 -42q-18 0 -21 21q-1 3 -1 35v364h-89v-391q0 -49 8 -73 q12 -37 58 -37q48 0 102 61v-54zM1060 88v197q0 73 -9 99q-17 56 -71 56q-50 0 -93 -54v217h-89v-663h89v48q45 -55 93 -55q54 0 71 55q9 27 9 100zM1398 98v13h-91q0 -51 -2 -61q-7 -36 -40 -36q-46 0 -46 69v87h179v103q0 79 -27 116q-39 51 -106 51q-68 0 -107 -51 q-28 -37 -28 -116v-173q0 -79 29 -116q39 -51 108 -51q72 0 108 53q18 27 21 54q2 9 2 58zM790 1011v210q0 69 -43 69t-43 -69v-210q0 -70 43 -70t43 70zM1509 260q0 -234 -26 -350q-14 -59 -58 -99t-102 -46q-184 -21 -555 -21t-555 21q-58 6 -102.5 46t-57.5 99 q-26 112 -26 350q0 234 26 350q14 59 58 99t103 47q183 20 554 20t555 -20q58 -7 102.5 -47t57.5 -99q26 -112 26 -350zM511 1536h102l-121 -399v-271h-100v271q-14 74 -61 212q-37 103 -65 187h106l71 -263zM881 1203v-175q0 -81 -28 -118q-37 -51 -106 -51q-67 0 -105 51 q-28 38 -28 118v175q0 80 28 117q38 51 105 51q69 0 106 -51q28 -37 28 -117zM1216 1365v-499h-91v55q-53 -62 -103 -62q-46 0 -59 37q-8 24 -8 75v394h91v-367q0 -33 1 -35q3 -22 21 -22q27 0 57 43v381h91z" />
-<glyph unicode="" horiz-adv-x="1408" d="M597 869q-10 -18 -257 -456q-27 -46 -65 -46h-239q-21 0 -31 17t0 36l253 448q1 0 0 1l-161 279q-12 22 -1 37q9 15 32 15h239q40 0 66 -45zM1403 1511q11 -16 0 -37l-528 -934v-1l336 -615q11 -20 1 -37q-10 -15 -32 -15h-239q-42 0 -66 45l-339 622q18 32 531 942 q25 45 64 45h241q22 0 31 -15z" />
-<glyph unicode="" d="M685 771q0 1 -126 222q-21 34 -52 34h-184q-18 0 -26 -11q-7 -12 1 -29l125 -216v-1l-196 -346q-9 -14 0 -28q8 -13 24 -13h185q31 0 50 36zM1309 1268q-7 12 -24 12h-187q-30 0 -49 -35l-411 -729q1 -2 262 -481q20 -35 52 -35h184q18 0 25 12q8 13 -1 28l-260 476v1 l409 723q8 16 0 28zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1280 640q0 37 -30 54l-512 320q-31 20 -65 2q-33 -18 -33 -56v-640q0 -38 33 -56q16 -8 31 -8q20 0 34 10l512 320q30 17 30 54zM1792 640q0 -96 -1 -150t-8.5 -136.5t-22.5 -147.5q-16 -73 -69 -123t-124 -58q-222 -25 -671 -25t-671 25q-71 8 -124.5 58t-69.5 123 q-14 65 -21.5 147.5t-8.5 136.5t-1 150t1 150t8.5 136.5t22.5 147.5q16 73 69 123t124 58q222 25 671 25t671 -25q71 -8 124.5 -58t69.5 -123q14 -65 21.5 -147.5t8.5 -136.5t1 -150z" />
-<glyph unicode="" horiz-adv-x="1792" d="M402 829l494 -305l-342 -285l-490 319zM1388 274v-108l-490 -293v-1l-1 1l-1 -1v1l-489 293v108l147 -96l342 284v2l1 -1l1 1v-2l343 -284zM554 1418l342 -285l-494 -304l-338 270zM1390 829l338 -271l-489 -319l-343 285zM1239 1418l489 -319l-338 -270l-494 304z" />
-<glyph unicode="" horiz-adv-x="1408" d="M928 135v-151l-707 -1v151zM1169 481v-701l-1 -35v-1h-1132l-35 1h-1v736h121v-618h928v618h120zM241 393l704 -65l-13 -150l-705 65zM309 709l683 -183l-39 -146l-683 183zM472 1058l609 -360l-77 -130l-609 360zM832 1389l398 -585l-124 -85l-399 584zM1285 1536 l121 -697l-149 -26l-121 697z" />
-<glyph unicode="" d="M1362 110v648h-135q20 -63 20 -131q0 -126 -64 -232.5t-174 -168.5t-240 -62q-197 0 -337 135.5t-140 327.5q0 68 20 131h-141v-648q0 -26 17.5 -43.5t43.5 -17.5h1069q25 0 43 17.5t18 43.5zM1078 643q0 124 -90.5 211.5t-218.5 87.5q-127 0 -217.5 -87.5t-90.5 -211.5 t90.5 -211.5t217.5 -87.5q128 0 218.5 87.5t90.5 211.5zM1362 1003v165q0 28 -20 48.5t-49 20.5h-174q-29 0 -49 -20.5t-20 -48.5v-165q0 -29 20 -49t49 -20h174q29 0 49 20t20 49zM1536 1211v-1142q0 -81 -58 -139t-139 -58h-1142q-81 0 -139 58t-58 139v1142q0 81 58 139 t139 58h1142q81 0 139 -58t58 -139z" />
-<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960zM698 640q0 88 -62 150t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150zM1262 640q0 88 -62 150 t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150z" />
-<glyph unicode="" d="M768 914l201 -306h-402zM1133 384h94l-459 691l-459 -691h94l104 160h522zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M815 677q8 -63 -50.5 -101t-111.5 -6q-39 17 -53.5 58t-0.5 82t52 58q36 18 72.5 12t64 -35.5t27.5 -67.5zM926 698q-14 107 -113 164t-197 13q-63 -28 -100.5 -88.5t-34.5 -129.5q4 -91 77.5 -155t165.5 -56q91 8 152 84t50 168zM1165 1240q-20 27 -56 44.5t-58 22 t-71 12.5q-291 47 -566 -2q-43 -7 -66 -12t-55 -22t-50 -43q30 -28 76 -45.5t73.5 -22t87.5 -11.5q228 -29 448 -1q63 8 89.5 12t72.5 21.5t75 46.5zM1222 205q-8 -26 -15.5 -76.5t-14 -84t-28.5 -70t-58 -56.5q-86 -48 -189.5 -71.5t-202 -22t-201.5 18.5q-46 8 -81.5 18 t-76.5 27t-73 43.5t-52 61.5q-25 96 -57 292l6 16l18 9q223 -148 506.5 -148t507.5 148q21 -6 24 -23t-5 -45t-8 -37zM1403 1166q-26 -167 -111 -655q-5 -30 -27 -56t-43.5 -40t-54.5 -31q-252 -126 -610 -88q-248 27 -394 139q-15 12 -25.5 26.5t-17 35t-9 34t-6 39.5 t-5.5 35q-9 50 -26.5 150t-28 161.5t-23.5 147.5t-22 158q3 26 17.5 48.5t31.5 37.5t45 30t46 22.5t48 18.5q125 46 313 64q379 37 676 -50q155 -46 215 -122q16 -20 16.5 -51t-5.5 -54z" />
-<glyph unicode="" d="M848 666q0 43 -41 66t-77 1q-43 -20 -42.5 -72.5t43.5 -70.5q39 -23 81 4t36 72zM928 682q8 -66 -36 -121t-110 -61t-119 40t-56 113q-2 49 25.5 93t72.5 64q70 31 141.5 -10t81.5 -118zM1100 1073q-20 -21 -53.5 -34t-53 -16t-63.5 -8q-155 -20 -324 0q-44 6 -63 9.5 t-52.5 16t-54.5 32.5q13 19 36 31t40 15.5t47 8.5q198 35 408 1q33 -5 51 -8.5t43 -16t39 -31.5zM1142 327q0 7 5.5 26.5t3 32t-17.5 16.5q-161 -106 -365 -106t-366 106l-12 -6l-5 -12q26 -154 41 -210q47 -81 204 -108q249 -46 428 53q34 19 49 51.5t22.5 85.5t12.5 71z M1272 1020q9 53 -8 75q-43 55 -155 88q-216 63 -487 36q-132 -12 -226 -46q-38 -15 -59.5 -25t-47 -34t-29.5 -54q8 -68 19 -138t29 -171t24 -137q1 -5 5 -31t7 -36t12 -27t22 -28q105 -80 284 -100q259 -28 440 63q24 13 39.5 23t31 29t19.5 40q48 267 80 473zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M390 1408h219v-388h364v-241h-364v-394q0 -136 14 -172q13 -37 52 -60q50 -31 117 -31q117 0 232 76v-242q-102 -48 -178 -65q-77 -19 -173 -19q-105 0 -186 27q-78 25 -138 75q-58 51 -79 105q-22 54 -22 161v539h-170v217q91 30 155 84q64 55 103 132q39 78 54 196z " />
-<glyph unicode="" d="M1123 127v181q-88 -56 -174 -56q-51 0 -88 23q-29 17 -39 45q-11 30 -11 129v295h274v181h-274v291h-164q-11 -90 -40 -147t-78 -99q-48 -40 -116 -63v-163h127v-404q0 -78 17 -121q17 -42 59 -78q43 -37 104 -57q62 -20 140 -20q67 0 129 14q57 13 134 49zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="768" d="M765 237q8 -19 -5 -35l-350 -384q-10 -10 -23 -10q-14 0 -24 10l-355 384q-13 16 -5 35q9 19 29 19h224v1248q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1248h224q21 0 29 -19z" />
-<glyph unicode="" horiz-adv-x="768" d="M765 1043q-9 -19 -29 -19h-224v-1248q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1248h-224q-21 0 -29 19t5 35l350 384q10 10 23 10q14 0 24 -10l355 -384q13 -16 5 -35z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1792 736v-192q0 -14 -9 -23t-23 -9h-1248v-224q0 -21 -19 -29t-35 5l-384 350q-10 10 -10 23q0 14 10 24l384 354q16 14 35 6q19 -9 19 -29v-224h1248q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1728 643q0 -14 -10 -24l-384 -354q-16 -14 -35 -6q-19 9 -19 29v224h-1248q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h1248v224q0 21 19 29t35 -5l384 -350q10 -10 10 -23z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1393 321q-39 -125 -123 -250q-129 -196 -257 -196q-49 0 -140 32q-86 32 -151 32q-61 0 -142 -33q-81 -34 -132 -34q-152 0 -301 259q-147 261 -147 503q0 228 113 374q112 144 284 144q72 0 177 -30q104 -30 138 -30q45 0 143 34q102 34 173 34q119 0 213 -65 q52 -36 104 -100q-79 -67 -114 -118q-65 -94 -65 -207q0 -124 69 -223t158 -126zM1017 1494q0 -61 -29 -136q-30 -75 -93 -138q-54 -54 -108 -72q-37 -11 -104 -17q3 149 78 257q74 107 250 148q1 -3 2.5 -11t2.5 -11q0 -4 0.5 -10t0.5 -10z" />
-<glyph unicode="" horiz-adv-x="1664" d="M682 530v-651l-682 94v557h682zM682 1273v-659h-682v565zM1664 530v-786l-907 125v661h907zM1664 1408v-794h-907v669z" />
-<glyph unicode="" horiz-adv-x="1408" d="M493 1053q16 0 27.5 11.5t11.5 27.5t-11.5 27.5t-27.5 11.5t-27 -11.5t-11 -27.5t11 -27.5t27 -11.5zM915 1053q16 0 27 11.5t11 27.5t-11 27.5t-27 11.5t-27.5 -11.5t-11.5 -27.5t11.5 -27.5t27.5 -11.5zM103 869q42 0 72 -30t30 -72v-430q0 -43 -29.5 -73t-72.5 -30 t-73 30t-30 73v430q0 42 30 72t73 30zM1163 850v-666q0 -46 -32 -78t-77 -32h-75v-227q0 -43 -30 -73t-73 -30t-73 30t-30 73v227h-138v-227q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73l-1 227h-74q-46 0 -78 32t-32 78v666h918zM931 1255q107 -55 171 -153.5t64 -215.5 h-925q0 117 64 215.5t172 153.5l-71 131q-7 13 5 20q13 6 20 -6l72 -132q95 42 201 42t201 -42l72 132q7 12 20 6q12 -7 5 -20zM1408 767v-430q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73v430q0 43 30 72.5t72 29.5q43 0 73 -29.5t30 -72.5z" />
-<glyph unicode="" d="M663 1125q-11 -1 -15.5 -10.5t-8.5 -9.5q-5 -1 -5 5q0 12 19 15h10zM750 1111q-4 -1 -11.5 6.5t-17.5 4.5q24 11 32 -2q3 -6 -3 -9zM399 684q-4 1 -6 -3t-4.5 -12.5t-5.5 -13.5t-10 -13q-7 -10 -1 -12q4 -1 12.5 7t12.5 18q1 3 2 7t2 6t1.5 4.5t0.5 4v3t-1 2.5t-3 2z M1254 325q0 18 -55 42q4 15 7.5 27.5t5 26t3 21.5t0.5 22.5t-1 19.5t-3.5 22t-4 20.5t-5 25t-5.5 26.5q-10 48 -47 103t-72 75q24 -20 57 -83q87 -162 54 -278q-11 -40 -50 -42q-31 -4 -38.5 18.5t-8 83.5t-11.5 107q-9 39 -19.5 69t-19.5 45.5t-15.5 24.5t-13 15t-7.5 7 q-14 62 -31 103t-29.5 56t-23.5 33t-15 40q-4 21 6 53.5t4.5 49.5t-44.5 25q-15 3 -44.5 18t-35.5 16q-8 1 -11 26t8 51t36 27q37 3 51 -30t4 -58q-11 -19 -2 -26.5t30 -0.5q13 4 13 36v37q-5 30 -13.5 50t-21 30.5t-23.5 15t-27 7.5q-107 -8 -89 -134q0 -15 -1 -15 q-9 9 -29.5 10.5t-33 -0.5t-15.5 5q1 57 -16 90t-45 34q-27 1 -41.5 -27.5t-16.5 -59.5q-1 -15 3.5 -37t13 -37.5t15.5 -13.5q10 3 16 14q4 9 -7 8q-7 0 -15.5 14.5t-9.5 33.5q-1 22 9 37t34 14q17 0 27 -21t9.5 -39t-1.5 -22q-22 -15 -31 -29q-8 -12 -27.5 -23.5 t-20.5 -12.5q-13 -14 -15.5 -27t7.5 -18q14 -8 25 -19.5t16 -19t18.5 -13t35.5 -6.5q47 -2 102 15q2 1 23 7t34.5 10.5t29.5 13t21 17.5q9 14 20 8q5 -3 6.5 -8.5t-3 -12t-16.5 -9.5q-20 -6 -56.5 -21.5t-45.5 -19.5q-44 -19 -70 -23q-25 -5 -79 2q-10 2 -9 -2t17 -19 q25 -23 67 -22q17 1 36 7t36 14t33.5 17.5t30 17t24.5 12t17.5 2.5t8.5 -11q0 -2 -1 -4.5t-4 -5t-6 -4.5t-8.5 -5t-9 -4.5t-10 -5t-9.5 -4.5q-28 -14 -67.5 -44t-66.5 -43t-49 -1q-21 11 -63 73q-22 31 -25 22q-1 -3 -1 -10q0 -25 -15 -56.5t-29.5 -55.5t-21 -58t11.5 -63 q-23 -6 -62.5 -90t-47.5 -141q-2 -18 -1.5 -69t-5.5 -59q-8 -24 -29 -3q-32 31 -36 94q-2 28 4 56q4 19 -1 18l-4 -5q-36 -65 10 -166q5 -12 25 -28t24 -20q20 -23 104 -90.5t93 -76.5q16 -15 17.5 -38t-14 -43t-45.5 -23q8 -15 29 -44.5t28 -54t7 -70.5q46 24 7 92 q-4 8 -10.5 16t-9.5 12t-2 6q3 5 13 9.5t20 -2.5q46 -52 166 -36q133 15 177 87q23 38 34 30q12 -6 10 -52q-1 -25 -23 -92q-9 -23 -6 -37.5t24 -15.5q3 19 14.5 77t13.5 90q2 21 -6.5 73.5t-7.5 97t23 70.5q15 18 51 18q1 37 34.5 53t72.5 10.5t60 -22.5zM626 1152 q3 17 -2.5 30t-11.5 15q-9 2 -9 -7q2 -5 5 -6q10 0 7 -15q-3 -20 8 -20q3 0 3 3zM1045 955q-2 8 -6.5 11.5t-13 5t-14.5 5.5q-5 3 -9.5 8t-7 8t-5.5 6.5t-4 4t-4 -1.5q-14 -16 7 -43.5t39 -31.5q9 -1 14.5 8t3.5 20zM867 1168q0 11 -5 19.5t-11 12.5t-9 3q-14 -1 -7 -7l4 -2 q14 -4 18 -31q0 -3 8 2zM921 1401q0 2 -2.5 5t-9 7t-9.5 6q-15 15 -24 15q-9 -1 -11.5 -7.5t-1 -13t-0.5 -12.5q-1 -4 -6 -10.5t-6 -9t3 -8.5q4 -3 8 0t11 9t15 9q1 1 9 1t15 2t9 7zM1486 60q20 -12 31 -24.5t12 -24t-2.5 -22.5t-15.5 -22t-23.5 -19.5t-30 -18.5 t-31.5 -16.5t-32 -15.5t-27 -13q-38 -19 -85.5 -56t-75.5 -64q-17 -16 -68 -19.5t-89 14.5q-18 9 -29.5 23.5t-16.5 25.5t-22 19.5t-47 9.5q-44 1 -130 1q-19 0 -57 -1.5t-58 -2.5q-44 -1 -79.5 -15t-53.5 -30t-43.5 -28.5t-53.5 -11.5q-29 1 -111 31t-146 43q-19 4 -51 9.5 t-50 9t-39.5 9.5t-33.5 14.5t-17 19.5q-10 23 7 66.5t18 54.5q1 16 -4 40t-10 42.5t-4.5 36.5t10.5 27q14 12 57 14t60 12q30 18 42 35t12 51q21 -73 -32 -106q-32 -20 -83 -15q-34 3 -43 -10q-13 -15 5 -57q2 -6 8 -18t8.5 -18t4.5 -17t1 -22q0 -15 -17 -49t-14 -48 q3 -17 37 -26q20 -6 84.5 -18.5t99.5 -20.5q24 -6 74 -22t82.5 -23t55.5 -4q43 6 64.5 28t23 48t-7.5 58.5t-19 52t-20 36.5q-121 190 -169 242q-68 74 -113 40q-11 -9 -15 15q-3 16 -2 38q1 29 10 52t24 47t22 42q8 21 26.5 72t29.5 78t30 61t39 54q110 143 124 195 q-12 112 -16 310q-2 90 24 151.5t106 104.5q39 21 104 21q53 1 106 -13.5t89 -41.5q57 -42 91.5 -121.5t29.5 -147.5q-5 -95 30 -214q34 -113 133 -218q55 -59 99.5 -163t59.5 -191q8 -49 5 -84.5t-12 -55.5t-20 -22q-10 -2 -23.5 -19t-27 -35.5t-40.5 -33.5t-61 -14 q-18 1 -31.5 5t-22.5 13.5t-13.5 15.5t-11.5 20.5t-9 19.5q-22 37 -41 30t-28 -49t7 -97q20 -70 1 -195q-10 -65 18 -100.5t73 -33t85 35.5q59 49 89.5 66.5t103.5 42.5q53 18 77 36.5t18.5 34.5t-25 28.5t-51.5 23.5q-33 11 -49.5 48t-15 72.5t15.5 47.5q1 -31 8 -56.5 t14.5 -40.5t20.5 -28.5t21 -19t21.5 -13t16.5 -9.5z" />
-<glyph unicode="" d="M1024 36q-42 241 -140 498h-2l-2 -1q-16 -6 -43 -16.5t-101 -49t-137 -82t-131 -114.5t-103 -148l-15 11q184 -150 418 -150q132 0 256 52zM839 643q-21 49 -53 111q-311 -93 -673 -93q-1 -7 -1 -21q0 -124 44 -236.5t124 -201.5q50 89 123.5 166.5t142.5 124.5t130.5 81 t99.5 48l37 13q4 1 13 3.5t13 4.5zM732 855q-120 213 -244 378q-138 -65 -234 -186t-128 -272q302 0 606 80zM1416 536q-210 60 -409 29q87 -239 128 -469q111 75 185 189.5t96 250.5zM611 1277q-1 0 -2 -1q1 1 2 1zM1201 1132q-185 164 -433 164q-76 0 -155 -19 q131 -170 246 -382q69 26 130 60.5t96.5 61.5t65.5 57t37.5 40.5zM1424 647q-3 232 -149 410l-1 -1q-9 -12 -19 -24.5t-43.5 -44.5t-71 -60.5t-100 -65t-131.5 -64.5q25 -53 44 -95q2 -6 6.5 -17.5t7.5 -16.5q36 5 74.5 7t73.5 2t69 -1.5t64 -4t56.5 -5.5t48 -6.5t36.5 -6 t25 -4.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1173 473q0 50 -19.5 91.5t-48.5 68.5t-73 49t-82.5 34t-87.5 23l-104 24q-30 7 -44 10.5t-35 11.5t-30 16t-16.5 21t-7.5 30q0 77 144 77q43 0 77 -12t54 -28.5t38 -33.5t40 -29t48 -12q47 0 75.5 32t28.5 77q0 55 -56 99.5t-142 67.5t-182 23q-68 0 -132 -15.5 t-119.5 -47t-89 -87t-33.5 -128.5q0 -61 19 -106.5t56 -75.5t80 -48.5t103 -32.5l146 -36q90 -22 112 -36q32 -20 32 -60q0 -39 -40 -64.5t-105 -25.5q-51 0 -91.5 16t-65 38.5t-45.5 45t-46 38.5t-54 16q-50 0 -75.5 -30t-25.5 -75q0 -92 122 -157.5t291 -65.5 q73 0 140 18.5t122.5 53.5t88.5 93.5t33 131.5zM1536 256q0 -159 -112.5 -271.5t-271.5 -112.5q-130 0 -234 80q-77 -16 -150 -16q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5q0 73 16 150q-80 104 -80 234q0 159 112.5 271.5t271.5 112.5q130 0 234 -80 q77 16 150 16q143 0 273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -73 -16 -150q80 -104 80 -234z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1483 512l-587 -587q-52 -53 -127.5 -53t-128.5 53l-587 587q-53 53 -53 128t53 128l587 587q53 53 128 53t128 -53l265 -265l-398 -399l-188 188q-42 42 -99 42q-59 0 -100 -41l-120 -121q-42 -40 -42 -99q0 -58 42 -100l406 -408q30 -28 67 -37l6 -4h28q60 0 99 41 l619 619l2 -3q53 -53 53 -128t-53 -128zM1406 1138l120 -120q14 -15 14 -36t-14 -36l-730 -730q-17 -15 -37 -15v0q-4 0 -6 1q-18 2 -30 14l-407 408q-14 15 -14 36t14 35l121 120q13 15 35 15t36 -15l252 -252l574 575q15 15 36 15t36 -15z" />
-<glyph unicode="" d="M704 192v1024q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-1024q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1376 576v640q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-640q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408 q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1280 480q0 -40 -28 -68t-68 -28q-51 0 -80 43l-227 341h-45v-132l247 -411q9 -15 9 -33q0 -26 -19 -45t-45 -19h-192v-272q0 -46 -33 -79t-79 -33h-160q-46 0 -79 33t-33 79v272h-192q-26 0 -45 19t-19 45q0 18 9 33l247 411v132h-45l-227 -341q-29 -43 -80 -43 q-40 0 -68 28t-28 68q0 29 16 53l256 384q73 107 176 107h384q103 0 176 -107l256 -384q16 -24 16 -53zM864 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" />
-<glyph unicode="" horiz-adv-x="1024" d="M1024 832v-416q0 -40 -28 -68t-68 -28t-68 28t-28 68v352h-64v-912q0 -46 -33 -79t-79 -33t-79 33t-33 79v464h-64v-464q0 -46 -33 -79t-79 -33t-79 33t-33 79v912h-64v-352q0 -40 -28 -68t-68 -28t-68 28t-28 68v416q0 80 56 136t136 56h640q80 0 136 -56t56 -136z M736 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" />
-<glyph unicode="" d="M773 234l350 473q16 22 24.5 59t-6 85t-61.5 79q-40 26 -83 25.5t-73.5 -17.5t-54.5 -45q-36 -40 -96 -40q-59 0 -95 40q-24 28 -54.5 45t-73.5 17.5t-84 -25.5q-46 -31 -60.5 -79t-6 -85t24.5 -59zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1472 640q0 117 -45.5 223.5t-123 184t-184 123t-223.5 45.5t-223.5 -45.5t-184 -123t-123 -184t-45.5 -223.5t45.5 -223.5t123 -184t184 -123t223.5 -45.5t223.5 45.5t184 123t123 184t45.5 223.5zM1748 363q-4 -15 -20 -20l-292 -96v-306q0 -16 -13 -26q-15 -10 -29 -4 l-292 94l-180 -248q-10 -13 -26 -13t-26 13l-180 248l-292 -94q-14 -6 -29 4q-13 10 -13 26v306l-292 96q-16 5 -20 20q-5 17 4 29l180 248l-180 248q-9 13 -4 29q4 15 20 20l292 96v306q0 16 13 26q15 10 29 4l292 -94l180 248q9 12 26 12t26 -12l180 -248l292 94 q14 6 29 -4q13 -10 13 -26v-306l292 -96q16 -5 20 -20q5 -16 -4 -29l-180 -248l180 -248q9 -12 4 -29z" />
-<glyph unicode="" d="M1262 233q-54 -9 -110 -9q-182 0 -337 90t-245 245t-90 337q0 192 104 357q-201 -60 -328.5 -229t-127.5 -384q0 -130 51 -248.5t136.5 -204t204 -136.5t248.5 -51q144 0 273.5 61.5t220.5 171.5zM1465 318q-94 -203 -283.5 -324.5t-413.5 -121.5q-156 0 -298 61 t-245 164t-164 245t-61 298q0 153 57.5 292.5t156 241.5t235.5 164.5t290 68.5q44 2 61 -39q18 -41 -15 -72q-86 -78 -131.5 -181.5t-45.5 -218.5q0 -148 73 -273t198 -198t273 -73q118 0 228 51q41 18 72 -13q14 -14 17.5 -34t-4.5 -38z" />
-<glyph unicode="" horiz-adv-x="1792" d="M1088 704q0 26 -19 45t-45 19h-256q-26 0 -45 -19t-19 -45t19 -45t45 -19h256q26 0 45 19t19 45zM1664 896v-960q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v960q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1728 1344v-256q0 -26 -19 -45t-45 -19h-1536 q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1536q26 0 45 -19t19 -45z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1632 576q0 -26 -19 -45t-45 -19h-224q0 -171 -67 -290l208 -209q19 -19 19 -45t-19 -45q-18 -19 -45 -19t-45 19l-198 197q-5 -5 -15 -13t-42 -28.5t-65 -36.5t-82 -29t-97 -13v896h-128v-896q-51 0 -101.5 13.5t-87 33t-66 39t-43.5 32.5l-15 14l-183 -207 q-20 -21 -48 -21q-24 0 -43 16q-19 18 -20.5 44.5t15.5 46.5l202 227q-58 114 -58 274h-224q-26 0 -45 19t-19 45t19 45t45 19h224v294l-173 173q-19 19 -19 45t19 45t45 19t45 -19l173 -173h844l173 173q19 19 45 19t45 -19t19 -45t-19 -45l-173 -173v-294h224q26 0 45 -19 t19 -45zM1152 1152h-640q0 133 93.5 226.5t226.5 93.5t226.5 -93.5t93.5 -226.5z" />
-<glyph unicode="" horiz-adv-x="1920" d="M1917 1016q23 -64 -150 -294q-24 -32 -65 -85q-78 -100 -90 -131q-17 -41 14 -81q17 -21 81 -82h1l1 -1l1 -1l2 -2q141 -131 191 -221q3 -5 6.5 -12.5t7 -26.5t-0.5 -34t-25 -27.5t-59 -12.5l-256 -4q-24 -5 -56 5t-52 22l-20 12q-30 21 -70 64t-68.5 77.5t-61 58 t-56.5 15.5q-3 -1 -8 -3.5t-17 -14.5t-21.5 -29.5t-17 -52t-6.5 -77.5q0 -15 -3.5 -27.5t-7.5 -18.5l-4 -5q-18 -19 -53 -22h-115q-71 -4 -146 16.5t-131.5 53t-103 66t-70.5 57.5l-25 24q-10 10 -27.5 30t-71.5 91t-106 151t-122.5 211t-130.5 272q-6 16 -6 27t3 16l4 6 q15 19 57 19l274 2q12 -2 23 -6.5t16 -8.5l5 -3q16 -11 24 -32q20 -50 46 -103.5t41 -81.5l16 -29q29 -60 56 -104t48.5 -68.5t41.5 -38.5t34 -14t27 5q2 1 5 5t12 22t13.5 47t9.5 81t0 125q-2 40 -9 73t-14 46l-6 12q-25 34 -85 43q-13 2 5 24q17 19 38 30q53 26 239 24 q82 -1 135 -13q20 -5 33.5 -13.5t20.5 -24t10.5 -32t3.5 -45.5t-1 -55t-2.5 -70.5t-1.5 -82.5q0 -11 -1 -42t-0.5 -48t3.5 -40.5t11.5 -39t22.5 -24.5q8 -2 17 -4t26 11t38 34.5t52 67t68 107.5q60 104 107 225q4 10 10 17.5t11 10.5l4 3l5 2.5t13 3t20 0.5l288 2 q39 5 64 -2.5t31 -16.5z" />
-<glyph unicode="" horiz-adv-x="1792" d="M675 252q21 34 11 69t-45 50q-34 14 -73 1t-60 -46q-22 -34 -13 -68.5t43 -50.5t74.5 -2.5t62.5 47.5zM769 373q8 13 3.5 26.5t-17.5 18.5q-14 5 -28.5 -0.5t-21.5 -18.5q-17 -31 13 -45q14 -5 29 0.5t22 18.5zM943 266q-45 -102 -158 -150t-224 -12 q-107 34 -147.5 126.5t6.5 187.5q47 93 151.5 139t210.5 19q111 -29 158.5 -119.5t2.5 -190.5zM1255 426q-9 96 -89 170t-208.5 109t-274.5 21q-223 -23 -369.5 -141.5t-132.5 -264.5q9 -96 89 -170t208.5 -109t274.5 -21q223 23 369.5 141.5t132.5 264.5zM1563 422 q0 -68 -37 -139.5t-109 -137t-168.5 -117.5t-226 -83t-270.5 -31t-275 33.5t-240.5 93t-171.5 151t-65 199.5q0 115 69.5 245t197.5 258q169 169 341.5 236t246.5 -7q65 -64 20 -209q-4 -14 -1 -20t10 -7t14.5 0.5t13.5 3.5l6 2q139 59 246 59t153 -61q45 -63 0 -178 q-2 -13 -4.5 -20t4.5 -12.5t12 -7.5t17 -6q57 -18 103 -47t80 -81.5t34 -116.5zM1489 1046q42 -47 54.5 -108.5t-6.5 -117.5q-8 -23 -29.5 -34t-44.5 -4q-23 8 -34 29.5t-4 44.5q20 63 -24 111t-107 35q-24 -5 -45 8t-25 37q-5 24 8 44.5t37 25.5q60 13 119 -5.5t101 -65.5z M1670 1209q87 -96 112.5 -222.5t-13.5 -241.5q-9 -27 -34 -40t-52 -4t-40 34t-5 52q28 82 10 172t-80 158q-62 69 -148 95.5t-173 8.5q-28 -6 -52 9.5t-30 43.5t9.5 51.5t43.5 29.5q123 26 244 -11.5t208 -134.5z" />
-<glyph unicode="" d="M1133 -34q-171 -94 -368 -94q-196 0 -367 94q138 87 235.5 211t131.5 268q35 -144 132.5 -268t235.5 -211zM638 1394v-485q0 -252 -126.5 -459.5t-330.5 -306.5q-181 215 -181 495q0 187 83.5 349.5t229.5 269.5t325 137zM1536 638q0 -280 -181 -495 q-204 99 -330.5 306.5t-126.5 459.5v485q179 -30 325 -137t229.5 -269.5t83.5 -349.5z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1402 433q-32 -80 -76 -138t-91 -88.5t-99 -46.5t-101.5 -14.5t-96.5 8.5t-86.5 22t-69.5 27.5t-46 22.5l-17 10q-113 -228 -289.5 -359.5t-384.5 -132.5q-19 0 -32 13t-13 32t13 31.5t32 12.5q173 1 322.5 107.5t251.5 294.5q-36 -14 -72 -23t-83 -13t-91 2.5t-93 28.5 t-92 59t-84.5 100t-74.5 146q114 47 214 57t167.5 -7.5t124.5 -56.5t88.5 -77t56.5 -82q53 131 79 291q-7 -1 -18 -2.5t-46.5 -2.5t-69.5 0.5t-81.5 10t-88.5 23t-84 42.5t-75 65t-54.5 94.5t-28.5 127.5q70 28 133.5 36.5t112.5 -1t92 -30t73.5 -50t56 -61t42 -63t27.5 -56 t16 -39.5l4 -16q12 122 12 195q-8 6 -21.5 16t-49 44.5t-63.5 71.5t-54 93t-33 112.5t12 127t70 138.5q73 -25 127.5 -61.5t84.5 -76.5t48 -85t20.5 -89t-0.5 -85.5t-13 -76.5t-19 -62t-17 -42l-7 -15q1 -5 1 -50.5t-1 -71.5q3 7 10 18.5t30.5 43t50.5 58t71 55.5t91.5 44.5 t112 14.5t132.5 -24q-2 -78 -21.5 -141.5t-50 -104.5t-69.5 -71.5t-81.5 -45.5t-84.5 -24t-80 -9.5t-67.5 1t-46.5 4.5l-17 3q-23 -147 -73 -283q6 7 18 18.5t49.5 41t77.5 52.5t99.5 42t117.5 20t129 -23.5t137 -77.5z" />
-<glyph unicode="" horiz-adv-x="1280" d="M1259 283v-66q0 -85 -57.5 -144.5t-138.5 -59.5h-57l-260 -269v269h-529q-81 0 -138.5 59.5t-57.5 144.5v66h1238zM1259 609v-255h-1238v255h1238zM1259 937v-255h-1238v255h1238zM1259 1077v-67h-1238v67q0 84 57.5 143.5t138.5 59.5h846q81 0 138.5 -59.5t57.5 -143.5z " />
-<glyph unicode="" d="M1152 640q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v192h-352q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h352v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198 t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1152 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-352v-192q0 -14 -9 -23t-23 -9q-12 0 -24 10l-319 319q-9 9 -9 23t9 23l320 320q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5v-192h352q13 0 22.5 -9.5t9.5 -22.5zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198 t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" d="M1024 960v-640q0 -26 -19 -45t-45 -19q-20 0 -37 12l-448 320q-27 19 -27 52t27 52l448 320q17 12 37 12q26 0 45 -19t19 -45zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5z M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" d="M1024 640q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5 t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" />
-<glyph unicode="" horiz-adv-x="1664" d="M1023 349l102 -204q-58 -179 -210 -290t-339 -111q-156 0 -288.5 77.5t-210 210t-77.5 288.5q0 181 104.5 330t274.5 211l17 -131q-122 -54 -195 -165.5t-73 -244.5q0 -185 131.5 -316.5t316.5 -131.5q126 0 232.5 65t165 175.5t49.5 236.5zM1571 249l58 -114l-256 -128 q-13 -7 -29 -7q-40 0 -57 35l-239 477h-472q-24 0 -42.5 16.5t-21.5 40.5l-96 779q-2 16 6 42q14 51 57 82.5t97 31.5q66 0 113 -47t47 -113q0 -69 -52 -117.5t-120 -41.5l37 -289h423v-128h-407l16 -128h455q40 0 57 -35l228 -455z" />
-<glyph unicode="" d="M1254 899q16 85 -21 132q-52 65 -187 45q-17 -3 -41 -12.5t-57.5 -30.5t-64.5 -48.5t-59.5 -70t-44.5 -91.5q80 7 113.5 -16t26.5 -99q-5 -52 -52 -143q-43 -78 -71 -99q-44 -32 -87 14q-23 24 -37.5 64.5t-19 73t-10 84t-8.5 71.5q-23 129 -34 164q-12 37 -35.5 69 t-50.5 40q-57 16 -127 -25q-54 -32 -136.5 -106t-122.5 -102v-7q16 -8 25.5 -26t21.5 -20q21 -3 54.5 8.5t58 10.5t41.5 -30q11 -18 18.5 -38.5t15 -48t12.5 -40.5q17 -46 53 -187q36 -146 57 -197q42 -99 103 -125q43 -12 85 -1.5t76 31.5q131 77 250 237 q104 139 172.5 292.5t82.5 226.5zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1152" d="M1152 704q0 -191 -94.5 -353t-256.5 -256.5t-353 -94.5h-160q-14 0 -23 9t-9 23v611l-215 -66q-3 -1 -9 -1q-10 0 -19 6q-13 10 -13 26v128q0 23 23 31l233 71v93l-215 -66q-3 -1 -9 -1q-10 0 -19 6q-13 10 -13 26v128q0 23 23 31l233 71v250q0 14 9 23t23 9h160 q14 0 23 -9t9 -23v-181l375 116q15 5 28 -5t13 -26v-128q0 -23 -23 -31l-393 -121v-93l375 116q15 5 28 -5t13 -26v-128q0 -23 -23 -31l-393 -121v-487q188 13 318 151t130 328q0 14 9 23t23 9h160q14 0 23 -9t9 -23z" />
-<glyph unicode="" horiz-adv-x="1408" d="M1152 736v-64q0 -14 -9 -23t-23 -9h-352v-352q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v352h-352q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h352v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-352h352q14 0 23 -9t9 -23zM1280 288v832q0 66 -47 113t-113 47h-832 q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113zM1408 1120v-832q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q119 0 203.5 -84.5t84.5 -203.5z" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-<glyph unicode="" horiz-adv-x="1792" />
-</font>
-</defs></svg>
\ No newline at end of file
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.ttf b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.ttf
deleted file mode 100644
index e89738de5eaf8fca33a2f2cdc5cb4929caa62b71..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 80652
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
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.woff b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/fonts/fontawesome-webfont.woff
deleted file mode 100644
index 8c1748aab7a790d510fb3f42a8a8971d96efa79d..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 44432
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
VoBs<Y3))5i00001Qg^x6008Up2bBN-
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/js/theme.js b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/js/theme.js
deleted file mode 100644
index 60520cc3ad..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/static/js/theme.js
+++ /dev/null
@@ -1,47 +0,0 @@
-$( document ).ready(function() {
- // Shift nav in mobile when clicking the menu.
- $(document).on('click', "[data-toggle='wy-nav-top']", function() {
- $("[data-toggle='wy-nav-shift']").toggleClass("shift");
- $("[data-toggle='rst-versions']").toggleClass("shift");
- });
- // Close menu when you click a link.
- $(document).on('click', ".wy-menu-vertical .current ul li a", function() {
- $("[data-toggle='wy-nav-shift']").removeClass("shift");
- $("[data-toggle='rst-versions']").toggleClass("shift");
- });
- $(document).on('click', "[data-toggle='rst-current-version']", function() {
- $("[data-toggle='rst-versions']").toggleClass("shift-up");
- });
- // Make tables responsive
- $("table.docutils:not(.field-list)").wrap("<div class='wy-table-responsive'></div>");
-});
-
-window.SphinxRtdTheme = (function (jquery) {
- var stickyNav = (function () {
- var navBar,
- win,
- stickyNavCssClass = 'stickynav',
- applyStickNav = function () {
- if (navBar.height() <= win.height()) {
- navBar.addClass(stickyNavCssClass);
- } else {
- navBar.removeClass(stickyNavCssClass);
- }
- },
- enable = function () {
- applyStickNav();
- win.on('resize', applyStickNav);
- },
- init = function () {
- navBar = jquery('nav.wy-nav-side:first');
- win = jquery(window);
- };
- jquery(init);
- return {
- enable : enable
- };
- }());
- return {
- StickyNav : stickyNav
- };
-}($));
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/theme.conf b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/theme.conf
deleted file mode 100644
index dcfbf8c224..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/theme.conf
+++ /dev/null
@@ -1,8 +0,0 @@
-[theme]
-inherit = basic
-stylesheet = css/theme.css
-
-[options]
-typekit_id = hiw1hhg
-analytics_id =
-sticky_navigation = False
diff --git a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/versions.html b/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/versions.html
deleted file mode 100644
index 8b3eb79d25..0000000000
--- a/doc/source/ext/sphinx_rtd_theme/sphinx_rtd_theme/versions.html
+++ /dev/null
@@ -1,37 +0,0 @@
-{% if READTHEDOCS %}
-{# Add rst-badge after rst-versions for small badge style. #}
- <div class="rst-versions" data-toggle="rst-versions" role="note" aria-label="versions">
- <span class="rst-current-version" data-toggle="rst-current-version">
- <span class="fa fa-book"> Read the Docs</span>
- v: {{ current_version }}
- <span class="fa fa-caret-down"></span>
- </span>
- <div class="rst-other-versions">
- <dl>
- <dt>Versions</dt>
- {% for slug, url in versions %}
- <dd><a href="{{ url }}">{{ slug }}</a></dd>
- {% endfor %}
- </dl>
- <dl>
- <dt>Downloads</dt>
- {% for type, url in downloads %}
- <dd><a href="{{ url }}">{{ type }}</a></dd>
- {% endfor %}
- </dl>
- <dl>
- <dt>On Read the Docs</dt>
- <dd>
- <a href="//{{ PRODUCTION_DOMAIN }}/projects/{{ slug }}/?fromdocs={{ slug }}">Project Home</a>
- </dd>
- <dd>
- <a href="//{{ PRODUCTION_DOMAIN }}/builds/{{ slug }}/?fromdocs={{ slug }}">Builds</a>
- </dd>
- </dl>
- <hr/>
- Free document hosting provided by <a href="http://www.readthedocs.org">Read the Docs</a>.
-
- </div>
- </div>
-{% endif %}
-
diff --git a/doc/source/index.rst b/doc/source/index.rst
index 84a02b962e..5faf83da49 100644
--- a/doc/source/index.rst
+++ b/doc/source/index.rst
@@ -5,7 +5,7 @@ python-for-android is an open source build tool to let you package
Python code into standalone android APKs. These can be passed around,
installed, or uploaded to marketplaces such as the Play Store just
like any other Android app. This tool was originally developed for the
-`Kivy cross-platform graphical framework <http://kivy.org/#home>`_,
+`Kivy cross-platform graphical framework <https://kivy.org/>`_,
but now supports multiple bootstraps and can be easily extended to
package other types of Python apps for Android.
@@ -30,7 +30,6 @@ Contents
buildoptions
commands
apis
- launcher
distutils
recipes
bootstraps
diff --git a/doc/source/launcher.rst b/doc/source/launcher.rst
deleted file mode 100644
index cc4ccf6ea5..0000000000
--- a/doc/source/launcher.rst
+++ /dev/null
@@ -1,101 +0,0 @@
-.. _launcher:
-
-Launcher
-========
-
-The Kivy Launcher is an Android application that can run any Kivy app
-stored in the `kivy` folder on the SD Card. You can download the latest stable
-version for your android device from the
-`Play Store <https://play.google.com/store/apps/details?id=org.kivy.pygame>`_.
-
-The stable launcher comes with various Python packages and
-permissions, usually listed in the description in the store. Those
-aren't always enough for an application to run or even launch if you
-work with other dependencies that are not packaged.
-
-The Kivy Launcher is intended for quick and simple testing. For
-anything more advanced we recommend building your own APK with
-python-for-android.
-
-Building
---------
-
-The Kivy Launcher is built using python-for-android. To get the most recent
-versions of packages you need to clean them first, so that the packager won't
-grab an old (cached) package instead of a fresh one.
-
-.. highlight:: none
-
-::
-
- p4a clean_download_cache requirements
- p4a clean_dists && p4a clean_builds
- p4a apk --requirements=requirements \
- --permission PERMISSION \
- --package=the.package.name \
- --name="App name" \
- --version=x.y.z \
- --android_api XY \
- --bootstrap=sdl2 \
- --launcher \
- --minsdk 13
-
-.. note::
-
- `--minsdk 13` is necessary for the new toolchain, otherwise you'll be able
- to run apps only in `landscape` orientation.
-
-.. warning::
-
- Do not use any of `--private`, `--public`, `--dir` or other arguments for
- adding `main.py` or `main.pyc` to the app. The argument `--launcher` is
- above them and tells the p4a to build the launcher version of the APK.
-
-Usage
------
-
-Once the launcher is installed, you need to create a folder in your
-external storage directory (e.g. ``/storage/emulated/0`` or
-``/sdcard``) - this is normally your 'home' directory in a file
-browser. Each new folder inside `kivy` represents a
-separate application::
-
- /sdcard/kivy/<yourapplication>
-
-Each application folder must contain an
-`android.txt` file. The file has to contain three basic
-lines::
-
- title=<Application Title>
- author=<Your Name>
- orientation=<portrait|landscape>
-
-The file is editable so you can change for example orientation or
-name. These are the only options dynamically configurable here,
-although when the app runs you can call the Android API with PyJNIus
-to change other settings.
-
-After you set your `android.txt` file, you can now run the launcher
-and start any available app from the list.
-
-To differentiate between apps in ``/sdcard/kivy``, you can include an icon
-named ``icon.png`` in the folder. The icon should be a square.
-
-Release on the market
----------------------
-
-Launcher is released on Google Play with each new Kivy stable
-branch. The master branch is not suitable for a regular user because
-it changes quickly and needs testing.
-
-Source code
------------
-
-.. |kivy| replace:: sdl2 org.kivy.android
-
-.. _sdl2:
- https://github.com/kivy/python-for-android/tree/master/\
- pythonforandroid/bootstraps/sdl2/build/src/org/kivy/android
-
-If you feel confident, feel free to improve the launcher. You can find the
-source code at |renpy|_ or at |kivy|_.
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 2e3fe21108..46612e5ad5 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -366,5 +366,4 @@ See the other pages of this doc for more information on specific topics:
- :doc:`bootstraps`
- :doc:`apis`
- :doc:`troubleshooting`
-- :doc:`launcher`
- :doc:`contribute`
diff --git a/doc/source/recipes.rst b/doc/source/recipes.rst
index c3a97aab4f..0a2a736592 100644
--- a/doc/source/recipes.rst
+++ b/doc/source/recipes.rst
@@ -486,7 +486,7 @@ how to create your own Recipe subclass.
.. autoclass:: toolchain.Recipe
:members:
- :member-order: = 'bysource'
+ :member-order: bysource
diff --git a/doc/source/services.rst b/doc/source/services.rst
index 8b5d9a7851..c71b035a76 100644
--- a/doc/source/services.rst
+++ b/doc/source/services.rst
@@ -15,8 +15,8 @@ the ``android:process`` attribute of the ``AndroidManifest.xml`` file.
This is not the default behavior, see `Android service documentation
<https://developer.android.com/guide/topics/manifest/service-element>`__.
You can communicate with the service process from your app using e.g.
-`osc <https://pypi.python.org/pypi/python-osc>`__ or (a heavier option)
-`twisted <https://twistedmatrix.com/trac/>`__.
+`osc <https://pypi.org/project/python-osc/>`__ or (a heavier option)
+`twisted <https://twisted.org/>`__.
Service creation
----------------
diff --git a/doc/source/troubleshooting.rst b/doc/source/troubleshooting.rst
index 4d04c9954e..d9f309db9e 100644
--- a/doc/source/troubleshooting.rst
+++ b/doc/source/troubleshooting.rst
@@ -12,7 +12,7 @@ in the compilation and packaging steps.
If reporting a problem by email or Discord, it is usually helpful to
include this full log, e.g. via a `pastebin
-<http://paste.ubuntu.com/>`_ or `Github gist
+<https://pastebin.ubuntu.com/>`_ or `Github gist
<https://gist.github.com/>`_.
Getting help
@@ -64,9 +64,9 @@ can also do other debugging tasks such as ``python-for-android adb
devices`` to get the list of connected devices.
For further information, see the Android docs on `adb
-<http://developer.android.com/intl/zh-cn/tools/help/adb.html>`_, and
+<https://developer.android.com/tools/adb>`_, and
on `logcat
-<http://developer.android.com/intl/zh-cn/tools/help/logcat.html>`_ in
+<https://developer.android.com/tools/logcat>`_ in
particular.
Unpacking an APK
diff --git a/setup.py b/setup.py
index 9ced788ea8..f358bde7bc 100644
--- a/setup.py
+++ b/setup.py
@@ -95,6 +95,9 @@ def recursively_include(results, directory, patterns):
url='https://github.com/kivy/python-for-android',
license='MIT',
install_requires=install_reqs,
+ extras_require={
+ "docs": ["sphinx", "sphinx-rtd-theme"],
+ },
entry_points={
'console_scripts': [
'python-for-android = pythonforandroid.entrypoints:main',
From d8c39478c22c1974219faaa9ff9150765acc52e0 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Mon, 6 Nov 2023 19:20:32 +0100
Subject: [PATCH 005/158] Remove `distutils` usage, as is not available anymore
on Python `3.12` (#2912)
* Remove distutils usage, as is not available anymore on Python 3.12
* Updated testapps to use setuptools instead of distutils
---
pythonforandroid/archs.py | 4 +-
.../bootstraps/common/build/build.py | 7 +--
pythonforandroid/recipe.py | 7 ++-
pythonforandroid/recommendations.py | 27 ++++-----
pythonforandroid/toolchain.py | 22 +++-----
pythonforandroid/util.py | 35 ++++++++++++
setup.py | 2 +-
testapps/on_device_unit_tests/setup.py | 4 +-
.../setup_testapp_python3_sqlite_openssl.py | 4 +-
testapps/setup_vispy.py | 4 +-
testapps/testlauncher_setup/sdl2.py | 3 +-
testapps/testlauncherreboot_setup/sdl2.py | 3 +-
tests/recipes/recipe_lib_test.py | 24 ++++----
tests/recipes/test_icu.py | 12 ++--
tests/recipes/test_libgeos.py | 4 +-
tests/recipes/test_libmysqlclient.py | 4 +-
tests/recipes/test_libpq.py | 4 +-
tests/recipes/test_libvorbis.py | 4 +-
tests/recipes/test_openal.py | 12 ++--
tests/recipes/test_openssl.py | 4 +-
tests/recipes/test_pandas.py | 8 +--
tests/recipes/test_pyicu.py | 8 +--
tests/recipes/test_python3.py | 12 ++--
tests/test_archs.py | 56 +++++++++----------
tests/test_bootstrap.py | 11 ++--
tests/test_build.py | 2 +-
tests/test_recipe.py | 24 ++++++--
tests/test_recommendations.py | 11 +++-
tests/test_util.py | 43 ++++++++++++++
29 files changed, 223 insertions(+), 142 deletions(-)
diff --git a/pythonforandroid/archs.py b/pythonforandroid/archs.py
index b960ca6b4d..b065174592 100644
--- a/pythonforandroid/archs.py
+++ b/pythonforandroid/archs.py
@@ -1,7 +1,7 @@
-from distutils.spawn import find_executable
from os import environ
from os.path import join
from multiprocessing import cpu_count
+import shutil
from pythonforandroid.recipe import Recipe
from pythonforandroid.util import BuildInterruptingException, build_platform
@@ -172,7 +172,7 @@ def get_env(self, with_flags_in_cc=True):
# Compiler: `CC` and `CXX` (and make sure that the compiler exists)
env['PATH'] = self.ctx.env['PATH']
- cc = find_executable(self.clang_exe, path=env['PATH'])
+ cc = shutil.which(self.clang_exe, path=env['PATH'])
if cc is None:
print('Searching path are: {!r}'.format(env['PATH']))
raise BuildInterruptingException(
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index 0b6b9832f0..59c38b609c 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -17,11 +17,10 @@
import tempfile
import time
-from distutils.version import LooseVersion
from fnmatch import fnmatch
import jinja2
-from pythonforandroid.util import rmdir, ensure_dir
+from pythonforandroid.util import rmdir, ensure_dir, max_build_tool_version
def get_dist_info_for(key, error_if_missing=True):
@@ -512,9 +511,7 @@ def make_package(args):
# Try to build with the newest available build tools
ignored = {".DS_Store", ".ds_store"}
build_tools_versions = [x for x in listdir(join(sdk_dir, 'build-tools')) if x not in ignored]
- build_tools_versions = sorted(build_tools_versions,
- key=LooseVersion)
- build_tools_version = build_tools_versions[-1]
+ build_tools_version = max_build_tool_version(build_tools_versions)
# Folder name for launcher (used by SDL2 bootstrap)
url_scheme = 'kivy'
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index fa0bb4e790..dc53fd4337 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -16,6 +16,9 @@
from urlparse import urlparse
except ImportError:
from urllib.parse import urlparse
+
+import packaging.version
+
from pythonforandroid.logger import (
logger, info, warning, debug, shprint, info_main)
from pythonforandroid.util import (
@@ -1145,8 +1148,8 @@ def link_root(self):
@property
def major_minor_version_string(self):
- from distutils.version import LooseVersion
- return '.'.join([str(v) for v in LooseVersion(self.version).version[:2]])
+ parsed_version = packaging.version.parse(self.version)
+ return f"{parsed_version.major}.{parsed_version.minor}"
def create_python_bundle(self, dirn, arch):
"""
diff --git a/pythonforandroid/recommendations.py b/pythonforandroid/recommendations.py
index 040c96234a..cbcfdd2b6e 100644
--- a/pythonforandroid/recommendations.py
+++ b/pythonforandroid/recommendations.py
@@ -1,9 +1,10 @@
"""Simple functions for checking dependency versions."""
import sys
-from distutils.version import LooseVersion
from os.path import join
+import packaging.version
+
from pythonforandroid.logger import info, warning
from pythonforandroid.util import BuildInterruptingException
@@ -59,9 +60,9 @@ def check_ndk_version(ndk_dir):
rewrote to raise an exception in case that an NDK version lower than
the minimum supported is detected.
"""
- version = read_ndk_version(ndk_dir)
+ ndk_version = read_ndk_version(ndk_dir)
- if version is None:
+ if ndk_version is None:
warning(READ_ERROR_NDK_MESSAGE.format(ndk_dir=ndk_dir))
warning(
ENSURE_RIGHT_NDK_MESSAGE.format(
@@ -81,16 +82,11 @@ def check_ndk_version(ndk_dir):
minor_to_letter.update(
{n + 1: chr(i) for n, i in enumerate(range(ord('b'), ord('b') + 25))}
)
-
- major_version = version.version[0]
- letter_version = minor_to_letter[version.version[1]]
- string_version = '{major_version}{letter_version}'.format(
- major_version=major_version, letter_version=letter_version
- )
+ string_version = f"{ndk_version.major}{minor_to_letter[ndk_version.minor]}"
info(CURRENT_NDK_VERSION_MESSAGE.format(ndk_version=string_version))
- if major_version < MIN_NDK_VERSION:
+ if ndk_version.major < MIN_NDK_VERSION:
raise BuildInterruptingException(
NDK_LOWER_THAN_SUPPORTED_MESSAGE.format(
min_supported=MIN_NDK_VERSION, ndk_url=NDK_DOWNLOAD_URL
@@ -104,7 +100,7 @@ def check_ndk_version(ndk_dir):
)
),
)
- elif major_version > MAX_NDK_VERSION:
+ elif ndk_version.major > MAX_NDK_VERSION:
warning(
RECOMMENDED_NDK_VERSION_MESSAGE.format(
recommended_ndk_version=RECOMMENDED_NDK_VERSION
@@ -130,9 +126,9 @@ def read_ndk_version(ndk_dir):
return
# Line should have the form "Pkg.Revision = ..."
- ndk_version = LooseVersion(line.split('=')[-1].strip())
+ unparsed_ndk_version = line.split('=')[-1].strip()
- return ndk_version
+ return packaging.version.parse(unparsed_ndk_version)
MIN_TARGET_API = 30
@@ -191,8 +187,9 @@ def check_ndk_api(ndk_api, android_api):
MIN_PYTHON_MAJOR_VERSION = 3
MIN_PYTHON_MINOR_VERSION = 6
-MIN_PYTHON_VERSION = LooseVersion('{major}.{minor}'.format(major=MIN_PYTHON_MAJOR_VERSION,
- minor=MIN_PYTHON_MINOR_VERSION))
+MIN_PYTHON_VERSION = packaging.version.Version(
+ f"{MIN_PYTHON_MAJOR_VERSION}.{MIN_PYTHON_MINOR_VERSION}"
+)
PY2_ERROR_TEXT = (
'python-for-android no longer supports running under Python 2. Either upgrade to '
'Python {min_version} or higher (recommended), or revert to python-for-android 2019.07.08.'
diff --git a/pythonforandroid/toolchain.py b/pythonforandroid/toolchain.py
index 7a5461f30d..1347038b8b 100644
--- a/pythonforandroid/toolchain.py
+++ b/pythonforandroid/toolchain.py
@@ -24,7 +24,7 @@
from pythonforandroid.checkdependencies import check
check()
-from packaging.version import Version, InvalidVersion
+from packaging.version import Version
import sh
from pythonforandroid import __version__
@@ -41,7 +41,12 @@
from pythonforandroid.recommendations import (
RECOMMENDED_NDK_API, RECOMMENDED_TARGET_API, print_recommendations)
from pythonforandroid.util import (
- current_directory, BuildInterruptingException, load_source, rmdir)
+ current_directory,
+ BuildInterruptingException,
+ load_source,
+ rmdir,
+ max_build_tool_version,
+)
user_dir = dirname(realpath(os.path.curdir))
toolchain_dir = dirname(__file__)
@@ -1009,18 +1014,7 @@ def _build_package(self, args, package_type):
self.hook("before_apk_assemble")
build_tools_versions = os.listdir(join(ctx.sdk_dir,
'build-tools'))
-
- def sort_key(version_text):
- try:
- # Historically, Android build release candidates have had
- # spaces in the version number.
- return Version(version_text.replace(" ", ""))
- except InvalidVersion:
- # Put badly named versions at worst position.
- return Version("0")
-
- build_tools_versions.sort(key=sort_key)
- build_tools_version = build_tools_versions[-1]
+ build_tools_version = max_build_tool_version(build_tools_versions)
info(('Detected highest available build tools '
'version to be {}').format(build_tools_version))
diff --git a/pythonforandroid/util.py b/pythonforandroid/util.py
index af363b2e3f..2738d59990 100644
--- a/pythonforandroid/util.py
+++ b/pythonforandroid/util.py
@@ -8,6 +8,8 @@
import shutil
from tempfile import mkdtemp
+import packaging.version
+
from pythonforandroid.logger import (logger, Err_Fore, error, info)
LOGGER = logging.getLogger("p4a.util")
@@ -128,3 +130,36 @@ def move(source, destination):
def touch(filename):
Path(filename).touch()
+
+
+def build_tools_version_sort_key(
+ version_string: str,
+) -> packaging.version.Version:
+ """
+ Returns a packaging.version.Version object for comparison purposes.
+ It includes canonicalization of the version string to allow for
+ comparison of versions with spaces in them (historically, RC candidates)
+
+ If the version string is invalid, it returns a version object with
+ version 0, which will be sorted at worst position.
+ """
+
+ try:
+ # Historically, Android build release candidates have had
+ # spaces in the version number.
+ return packaging.version.Version(version_string.replace(" ", ""))
+ except packaging.version.InvalidVersion:
+ # Put badly named versions at worst position.
+ return packaging.version.Version("0")
+
+
+def max_build_tool_version(
+ build_tools_versions: list,
+) -> str:
+ """
+ Returns the maximum build tools version from a list of build tools
+ versions. It uses the :meth:`build_tools_version_sort_key` function to
+ canonicalize the version strings and then returns the maximum version.
+ """
+
+ return max(build_tools_versions, key=build_tools_version_sort_key)
diff --git a/setup.py b/setup.py
index f358bde7bc..8099647a93 100644
--- a/setup.py
+++ b/setup.py
@@ -22,7 +22,7 @@
install_reqs = [
'appdirs', 'colorama>=0.3.3', 'jinja2',
'sh>=1.10, <2.0; sys_platform!="win32"',
- 'build', 'toml', 'packaging',
+ 'build', 'toml', 'packaging', 'setuptools'
]
# (build and toml are used by pythonpackage.py)
diff --git a/testapps/on_device_unit_tests/setup.py b/testapps/on_device_unit_tests/setup.py
index 2efeff0cc0..18fd82af75 100644
--- a/testapps/on_device_unit_tests/setup.py
+++ b/testapps/on_device_unit_tests/setup.py
@@ -32,8 +32,8 @@
import os
import sys
-from distutils.core import setup
-from setuptools import find_packages
+
+from setuptools import setup, find_packages
# define a basic test app, which can be override passing the proper args to cli
options = {
diff --git a/testapps/setup_testapp_python3_sqlite_openssl.py b/testapps/setup_testapp_python3_sqlite_openssl.py
index 0dfd375bba..c6360679f1 100644
--- a/testapps/setup_testapp_python3_sqlite_openssl.py
+++ b/testapps/setup_testapp_python3_sqlite_openssl.py
@@ -1,6 +1,4 @@
-
-from distutils.core import setup
-from setuptools import find_packages
+from setuptools import setup, find_packages
options = {'apk': {'requirements': 'requests,peewee,sdl2,pyjnius,kivy,python3',
'android-api': 27,
diff --git a/testapps/setup_vispy.py b/testapps/setup_vispy.py
index cbc909627b..f59d057b0b 100644
--- a/testapps/setup_vispy.py
+++ b/testapps/setup_vispy.py
@@ -1,6 +1,4 @@
-
-from distutils.core import setup
-from setuptools import find_packages
+from setuptools import setup, find_packages
options = {'apk': {'debug': None,
'requirements': 'python3,vispy',
diff --git a/testapps/testlauncher_setup/sdl2.py b/testapps/testlauncher_setup/sdl2.py
index 11f1a3785a..2e300277b7 100644
--- a/testapps/testlauncher_setup/sdl2.py
+++ b/testapps/testlauncher_setup/sdl2.py
@@ -1,5 +1,4 @@
-from distutils.core import setup
-from setuptools import find_packages
+from setuptools import setup
options = {'apk': {'debug': None,
'bootstrap': 'sdl2',
diff --git a/testapps/testlauncherreboot_setup/sdl2.py b/testapps/testlauncherreboot_setup/sdl2.py
index 30d79cc633..8ea0d43c4c 100644
--- a/testapps/testlauncherreboot_setup/sdl2.py
+++ b/testapps/testlauncherreboot_setup/sdl2.py
@@ -22,11 +22,10 @@
# pylint: disable=import-error,no-name-in-module
from subprocess import Popen
-from distutils.core import setup
from os import listdir
from os.path import join, dirname, abspath, exists
from pprint import pprint
-from setuptools import find_packages
+from setuptools import setup, find_packages
ROOT = dirname(abspath(__file__))
LAUNCHER = join(ROOT, 'launcherapp')
diff --git a/tests/recipes/recipe_lib_test.py b/tests/recipes/recipe_lib_test.py
index d1b058206e..1f57fe23c1 100644
--- a/tests/recipes/recipe_lib_test.py
+++ b/tests/recipes/recipe_lib_test.py
@@ -35,10 +35,10 @@ def __init__(self, *args, **kwargs):
@mock.patch("pythonforandroid.recipe.Recipe.check_recipe_choices")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_get_recipe_env(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_check_recipe_choices,
):
@@ -46,7 +46,7 @@ def test_get_recipe_env(
Test that get_recipe_env contains some expected arch flags and that
some internal methods has been called.
"""
- mock_find_executable.return_value = self.expected_compiler.format(
+ mock_shutil_which.return_value = self.expected_compiler.format(
android_ndk=self.ctx._ndk_dir, system=system().lower()
)
mock_check_recipe_choices.return_value = sorted(
@@ -67,19 +67,19 @@ def test_get_recipe_env(
# make sure that the mocked methods are actually called
mock_ensure_dir.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
mock_check_recipe_choices.assert_called()
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
):
- mock_find_executable.return_value = self.expected_compiler.format(
+ mock_shutil_which.return_value = self.expected_compiler.format(
android_ndk=self.ctx._ndk_dir, system=system().lower()
)
@@ -101,7 +101,7 @@ def test_build_arch(
mock_make.assert_called()
mock_ensure_dir.assert_called()
mock_current_directory.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
class BaseTestForCmakeRecipe(BaseTestForMakeRecipe):
@@ -116,14 +116,14 @@ class BaseTestForCmakeRecipe(BaseTestForMakeRecipe):
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
):
- mock_find_executable.return_value = self.expected_compiler.format(
+ mock_shutil_which.return_value = self.expected_compiler.format(
android_ndk=self.ctx._ndk_dir, system=system().lower()
)
@@ -141,4 +141,4 @@ def test_build_arch(
mock_make.assert_called()
mock_ensure_dir.assert_called()
mock_current_directory.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
diff --git a/tests/recipes/test_icu.py b/tests/recipes/test_icu.py
index b928b99bf5..239b99e4c1 100644
--- a/tests/recipes/test_icu.py
+++ b/tests/recipes/test_icu.py
@@ -33,17 +33,17 @@ def test_get_recipe_dir(self):
@mock.patch("pythonforandroid.bootstrap.sh.Command")
@mock.patch("pythonforandroid.recipes.icu.sh.make")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_sh_make,
mock_sh_command,
mock_chdir,
mock_makedirs,
):
- mock_find_executable.return_value = os.path.join(
+ mock_shutil_which.return_value = os.path.join(
self.ctx._ndk_dir,
f"toolchains/llvm/prebuilt/{self.ctx.ndk.host_tag}/bin/clang",
)
@@ -89,10 +89,10 @@ def test_build_arch(
)
mock_makedirs.assert_called()
- mock_find_executable.assert_called_once()
+ mock_shutil_which.assert_called_once()
self.assertEqual(
- mock_find_executable.call_args[0][0],
- mock_find_executable.return_value,
+ mock_shutil_which.call_args[0][0],
+ mock_shutil_which.return_value,
)
@mock.patch("pythonforandroid.recipes.icu.sh.cp")
diff --git a/tests/recipes/test_libgeos.py b/tests/recipes/test_libgeos.py
index d819825294..7a8b4258dc 100644
--- a/tests/recipes/test_libgeos.py
+++ b/tests/recipes/test_libgeos.py
@@ -12,10 +12,10 @@ class TestLibgeosRecipe(BaseTestForCmakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.util.makedirs")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_makedirs,
diff --git a/tests/recipes/test_libmysqlclient.py b/tests/recipes/test_libmysqlclient.py
index e484393398..4c85dc92e2 100644
--- a/tests/recipes/test_libmysqlclient.py
+++ b/tests/recipes/test_libmysqlclient.py
@@ -13,10 +13,10 @@ class TestLibmysqlclientRecipe(BaseTestForCmakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.recipes.libmysqlclient.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_cp,
diff --git a/tests/recipes/test_libpq.py b/tests/recipes/test_libpq.py
index c4ae38a886..5e2f9f3d61 100644
--- a/tests/recipes/test_libpq.py
+++ b/tests/recipes/test_libpq.py
@@ -13,10 +13,10 @@ class TestLibpqRecipe(BaseTestForMakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.recipes.libpq.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_cp,
diff --git a/tests/recipes/test_libvorbis.py b/tests/recipes/test_libvorbis.py
index 663c1ccfc1..d8aed7b728 100644
--- a/tests/recipes/test_libvorbis.py
+++ b/tests/recipes/test_libvorbis.py
@@ -14,10 +14,10 @@ class TestLibvorbisRecipe(BaseTestForMakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.recipes.libvorbis.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_cp,
diff --git a/tests/recipes/test_openal.py b/tests/recipes/test_openal.py
index 21f3196798..a03d5cd271 100644
--- a/tests/recipes/test_openal.py
+++ b/tests/recipes/test_openal.py
@@ -14,17 +14,17 @@ class TestOpenalRecipe(BaseTestForCmakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.recipes.openal.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_prebuild_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_cp,
mock_sh_make,
mock_sh_cmake,
):
- mock_find_executable.return_value = (
+ mock_shutil_which.return_value = (
"/opt/android/android-ndk/toolchains/"
"llvm/prebuilt/linux-x86_64/bin/clang"
)
@@ -33,7 +33,7 @@ def test_prebuild_arch(
# make sure that the mocked methods are actually called
mock_ensure_dir.assert_called()
mock_current_directory.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
mock_sh_cp.assert_called()
mock_sh_make.assert_called()
mock_sh_cmake.assert_called()
@@ -41,10 +41,10 @@ def test_prebuild_arch(
@mock.patch("pythonforandroid.recipes.openal.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_cp,
diff --git a/tests/recipes/test_openssl.py b/tests/recipes/test_openssl.py
index 861e73bd39..f7ed362f68 100644
--- a/tests/recipes/test_openssl.py
+++ b/tests/recipes/test_openssl.py
@@ -14,10 +14,10 @@ class TestOpensslRecipe(BaseTestForMakeRecipe, unittest.TestCase):
@mock.patch("pythonforandroid.recipes.openssl.sh.patch")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
mock_sh_patch,
diff --git a/tests/recipes/test_pandas.py b/tests/recipes/test_pandas.py
index 410c2c43c0..b8366863fe 100644
--- a/tests/recipes/test_pandas.py
+++ b/tests/recipes/test_pandas.py
@@ -14,10 +14,10 @@ class TestPandasRecipe(RecipeCtx, unittest.TestCase):
@mock.patch("pythonforandroid.recipe.Recipe.check_recipe_choices")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_get_recipe_env(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_check_recipe_choices,
):
@@ -27,7 +27,7 @@ def test_get_recipe_env(
returns the expected flags
"""
- mock_find_executable.return_value = (
+ mock_shutil_which.return_value = (
"/opt/android/android-ndk/toolchains/"
"llvm/prebuilt/linux-x86_64/bin/clang"
)
@@ -43,5 +43,5 @@ def test_get_recipe_env(
# make sure that the mocked methods are actually called
mock_ensure_dir.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
mock_check_recipe_choices.assert_called()
diff --git a/tests/recipes/test_pyicu.py b/tests/recipes/test_pyicu.py
index 8720a5f4d7..5babfbc114 100644
--- a/tests/recipes/test_pyicu.py
+++ b/tests/recipes/test_pyicu.py
@@ -12,10 +12,10 @@ class TestPyIcuRecipe(RecipeCtx, unittest.TestCase):
@mock.patch("pythonforandroid.recipe.Recipe.check_recipe_choices")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_get_recipe_env(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_check_recipe_choices,
):
@@ -26,7 +26,7 @@ def test_get_recipe_env(
"""
icu_recipe = Recipe.get_recipe("icu", self.ctx)
- mock_find_executable.return_value = (
+ mock_shutil_which.return_value = (
"/opt/android/android-ndk/toolchains/"
"llvm/prebuilt/linux-x86_64/bin/clang"
)
@@ -44,5 +44,5 @@ def test_get_recipe_env(
# make sure that the mocked methods are actually called
mock_ensure_dir.assert_called()
- mock_find_executable.assert_called()
+ mock_shutil_which.assert_called()
mock_check_recipe_choices.assert_called()
diff --git a/tests/recipes/test_python3.py b/tests/recipes/test_python3.py
index f22b2ffdb2..f1d652b6c1 100644
--- a/tests/recipes/test_python3.py
+++ b/tests/recipes/test_python3.py
@@ -60,10 +60,10 @@ def test_compile_python_files(self, mock_subprocess):
)
@mock.patch("pythonforandroid.recipe.Recipe.check_recipe_choices")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_get_recipe_env(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_check_recipe_choices,
):
"""
@@ -71,7 +71,7 @@ def test_get_recipe_env(
:meth:`~pythonforandroid.recipes.python3.Python3Recipe.get_recipe_env`
returns the expected flags
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_check_recipe_choices.return_value = sorted(
self.ctx.recipe_build_order
)
@@ -91,13 +91,13 @@ def test_set_libs_flags(self):
# and `set_libs_flags`, since these calls are tested separately
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.util.makedirs")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_build_arch(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_makedirs,
mock_chdir):
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
# specific `build_arch` mocks
with mock.patch(
diff --git a/tests/test_archs.py b/tests/test_archs.py
index 44ff6237ba..58530886ff 100644
--- a/tests/test_archs.py
+++ b/tests/test_archs.py
@@ -92,9 +92,9 @@ class TestArchARM(ArchSetUpBaseClass, unittest.TestCase):
will be used to perform tests for :class:`~pythonforandroid.archs.ArchARM`.
"""
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
@mock.patch("pythonforandroid.build.ensure_dir")
- def test_arch_arm(self, mock_ensure_dir, mock_find_executable):
+ def test_arch_arm(self, mock_ensure_dir, mock_shutil_which):
"""
Test that class :class:`~pythonforandroid.archs.ArchARM` returns some
expected attributes and environment variables.
@@ -103,13 +103,13 @@ def test_arch_arm(self, mock_ensure_dir, mock_find_executable):
Here we mock two methods:
- `ensure_dir` because we don't want to create any directory
- - `find_executable` because otherwise we will
+ - `shutil.which` because otherwise we will
get an error when trying to find the compiler (we are setting
some fake paths for our android sdk and ndk so probably will
not exist)
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_ensure_dir.return_value = True
arch = ArchARM(self.ctx)
@@ -126,8 +126,8 @@ def test_arch_arm(self, mock_ensure_dir, mock_find_executable):
expected_env_gcc_keys, set(env.keys()) & expected_env_gcc_keys
)
- # check find_executable calls
- mock_find_executable.assert_called_once_with(
+ # check shutil.which calls
+ mock_shutil_which.assert_called_once_with(
self.expected_compiler, path=environ["PATH"]
)
@@ -163,7 +163,7 @@ def test_arch_arm(self, mock_ensure_dir, mock_find_executable):
self.assertEqual(env["NDK_CCACHE"], "/usr/bin/ccache")
# Check exception in case that CC is not found
- mock_find_executable.return_value = None
+ mock_shutil_which.return_value = None
with self.assertRaises(BuildInterruptingException) as e:
arch.get_env()
self.assertEqual(
@@ -182,10 +182,10 @@ class TestArchARMv7a(ArchSetUpBaseClass, unittest.TestCase):
:class:`~pythonforandroid.archs.ArchARMv7_a`.
"""
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
@mock.patch("pythonforandroid.build.ensure_dir")
def test_arch_armv7a(
- self, mock_ensure_dir, mock_find_executable
+ self, mock_ensure_dir, mock_shutil_which
):
"""
Test that class :class:`~pythonforandroid.archs.ArchARMv7_a` returns
@@ -197,7 +197,7 @@ def test_arch_armv7a(
This has to be done because here we tests the `get_env` with clang
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_ensure_dir.return_value = True
arch = ArchARMv7_a(self.ctx)
@@ -207,8 +207,8 @@ def test_arch_armv7a(
self.assertEqual(arch.target, "armv7a-linux-androideabi21")
env = arch.get_env()
- # check find_executable calls
- mock_find_executable.assert_called_once_with(
+ # check shutil.which calls
+ mock_shutil_which.assert_called_once_with(
self.expected_compiler, path=environ["PATH"]
)
@@ -241,9 +241,9 @@ class TestArchX86(ArchSetUpBaseClass, unittest.TestCase):
will be used to perform tests for :class:`~pythonforandroid.archs.Archx86`.
"""
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
@mock.patch("pythonforandroid.build.ensure_dir")
- def test_arch_x86(self, mock_ensure_dir, mock_find_executable):
+ def test_arch_x86(self, mock_ensure_dir, mock_shutil_which):
"""
Test that class :class:`~pythonforandroid.archs.Archx86` returns
some expected attributes and environment variables.
@@ -255,7 +255,7 @@ def test_arch_x86(self, mock_ensure_dir, mock_find_executable):
which is probably the case. This has to be done because here we
tests the `get_env` with clang
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_ensure_dir.return_value = True
arch = Archx86(self.ctx)
@@ -265,8 +265,8 @@ def test_arch_x86(self, mock_ensure_dir, mock_find_executable):
self.assertEqual(arch.target, "i686-linux-android21")
env = arch.get_env()
- # check find_executable calls
- mock_find_executable.assert_called_once_with(
+ # check shutil.which calls
+ mock_shutil_which.assert_called_once_with(
self.expected_compiler, path=environ["PATH"]
)
@@ -284,10 +284,10 @@ class TestArchX86_64(ArchSetUpBaseClass, unittest.TestCase):
:class:`~pythonforandroid.archs.Archx86_64`.
"""
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
@mock.patch("pythonforandroid.build.ensure_dir")
def test_arch_x86_64(
- self, mock_ensure_dir, mock_find_executable
+ self, mock_ensure_dir, mock_shutil_which
):
"""
Test that class :class:`~pythonforandroid.archs.Archx86_64` returns
@@ -300,7 +300,7 @@ def test_arch_x86_64(
which is probably the case. This has to be done because here we
tests the `get_env` with clang
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_ensure_dir.return_value = True
arch = Archx86_64(self.ctx)
@@ -310,13 +310,13 @@ def test_arch_x86_64(
self.assertEqual(arch.target, "x86_64-linux-android21")
env = arch.get_env()
- # check find_executable calls
- mock_find_executable.assert_called_once_with(
+ # check shutil.which calls
+ mock_shutil_which.assert_called_once_with(
self.expected_compiler, path=environ["PATH"]
)
# For x86_64 we expect some extra cflags in our `environment`
- mock_find_executable.assert_called_once()
+ mock_shutil_which.assert_called_once()
self.assertIn(
" -march=x86-64 -msse4.2 -mpopcnt -m64", env["CFLAGS"]
)
@@ -329,10 +329,10 @@ class TestArchAArch64(ArchSetUpBaseClass, unittest.TestCase):
:class:`~pythonforandroid.archs.ArchAarch_64`.
"""
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
@mock.patch("pythonforandroid.build.ensure_dir")
def test_arch_aarch_64(
- self, mock_ensure_dir, mock_find_executable
+ self, mock_ensure_dir, mock_shutil_which
):
"""
Test that class :class:`~pythonforandroid.archs.ArchAarch_64` returns
@@ -345,7 +345,7 @@ def test_arch_aarch_64(
which is probably the case. This has to be done because here we
tests the `get_env` with clang
"""
- mock_find_executable.return_value = self.expected_compiler
+ mock_shutil_which.return_value = self.expected_compiler
mock_ensure_dir.return_value = True
arch = ArchAarch_64(self.ctx)
@@ -355,8 +355,8 @@ def test_arch_aarch_64(
self.assertEqual(arch.target, "aarch64-linux-android21")
env = arch.get_env()
- # check find_executable calls
- mock_find_executable.assert_called_once_with(
+ # check shutil.which calls
+ mock_shutil_which.assert_called_once_with(
self.expected_compiler, path=environ["PATH"]
)
diff --git a/tests/test_bootstrap.py b/tests/test_bootstrap.py
index 99620fee75..1e17a9bd18 100644
--- a/tests/test_bootstrap.py
+++ b/tests/test_bootstrap.py
@@ -523,15 +523,15 @@ def reset_mocks():
@mock.patch("pythonforandroid.bootstrap.shprint")
@mock.patch("pythonforandroid.bootstrap.sh.Command")
@mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("pythonforandroid.archs.find_executable")
+ @mock.patch("shutil.which")
def test_bootstrap_strip(
self,
- mock_find_executable,
+ mock_shutil_which,
mock_ensure_dir,
mock_sh_command,
mock_sh_print,
):
- mock_find_executable.return_value = os.path.join(
+ mock_shutil_which.return_value = os.path.join(
self.ctx._ndk_dir,
f"toolchains/llvm/prebuilt/{self.ctx.ndk.host_tag}/bin/clang",
)
@@ -544,10 +544,9 @@ def test_bootstrap_strip(
# test that strip_libraries runs with a fake distribution
bs.strip_libraries(arch)
- mock_find_executable.assert_called_once()
self.assertEqual(
- mock_find_executable.call_args[0][0],
- mock_find_executable.return_value,
+ mock_shutil_which.call_args[0][0],
+ mock_shutil_which.return_value,
)
mock_sh_command.assert_called_once_with(
os.path.join(
diff --git a/tests/test_build.py b/tests/test_build.py
index cf9fa7801d..7556d68bbc 100644
--- a/tests/test_build.py
+++ b/tests/test_build.py
@@ -28,7 +28,7 @@ def test_run_pymodules_install_optional_project_dir(self):
def test_strip_if_with_debug_symbols(self):
ctx = mock.Mock()
- ctx.python_recipe.major_minor_version_string = "python3.6"
+ ctx.python_recipe.major_minor_version_string = "3.6"
ctx.get_site_packages_dir.return_value = "test-doesntexist"
ctx.build_dir = "nonexistant_directory"
ctx.archs = ["arm64"]
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index b6e4c99225..a50ca444ab 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -7,7 +7,7 @@
from backports import tempfile
from pythonforandroid.build import Context
-from pythonforandroid.recipe import Recipe, import_recipe
+from pythonforandroid.recipe import Recipe, TargetPythonRecipe, import_recipe
from pythonforandroid.archs import ArchAarch_64
from pythonforandroid.bootstrap import Bootstrap
from test_bootstrap import BaseClassSetupBootstrap
@@ -182,6 +182,20 @@ def test_download_file_scheme_https_oserror(self):
assert m_sleep.call_args_list == expected_call_args_list
+class TestTargetPythonRecipe(unittest.TestCase):
+
+ def test_major_minor_version_string(self):
+ """
+ Test that the major_minor_version_string property returns the correct
+ string.
+ """
+ class DummyTargetPythonRecipe(TargetPythonRecipe):
+ version = '1.2.3'
+
+ recipe = DummyTargetPythonRecipe()
+ assert recipe.major_minor_version_string == '1.2'
+
+
class TestLibraryRecipe(BaseClassSetupBootstrap, unittest.TestCase):
def setUp(self):
"""
@@ -249,10 +263,10 @@ def setUp(self):
self.setUp_distribution_with_bootstrap(self.ctx.bootstrap)
self.ctx.python_recipe = Recipe.get_recipe('python3', self.ctx)
- @mock.patch('pythonforandroid.archs.find_executable')
+ @mock.patch('shutil.which')
@mock.patch('pythonforandroid.build.ensure_dir')
def test_get_recipe_env_with(
- self, mock_ensure_dir, mock_find_executable
+ self, mock_ensure_dir, mock_shutil_which
):
"""
Test that :meth:`~pythonforandroid.recipe.STLRecipe.get_recipe_env`
@@ -266,7 +280,7 @@ def test_get_recipe_env_with(
f"/opt/android/android-ndk/toolchains/"
f"llvm/prebuilt/{self.ctx.ndk.host_tag}/bin/clang"
)
- mock_find_executable.return_value = expected_compiler
+ mock_shutil_which.return_value = expected_compiler
arch = ArchAarch_64(self.ctx)
recipe = Recipe.get_recipe('libgeos', self.ctx)
@@ -274,7 +288,7 @@ def test_get_recipe_env_with(
env = recipe.get_recipe_env(arch)
# check that the mocks have been called
mock_ensure_dir.assert_called()
- mock_find_executable.assert_called_once_with(
+ mock_shutil_which.assert_called_once_with(
expected_compiler, path=self.ctx.env['PATH']
)
self.assertIsInstance(env, dict)
diff --git a/tests/test_recommendations.py b/tests/test_recommendations.py
index df68bbe3e0..443ec52b39 100644
--- a/tests/test_recommendations.py
+++ b/tests/test_recommendations.py
@@ -2,6 +2,7 @@
from os.path import join
from sys import version as py_version
+import packaging.version
from unittest import mock
from pythonforandroid.recommendations import (
check_ndk_api,
@@ -53,7 +54,8 @@ def setUp(self):
@unittest.skipIf(running_in_py2, "`assertLogs` requires Python 3.4+")
@mock.patch("pythonforandroid.recommendations.read_ndk_version")
def test_check_ndk_version_greater_than_recommended(self, mock_read_ndk):
- mock_read_ndk.return_value.version = [MAX_NDK_VERSION + 1, 0, 5232133]
+ _version_string = f"{MIN_NDK_VERSION + 1}.0.5232133"
+ mock_read_ndk.return_value = packaging.version.Version(_version_string)
with self.assertLogs(level="INFO") as cm:
check_ndk_version(self.ndk_dir)
mock_read_ndk.assert_called_once_with(self.ndk_dir)
@@ -76,7 +78,8 @@ def test_check_ndk_version_greater_than_recommended(self, mock_read_ndk):
@mock.patch("pythonforandroid.recommendations.read_ndk_version")
def test_check_ndk_version_lower_than_recommended(self, mock_read_ndk):
- mock_read_ndk.return_value.version = [MIN_NDK_VERSION - 1, 0, 5232133]
+ _version_string = f"{MIN_NDK_VERSION - 1}.0.5232133"
+ mock_read_ndk.return_value = packaging.version.Version(_version_string)
with self.assertRaises(BuildInterruptingException) as e:
check_ndk_version(self.ndk_dir)
self.assertEqual(
@@ -124,7 +127,9 @@ def test_read_ndk_version(self, mock_open_src_prop):
mock_open_src_prop.assert_called_once_with(
join(self.ndk_dir, "source.properties")
)
- assert version == "17.2.4988734"
+ assert version.major == 17
+ assert version.minor == 2
+ assert version.micro == 4988734
@unittest.skipIf(running_in_py2, "`assertLogs` requires Python 3.4+")
@mock.patch("pythonforandroid.recommendations.open")
diff --git a/tests/test_util.py b/tests/test_util.py
index a4d7ea816e..7a60bc73fb 100644
--- a/tests/test_util.py
+++ b/tests/test_util.py
@@ -187,3 +187,46 @@ def test_touch(self):
assert not new_file_path.exists()
util.touch(new_file_path)
assert new_file_path.exists()
+
+ def test_build_tools_version_sort_key(self):
+
+ build_tools_versions = [
+ "26.0.1",
+ "26.0.0",
+ "26.0.2",
+ "32.0.0 rc1",
+ "31.0.0",
+ "999something",
+ ]
+
+ expected_result = [
+ "999something", # invalid version
+ "26.0.0",
+ "26.0.1",
+ "26.0.2",
+ "31.0.0",
+ "32.0.0 rc1",
+ ]
+
+ result = sorted(
+ build_tools_versions, key=util.build_tools_version_sort_key
+ )
+
+ self.assertEqual(result, expected_result)
+
+ def test_max_build_tool_version(self):
+
+ build_tools_versions = [
+ "26.0.1",
+ "26.0.0",
+ "26.0.2",
+ "32.0.0 rc1",
+ "31.0.0",
+ "999something",
+ ]
+
+ expected_result = "32.0.0 rc1"
+
+ result = util.max_build_tool_version(build_tools_versions)
+
+ self.assertEqual(result, expected_result)
From eb5b46a144624e26089eb868daebcdfbefe99af1 Mon Sep 17 00:00:00 2001
From: HyTurtle <81598434+HyTurtle@users.noreply.github.com>
Date: Thu, 9 Nov 2023 00:10:28 +0000
Subject: [PATCH 006/158] enable json1 extenstion
---
pythonforandroid/recipes/sqlite3/Android.mk | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sqlite3/Android.mk b/pythonforandroid/recipes/sqlite3/Android.mk
index f52bc46f97..57bc81573d 100644
--- a/pythonforandroid/recipes/sqlite3/Android.mk
+++ b/pythonforandroid/recipes/sqlite3/Android.mk
@@ -6,6 +6,6 @@ LOCAL_SRC_FILES := sqlite3.c
LOCAL_MODULE := sqlite3
-LOCAL_CFLAGS := -DSQLITE_ENABLE_FTS4 -D_FILE_OFFSET_BITS=32
+LOCAL_CFLAGS := -DSQLITE_ENABLE_FTS4 -D_FILE_OFFSET_BITS=32 -DSQLITE_ENABLE_JSON1
include $(BUILD_SHARED_LIBRARY)
From 64ff7a84c37fd8ba39f93d1284275074f0d03d3f Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Thu, 9 Nov 2023 15:47:02 +0100
Subject: [PATCH 007/158] Bump `pyjnius` version to `1.6.1` (#2914)
---
pythonforandroid/recipes/pyjnius/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/pyjnius/__init__.py b/pythonforandroid/recipes/pyjnius/__init__.py
index 85a5860d5c..0bcb74d392 100644
--- a/pythonforandroid/recipes/pyjnius/__init__.py
+++ b/pythonforandroid/recipes/pyjnius/__init__.py
@@ -6,7 +6,7 @@
class PyjniusRecipe(CythonRecipe):
- version = '1.5.0'
+ version = '1.6.1'
url = 'https://github.com/kivy/pyjnius/archive/{version}.zip'
name = 'pyjnius'
depends = [('genericndkbuild', 'sdl2'), 'six']
From 609ad2d61c242d8856c4d5d3bd4d4e7b8950b65c Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Fri, 10 Nov 2023 20:52:22 +0100
Subject: [PATCH 008/158] Add (now mandatory) `.readthedocs.yaml` file, add
docs `requirements.txt` and update sphinx conf (#2916)
* Add (now mandatory) .readthedocs.yaml file, add docs requirements.txt and update sphinx conf
* Do not use a double source of truth for docs dependencies. extra_require is great, but unfortunately we can't use it on readthedocs builds
---
.github/workflows/push.yml | 4 +++-
.readthedocs.yaml | 16 ++++++++++++++
doc/requirements.txt | 2 ++
doc/source/conf.py | 45 +++++++++++++++++++++++++-------------
setup.py | 3 ---
5 files changed, 51 insertions(+), 19 deletions(-)
create mode 100644 .readthedocs.yaml
create mode 100644 doc/requirements.txt
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 7b79a30a63..d7f789aaaf 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -244,7 +244,9 @@ jobs:
steps:
- uses: actions/checkout@v4
- name: Requirements
- run: python3 -m pip install .[docs]
+ run: |
+ python -m pip install --upgrade pip
+ pip install -r doc/requirements.txt
- name: Check links
run: sphinx-build -b linkcheck doc/source doc/build
- name: Generate documentation
diff --git a/.readthedocs.yaml b/.readthedocs.yaml
new file mode 100644
index 0000000000..e3f99e76b0
--- /dev/null
+++ b/.readthedocs.yaml
@@ -0,0 +1,16 @@
+# Read the Docs configuration file for Sphinx projects
+# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details
+
+version: 2
+
+build:
+ os: ubuntu-22.04
+ tools:
+ python: "3"
+
+python:
+ install:
+ - requirements: doc/requirements.txt
+
+sphinx:
+ configuration: doc/source/conf.py
\ No newline at end of file
diff --git a/doc/requirements.txt b/doc/requirements.txt
new file mode 100644
index 0000000000..2a72e68703
--- /dev/null
+++ b/doc/requirements.txt
@@ -0,0 +1,2 @@
+Sphinx~=7.2.6
+furo==2023.9.10
\ No newline at end of file
diff --git a/doc/source/conf.py b/doc/source/conf.py
index e4ea304fde..a355505748 100644
--- a/doc/source/conf.py
+++ b/doc/source/conf.py
@@ -12,19 +12,17 @@
# All configuration values have a default; values that are commented out
# serve to show the default.
-import sys
+import datetime
import os
-import shlex
+import re
+import sys
# If extensions (or modules to document with autodoc) are in another directory,
# add these directories to sys.path here. If the directory is relative to the
# documentation root, use os.path.abspath to make it absolute, like shown here.
#sys.path.insert(0, os.path.abspath('.'))
-sys.path.append(os.path.abspath('ext/sphinx_rtd_theme'))
sys.path.append(os.path.abspath('../../pythonforandroid'))
-import sphinx_rtd_theme
-
# -- General configuration ------------------------------------------------
# If your documentation needs a minimal Sphinx version, state it here.
@@ -54,18 +52,35 @@
master_doc = 'index'
# General information about the project.
-project = u'python-for-android'
-copyright = u'2015, Alexander Taylor'
-author = u'Alexander Taylor'
+project = 'python-for-android'
+
+_today = datetime.datetime.now()
+
+author = 'Kivy Team and other contributors'
+
+copyright = f'{_today.year}, {author}'
# The version info for the project you're documenting, acts as replacement for
# |version| and |release|, also used in various other places throughout the
# built documents.
#
+
+# Lookup the version from the pyjnius module, without installing it
+# since readthedocs.org may have issue to install it.
+# Read the version from the __init__.py file, without importing it.
+def get_version():
+ with open(
+ os.path.join(os.path.abspath("../.."), "pythonforandroid", "__init__.py")
+ ) as fp:
+ for line in fp:
+ m = re.search(r'^\s*__version__\s*=\s*([\'"])([^\'"]+)\1\s*$', line)
+ if m:
+ return m.group(2)
+
# The short X.Y version.
-version = '0.1'
+version = get_version()
# The full version, including alpha/beta/rc tags.
-release = '0.1'
+release = get_version()
# The language for content autogenerated by Sphinx. Refer to documentation
# for a list of supported languages.
@@ -82,7 +97,7 @@
# List of patterns, relative to source directory, that match files and
# directories to ignore when looking for source files.
-exclude_patterns = ['ext/*', ]
+exclude_patterns = []
# The reST default role (used for this markup: `text`) to use for all
# documents.
@@ -116,7 +131,7 @@
# The theme to use for HTML and HTML Help pages. See the documentation for
# a list of builtin themes.
-html_theme = 'sphinx_rtd_theme'
+html_theme = 'furo'
# Theme options are theme-specific and customize the look and feel of a theme
# further. For a list of options available for each theme, see the
@@ -124,7 +139,7 @@
#html_theme_options = {}
# Add any paths that contain custom themes here, relative to this directory.
-html_theme_path = [sphinx_rtd_theme.get_html_theme_path()]
+# html_theme_path = []
# The name for this set of Sphinx documents. If None, it defaults to
# "<project> v<release> documentation".
@@ -230,8 +245,8 @@
# (source start file, target name, title,
# author, documentclass [howto, manual, or own class]).
latex_documents = [
- (master_doc, 'python-for-android.tex', u'python-for-android Documentation',
- u'Alexander Taylor', 'manual'),
+ (master_doc, 'python-for-android.tex', 'python-for-android Documentation',
+ author, 'manual'),
]
# The name of an image file (relative to this directory) to place at the top of
diff --git a/setup.py b/setup.py
index 8099647a93..6351cdcf8a 100644
--- a/setup.py
+++ b/setup.py
@@ -95,9 +95,6 @@ def recursively_include(results, directory, patterns):
url='https://github.com/kivy/python-for-android',
license='MIT',
install_requires=install_reqs,
- extras_require={
- "docs": ["sphinx", "sphinx-rtd-theme"],
- },
entry_points={
'console_scripts': [
'python-for-android = pythonforandroid.entrypoints:main',
From 59d6985a93dba64d6380688d546d554d6abe213a Mon Sep 17 00:00:00 2001
From: Julian <github@somethinkodd.com>
Date: Mon, 13 Nov 2023 01:36:02 +1100
Subject: [PATCH 009/158] Introduce FAQ in docs (#2917)
* Introduce FAQ
Future work:
* Remove much of same info from trouble-shooting section.
* Remove duplicate info from Kivy framework FAQ.
* Link to this from Kivy framework FAQ.
* Apply suggestions from code review
Still to do: Remove obsolete section.
Co-authored-by: Mirko Galimberti <me@mirkogalimberti.com>
* Remove obsolete entry
---------
Co-authored-by: Mirko Galimberti <me@mirkogalimberti.com>
---
FAQ.md | 73 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 73 insertions(+)
create mode 100644 FAQ.md
diff --git a/FAQ.md b/FAQ.md
new file mode 100644
index 0000000000..9e37084375
--- /dev/null
+++ b/FAQ.md
@@ -0,0 +1,73 @@
+# FAQ for python-for-android (p4a)
+
+## Introduction
+
+[python-for-android](https://python-for-android.readthedocs.io/) is an open
+source build tool to let you package Python code into standalone Android artifacts (APKs, AABs, AARs).
+These can be passed around, installed, or uploaded to marketplaces such as the
+Play Store just like any other Android app.
+
+### Sibling Projects:
+
+This tool was originally developed for apps produced with
+the [Kivy framework](https://github.com/kivy/kivy), and is
+managed by the same team. However, it can be used to build other types of Python
+apps for Android.
+
+p4a is often used in conjunction
+with [Buildozer](https://github.com/kivy/buildozer), which can download, install
+and keep up-to-date any necessary prerequisites (including p4a itself), for a
+number of target platforms, using a specification file to define the build.
+
+### Is it possible to have a kiosk app on Android?
+
+Thomas Hansen wrote a detailed answer
+on [the kivy-users mailing list](https://groups.google.com/d/msg/kivy-users/QKoCekAR1c0/yV-85Y_iAwoJ)
+
+Basically, you need to root the device, remove the SystemUI package, add some
+lines to the xml configuration, and you're done.
+
+### Common Errors
+
+The following are common problems and resolutions that users have reported.
+
+
+#### AttributeError: ‘Context’ object has no attribute ‘hostpython’
+
+This is a known bug in some releases. To work around it, add your python
+requirement explicitly, e.g. `--requirements=python3,kivy`. This also applies
+when using buildozer, in which case add python3 to your buildozer.spec
+requirements.
+
+#### linkname too long
+
+This can happen when you try to include a very long filename, which doesn’t
+normally happen but can occur accidentally if the p4a directory contains a
+`.buildozer` directory that is not excluded from the build (e.g. if buildozer
+was previously used). Removing this directory should fix the problem, and is
+desirable anyway since you don’t want it in the APK.
+
+#### Requested API target XX is not available, install it with the SDK android tool
+
+This means that your SDK is missing the required platform tools. You need to
+install the `platforms;android-XX` package in your SDK, using the android or
+sdkmanager tools (depending on SDK version).
+
+If using buildozer this should be done automatically, but as a workaround you
+can run these from `~/.buildozer/android/platform/android-sdk-XX/tools/android`
+
+#### SSLError(“Can’t connect to HTTPS URL because the SSL module is not available.”)
+Your hostpython3 was compiled without SSL support. You need to install the SSL
+development files before rebuilding the hostpython3 recipe. Remember to always
+clean the build before rebuilding (`p4a clean builds`, or with buildozer `buildozer
+android clean`).
+
+On Ubuntu and derivatives:
+
+ apt install libssl-dev
+ p4a clean builds # or with: buildozer `buildozer android clean
+
+On macOS:
+
+ brew install openssl
+ p4a clean builds # or with: buildozer `buildozer android clean
From 17bf5322791ec8cec85836fbe906e63664a05445 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sat, 18 Nov 2023 17:33:59 +0100
Subject: [PATCH 010/158] Update Android gradle plugin to 8.1.1 and gradle to
8.0.2 (#2887)
---
.../common/build/gradle/wrapper/gradle-wrapper.properties | 2 +-
.../bootstraps/common/build/templates/build.tmpl.gradle | 3 ++-
.../bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml | 1 -
.../service_library/build/templates/AndroidManifest.tmpl.xml | 1 -
.../service_only/build/templates/AndroidManifest.tmpl.xml | 1 -
.../webview/build/templates/AndroidManifest.tmpl.xml | 1 -
6 files changed, 3 insertions(+), 6 deletions(-)
diff --git a/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties b/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
index dd012b89f2..8f174bc31b 100644
--- a/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
+++ b/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
@@ -3,4 +3,4 @@ distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
-distributionUrl=https\://services.gradle.org/distributions/gradle-7.4.1-all.zip
+distributionUrl=https\://services.gradle.org/distributions/gradle-8.0.2-all.zip
diff --git a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
index d8d4326115..b46c9030ff 100644
--- a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
+++ b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
@@ -5,7 +5,7 @@ buildscript {
jcenter()
}
dependencies {
- classpath 'com.android.tools.build:gradle:7.1.2'
+ classpath 'com.android.tools.build:gradle:8.1.1'
}
}
@@ -26,6 +26,7 @@ apply plugin: 'com.android.application'
{% endif %}
android {
+ namespace '{{ args.package }}'
compileSdkVersion {{ android_api }}
buildToolsVersion '{{ build_tools_version }}'
defaultConfig {
diff --git a/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
index 3353c0a0d5..a887a53d54 100644
--- a/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
@@ -3,7 +3,6 @@
com.gamemaker.game
-->
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="{{ args.package }}"
android:versionCode="{{ args.numeric_version }}"
android:versionName="{{ args.version }}"
android:installLocation="auto">
diff --git a/pythonforandroid/bootstraps/service_library/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/service_library/build/templates/AndroidManifest.tmpl.xml
index f667651780..017a1588ec 100644
--- a/pythonforandroid/bootstraps/service_library/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/service_library/build/templates/AndroidManifest.tmpl.xml
@@ -1,6 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="{{ args.package }}"
android:versionCode="{{ args.numeric_version }}"
android:versionName="{{ args.version }}">
diff --git a/pythonforandroid/bootstraps/service_only/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/service_only/build/templates/AndroidManifest.tmpl.xml
index ab410330f2..f0034d7e73 100644
--- a/pythonforandroid/bootstraps/service_only/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/service_only/build/templates/AndroidManifest.tmpl.xml
@@ -1,6 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="{{ args.package }}"
android:versionCode="{{ args.numeric_version }}"
android:versionName="{{ args.version }}"
android:installLocation="auto">
diff --git a/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
index 1b83cd83e3..f9e4fa3c61 100644
--- a/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
@@ -1,6 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="{{ args.package }}"
android:versionCode="{{ args.numeric_version }}"
android:versionName="{{ args.version }}"
android:installLocation="auto">
From 83e74cac0fa230231f63b7d7e4f1e8581fd9e2ce Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 18 Nov 2023 22:42:03 +0530
Subject: [PATCH 011/158] Add support for Python `3.11` and make it the default
while building `hostpython3` and `python3` (#2850)
* python: update to 3.11.5
* scipy: update to 1.11.3
* add `pybind11` to `ci/constants.py` in BROKEN_RECIPES_PYTHON3
---
ci/constants.py | 3 +-
.../java/org/kivy/android/PythonUtil.java | 3 +-
pythonforandroid/recipes/cython/__init__.py | 2 +-
.../recipes/hostpython3/__init__.py | 2 +-
pythonforandroid/recipes/pybind11/__init__.py | 2 +-
pythonforandroid/recipes/python3/__init__.py | 27 +-
.../cpython-311-ctypes-find-library.patch | 19 +
pythonforandroid/recipes/scipy/__init__.py | 18 +-
pythonforandroid/recipes/scipy/setup.py.patch | 1098 +++++++++++++++++
9 files changed, 1158 insertions(+), 16 deletions(-)
create mode 100644 pythonforandroid/recipes/python3/patches/cpython-311-ctypes-find-library.patch
create mode 100644 pythonforandroid/recipes/scipy/setup.py.patch
diff --git a/ci/constants.py b/ci/constants.py
index 6a81a23405..52a996c90f 100644
--- a/ci/constants.py
+++ b/ci/constants.py
@@ -39,7 +39,8 @@ class TargetPython(Enum):
'boost',
# libtorrent gives errors (requires boost. Also, see issue #2809, to start with)
'libtorrent',
-
+ # pybind11 build fails on macos
+ 'pybind11',
])
BROKEN_RECIPES = {
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
index 5bc23bacf4..cc04d83f6b 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
@@ -55,6 +55,7 @@ protected static ArrayList<String> getLibraries(File libsDir) {
libsList.add("python3.8");
libsList.add("python3.9");
libsList.add("python3.10");
+ libsList.add("python3.11");
libsList.add("main");
return libsList;
}
@@ -74,7 +75,7 @@ public static void loadLibraries(File filesDir, File libsDir) {
// load, and it has failed, give a more
// general error
Log.v(TAG, "Library loading error: " + e.getMessage());
- if (lib.startsWith("python3.10") && !foundPython) {
+ if (lib.startsWith("python3.11") && !foundPython) {
throw new RuntimeException("Could not load any libpythonXXX.so");
} else if (lib.startsWith("python")) {
continue;
diff --git a/pythonforandroid/recipes/cython/__init__.py b/pythonforandroid/recipes/cython/__init__.py
index 9135e187ca..b8bac0ae18 100644
--- a/pythonforandroid/recipes/cython/__init__.py
+++ b/pythonforandroid/recipes/cython/__init__.py
@@ -3,7 +3,7 @@
class CythonRecipe(CompiledComponentsPythonRecipe):
- version = '0.29.28'
+ version = '0.29.36'
url = 'https://github.com/cython/cython/archive/{version}.tar.gz'
site_packages_name = 'cython'
depends = ['setuptools']
diff --git a/pythonforandroid/recipes/hostpython3/__init__.py b/pythonforandroid/recipes/hostpython3/__init__.py
index ee53b6ef09..9ba4580019 100644
--- a/pythonforandroid/recipes/hostpython3/__init__.py
+++ b/pythonforandroid/recipes/hostpython3/__init__.py
@@ -35,7 +35,7 @@ class HostPython3Recipe(Recipe):
:class:`~pythonforandroid.python.HostPythonRecipe`
'''
- version = '3.10.10'
+ version = '3.11.5'
name = 'hostpython3'
build_subdir = 'native-build'
diff --git a/pythonforandroid/recipes/pybind11/__init__.py b/pythonforandroid/recipes/pybind11/__init__.py
index affff8185b..3eb8871ff9 100644
--- a/pythonforandroid/recipes/pybind11/__init__.py
+++ b/pythonforandroid/recipes/pybind11/__init__.py
@@ -4,7 +4,7 @@
class Pybind11Recipe(PythonRecipe):
- version = '2.9.0'
+ version = '2.11.1'
url = 'https://github.com/pybind/pybind11/archive/refs/tags/v{version}.zip'
depends = ['setuptools']
call_hostpython_via_targetpython = False
diff --git a/pythonforandroid/recipes/python3/__init__.py b/pythonforandroid/recipes/python3/__init__.py
index 387922718e..3a9575148a 100644
--- a/pythonforandroid/recipes/python3/__init__.py
+++ b/pythonforandroid/recipes/python3/__init__.py
@@ -2,7 +2,6 @@
import sh
import subprocess
-from multiprocessing import cpu_count
from os import environ, utime
from os.path import dirname, exists, join
from pathlib import Path
@@ -56,7 +55,7 @@ class Python3Recipe(TargetPythonRecipe):
:class:`~pythonforandroid.python.GuestPythonRecipe`
'''
- version = '3.10.10'
+ version = '3.11.5'
url = 'https://www.python.org/ftp/python/{version}/Python-{version}.tgz'
name = 'python3'
@@ -71,15 +70,17 @@ class Python3Recipe(TargetPythonRecipe):
# Python 3.8.1 & 3.9.X
('patches/py3.8.1.patch', version_starts_with("3.8")),
('patches/py3.8.1.patch', version_starts_with("3.9")),
- ('patches/py3.8.1.patch', version_starts_with("3.10"))
+ ('patches/py3.8.1.patch', version_starts_with("3.10")),
+ ('patches/cpython-311-ctypes-find-library.patch', version_starts_with("3.11")),
]
if shutil.which('lld') is not None:
- patches = patches + [
+ patches += [
("patches/py3.7.1_fix_cortex_a8.patch", version_starts_with("3.7")),
("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.8")),
("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.9")),
- ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.10"))
+ ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.10")),
+ ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.11")),
]
depends = ['hostpython3', 'sqlite3', 'openssl', 'libffi']
@@ -101,7 +102,12 @@ class Python3Recipe(TargetPythonRecipe):
'ac_cv_header_sys_eventfd_h=no',
'--prefix={prefix}',
'--exec-prefix={exec_prefix}',
- '--enable-loadable-sqlite-extensions')
+ '--enable-loadable-sqlite-extensions'
+ )
+
+ if version_starts_with("3.11"):
+ configure_args += ('--with-build-python={python_host_bin}',)
+
'''The configure arguments needed to build the python recipe. Those are
used in method :meth:`build_arch` (if not overwritten like python3's
recipe does).
@@ -323,12 +329,19 @@ def build_arch(self, arch):
*(' '.join(self.configure_args).format(
android_host=env['HOSTARCH'],
android_build=android_build,
+ python_host_bin=join(self.get_recipe(
+ 'host' + self.name, self.ctx
+ ).get_path_to_python(), "python3"),
prefix=sys_prefix,
exec_prefix=sys_exec_prefix)).split(' '),
_env=env)
+ # Python build does not seem to play well with make -j option from Python 3.11 and onwards
+ # Before losing some time, please check issue
+ # https://github.com/python/cpython/issues/101295 , as the root cause looks similar
shprint(
- sh.make, 'all', '-j', str(cpu_count()),
+ sh.make,
+ 'all',
'INSTSONAME={lib_name}'.format(lib_name=self._libpython),
_env=env
)
diff --git a/pythonforandroid/recipes/python3/patches/cpython-311-ctypes-find-library.patch b/pythonforandroid/recipes/python3/patches/cpython-311-ctypes-find-library.patch
new file mode 100644
index 0000000000..7864d57ac8
--- /dev/null
+++ b/pythonforandroid/recipes/python3/patches/cpython-311-ctypes-find-library.patch
@@ -0,0 +1,19 @@
+--- Python-3.11.5/Lib/ctypes/util.py 2023-08-24 17:39:18.000000000 +0530
++++ Python-3.11.5.mod/Lib/ctypes/util.py 2023-11-18 22:12:17.356160615 +0530
+@@ -4,7 +4,15 @@
+ import sys
+
+ # find_library(name) returns the pathname of a library, or None.
+-if os.name == "nt":
++
++# This patch overrides the find_library to look in the right places on
++# Android
++if True:
++ from android._ctypes_library_finder import find_library as _find_lib
++ def find_library(name):
++ return _find_lib(name)
++
++elif os.name == "nt":
+
+ def _get_build_version():
+ """Return the version of MSVC that was used to build Python.
diff --git a/pythonforandroid/recipes/scipy/__init__.py b/pythonforandroid/recipes/scipy/__init__.py
index 455a9887d7..bde9758d8d 100644
--- a/pythonforandroid/recipes/scipy/__init__.py
+++ b/pythonforandroid/recipes/scipy/__init__.py
@@ -1,8 +1,10 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
from multiprocessing import cpu_count
from os.path import join
from os import environ
-from pythonforandroid.util import build_platform
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
+from pythonforandroid.util import build_platform, current_directory
def arch_to_toolchain(arch):
@@ -13,12 +15,14 @@ def arch_to_toolchain(arch):
class ScipyRecipe(CompiledComponentsPythonRecipe):
- version = '1.8.1'
- url = f'https://github.com/scipy/scipy/releases/download/v{version}/scipy-{version}.zip'
+ version = 'maintenance/1.11.x'
+ url = 'git+https://github.com/scipy/scipy.git'
+ git_commit = 'b430bf54b5064465983813e2cfef3fcb86c3df07' # version 1.11.3
site_packages_name = 'scipy'
depends = ['setuptools', 'cython', 'numpy', 'lapack', 'pybind11']
call_hostpython_via_targetpython = False
need_stl_shared = True
+ patches = ["setup.py.patch"]
def build_compiled_components(self, arch):
self.setup_extra_args = ['-j', str(cpu_count())]
@@ -30,6 +34,12 @@ def rebuild_compiled_components(self, arch, env):
super().rebuild_compiled_components(arch, env)
self.setup_extra_args = []
+ def download_file(self, url, target, cwd=None):
+ super().download_file(url, target, cwd=cwd)
+ with current_directory(target):
+ shprint(sh.git, 'fetch', '--unshallow')
+ shprint(sh.git, 'checkout', self.git_commit)
+
def get_recipe_env(self, arch):
env = super().get_recipe_env(arch)
arch_env = arch.get_env()
diff --git a/pythonforandroid/recipes/scipy/setup.py.patch b/pythonforandroid/recipes/scipy/setup.py.patch
new file mode 100644
index 0000000000..9fbc0ab5fb
--- /dev/null
+++ b/pythonforandroid/recipes/scipy/setup.py.patch
@@ -0,0 +1,1098 @@
+diff '--color=auto' -uNr scipy/_setup.py scipy.mod/_setup.py
+--- scipy/_setup.py 2023-10-30 19:20:36.545524745 +0530
++++ scipy.mod/_setup.py 1970-01-01 05:30:00.000000000 +0530
+@@ -1,545 +0,0 @@
+-#!/usr/bin/env python
+-"""SciPy: Scientific Library for Python
+-
+-SciPy (pronounced "Sigh Pie") is open-source software for mathematics,
+-science, and engineering. The SciPy library
+-depends on NumPy, which provides convenient and fast N-dimensional
+-array manipulation. The SciPy library is built to work with NumPy
+-arrays, and provides many user-friendly and efficient numerical
+-routines such as routines for numerical integration and optimization.
+-Together, they run on all popular operating systems, are quick to
+-install, and are free of charge. NumPy and SciPy are easy to use,
+-but powerful enough to be depended upon by some of the world's
+-leading scientists and engineers. If you need to manipulate
+-numbers on a computer and display or publish the results,
+-give SciPy a try!
+-
+-"""
+-
+-
+-# IMPORTANT:
+-#
+-# THIS FILE IS INTENTIONALLY RENAMED FROM setup.py TO _setup.py
+-# IT IS ONLY KEPT IN THE REPO BECAUSE conda-forge STILL NEEDS IT
+-# FOR BUILDING SCIPY ON WINDOWS. IT SHOULD NOT BE USED BY ANYONE
+-# ELSE. USE `pip install .` OR ANOTHER INSTALL COMMAND USING A
+-# BUILD FRONTEND LIKE pip OR pypa/build TO INSTALL SCIPY FROM SOURCE.
+-#
+-# SEE http://scipy.github.io/devdocs/building/index.html FOR BUILD
+-# INSTRUCTIONS.
+-
+-
+-DOCLINES = (__doc__ or '').split("\n")
+-
+-import os
+-import sys
+-import subprocess
+-import textwrap
+-import warnings
+-import sysconfig
+-from tools.version_utils import write_version_py, get_version_info
+-from tools.version_utils import IS_RELEASE_BRANCH
+-import importlib
+-
+-
+-if sys.version_info[:2] < (3, 9):
+- raise RuntimeError("Python version >= 3.9 required.")
+-
+-import builtins
+-
+-
+-CLASSIFIERS = """\
+-Development Status :: 5 - Production/Stable
+-Intended Audience :: Science/Research
+-Intended Audience :: Developers
+-License :: OSI Approved :: BSD License
+-Programming Language :: C
+-Programming Language :: Python
+-Programming Language :: Python :: 3
+-Programming Language :: Python :: 3.9
+-Programming Language :: Python :: 3.10
+-Programming Language :: Python :: 3.11
+-Topic :: Software Development :: Libraries
+-Topic :: Scientific/Engineering
+-Operating System :: Microsoft :: Windows
+-Operating System :: POSIX :: Linux
+-Operating System :: POSIX
+-Operating System :: Unix
+-Operating System :: MacOS
+-
+-"""
+-
+-
+-# BEFORE importing setuptools, remove MANIFEST. Otherwise it may not be
+-# properly updated when the contents of directories change (true for distutils,
+-# not sure about setuptools).
+-if os.path.exists('MANIFEST'):
+- os.remove('MANIFEST')
+-
+-# This is a bit hackish: we are setting a global variable so that the main
+-# scipy __init__ can detect if it is being loaded by the setup routine, to
+-# avoid attempting to load components that aren't built yet. While ugly, it's
+-# a lot more robust than what was previously being used.
+-builtins.__SCIPY_SETUP__ = True
+-
+-
+-def check_submodules():
+- """ verify that the submodules are checked out and clean
+- use `git submodule update --init`; on failure
+- """
+- if not os.path.exists('.git'):
+- return
+- with open('.gitmodules') as f:
+- for l in f:
+- if 'path' in l:
+- p = l.split('=')[-1].strip()
+- if not os.path.exists(p):
+- raise ValueError('Submodule %s missing' % p)
+-
+-
+- proc = subprocess.Popen(['git', 'submodule', 'status'],
+- stdout=subprocess.PIPE)
+- status, _ = proc.communicate()
+- status = status.decode("ascii", "replace")
+- for line in status.splitlines():
+- if line.startswith('-') or line.startswith('+'):
+- raise ValueError('Submodule not clean: %s' % line)
+-
+-
+-class concat_license_files():
+- """Merge LICENSE.txt and LICENSES_bundled.txt for sdist creation
+-
+- Done this way to keep LICENSE.txt in repo as exact BSD 3-clause (see
+- NumPy gh-13447). This makes GitHub state correctly how SciPy is licensed.
+- """
+- def __init__(self):
+- self.f1 = 'LICENSE.txt'
+- self.f2 = 'LICENSES_bundled.txt'
+-
+- def __enter__(self):
+- """Concatenate files and remove LICENSES_bundled.txt"""
+- with open(self.f1, 'r') as f1:
+- self.bsd_text = f1.read()
+-
+- with open(self.f1, 'a') as f1:
+- with open(self.f2, 'r') as f2:
+- self.bundled_text = f2.read()
+- f1.write('\n\n')
+- f1.write(self.bundled_text)
+-
+- def __exit__(self, exception_type, exception_value, traceback):
+- """Restore content of both files"""
+- with open(self.f1, 'w') as f:
+- f.write(self.bsd_text)
+-
+-
+-from distutils.command.sdist import sdist
+-class sdist_checked(sdist):
+- """ check submodules on sdist to prevent incomplete tarballs """
+- def run(self):
+- check_submodules()
+- with concat_license_files():
+- sdist.run(self)
+-
+-
+-def get_build_ext_override():
+- """
+- Custom build_ext command to tweak extension building.
+- """
+- from numpy.distutils.command.build_ext import build_ext as npy_build_ext
+- if int(os.environ.get('SCIPY_USE_PYTHRAN', 1)):
+- try:
+- import pythran
+- from pythran.dist import PythranBuildExt
+- except ImportError:
+- BaseBuildExt = npy_build_ext
+- else:
+- BaseBuildExt = PythranBuildExt[npy_build_ext]
+- _pep440 = importlib.import_module('scipy._lib._pep440')
+- if _pep440.parse(pythran.__version__) < _pep440.Version('0.11.0'):
+- raise RuntimeError("The installed `pythran` is too old, >= "
+- "0.11.0 is needed, {} detected. Please "
+- "upgrade Pythran, or use `export "
+- "SCIPY_USE_PYTHRAN=0`.".format(
+- pythran.__version__))
+- else:
+- BaseBuildExt = npy_build_ext
+-
+- class build_ext(BaseBuildExt):
+- def finalize_options(self):
+- super().finalize_options()
+-
+- # Disable distutils parallel build, due to race conditions
+- # in numpy.distutils (Numpy issue gh-15957)
+- if self.parallel:
+- print("NOTE: -j build option not supported. Set NPY_NUM_BUILD_JOBS=4 "
+- "for parallel build.")
+- self.parallel = None
+-
+- def build_extension(self, ext):
+- # When compiling with GNU compilers, use a version script to
+- # hide symbols during linking.
+- if self.__is_using_gnu_linker(ext):
+- export_symbols = self.get_export_symbols(ext)
+- text = '{global: %s; local: *; };' % (';'.join(export_symbols),)
+-
+- script_fn = os.path.join(self.build_temp, 'link-version-{}.map'.format(ext.name))
+- with open(script_fn, 'w') as f:
+- f.write(text)
+- # line below fixes gh-8680
+- ext.extra_link_args = [arg for arg in ext.extra_link_args if not "version-script" in arg]
+- ext.extra_link_args.append('-Wl,--version-script=' + script_fn)
+-
+- # Allow late configuration
+- hooks = getattr(ext, '_pre_build_hook', ())
+- _run_pre_build_hooks(hooks, (self, ext))
+-
+- super().build_extension(ext)
+-
+- def __is_using_gnu_linker(self, ext):
+- if not sys.platform.startswith('linux'):
+- return False
+-
+- # Fortran compilation with gfortran uses it also for
+- # linking. For the C compiler, we detect gcc in a similar
+- # way as distutils does it in
+- # UnixCCompiler.runtime_library_dir_option
+- if ext.language == 'f90':
+- is_gcc = (self._f90_compiler.compiler_type in ('gnu', 'gnu95'))
+- elif ext.language == 'f77':
+- is_gcc = (self._f77_compiler.compiler_type in ('gnu', 'gnu95'))
+- else:
+- is_gcc = False
+- if self.compiler.compiler_type == 'unix':
+- cc = sysconfig.get_config_var("CC")
+- if not cc:
+- cc = ""
+- compiler_name = os.path.basename(cc.split(" ")[0])
+- is_gcc = "gcc" in compiler_name or "g++" in compiler_name
+- return is_gcc and sysconfig.get_config_var('GNULD') == 'yes'
+-
+- return build_ext
+-
+-
+-def get_build_clib_override():
+- """
+- Custom build_clib command to tweak library building.
+- """
+- from numpy.distutils.command.build_clib import build_clib as old_build_clib
+-
+- class build_clib(old_build_clib):
+- def finalize_options(self):
+- super().finalize_options()
+-
+- # Disable parallelization (see build_ext above)
+- self.parallel = None
+-
+- def build_a_library(self, build_info, lib_name, libraries):
+- # Allow late configuration
+- hooks = build_info.get('_pre_build_hook', ())
+- _run_pre_build_hooks(hooks, (self, build_info))
+- old_build_clib.build_a_library(self, build_info, lib_name, libraries)
+-
+- return build_clib
+-
+-
+-def _run_pre_build_hooks(hooks, args):
+- """Call a sequence of pre-build hooks, if any"""
+- if hooks is None:
+- hooks = ()
+- elif not hasattr(hooks, '__iter__'):
+- hooks = (hooks,)
+- for hook in hooks:
+- hook(*args)
+-
+-
+-def generate_cython():
+- cwd = os.path.abspath(os.path.dirname(__file__))
+- print("Cythonizing sources")
+- p = subprocess.call([sys.executable,
+- os.path.join(cwd, 'tools', 'cythonize.py'),
+- 'scipy'],
+- cwd=cwd)
+- if p != 0:
+- # Could be due to a too old pip version and build isolation, check that
+- try:
+- # Note, pip may not be installed or not have been used
+- import pip
+- except (ImportError, ModuleNotFoundError):
+- raise RuntimeError("Running cythonize failed!")
+- else:
+- _pep440 = importlib.import_module('scipy._lib._pep440')
+- if _pep440.parse(pip.__version__) < _pep440.Version('18.0.0'):
+- raise RuntimeError("Cython not found or too old. Possibly due "
+- "to `pip` being too old, found version {}, "
+- "needed is >= 18.0.0.".format(
+- pip.__version__))
+- else:
+- raise RuntimeError("Running cythonize failed!")
+-
+-
+-def parse_setuppy_commands():
+- """Check the commands and respond appropriately. Disable broken commands.
+-
+- Return a boolean value for whether or not to run the build or not (avoid
+- parsing Cython and template files if False).
+- """
+- args = sys.argv[1:]
+-
+- if not args:
+- # User forgot to give an argument probably, let setuptools handle that.
+- return True
+-
+- info_commands = ['--help-commands', '--name', '--version', '-V',
+- '--fullname', '--author', '--author-email',
+- '--maintainer', '--maintainer-email', '--contact',
+- '--contact-email', '--url', '--license', '--description',
+- '--long-description', '--platforms', '--classifiers',
+- '--keywords', '--provides', '--requires', '--obsoletes']
+-
+- for command in info_commands:
+- if command in args:
+- return False
+-
+- # Note that 'alias', 'saveopts' and 'setopt' commands also seem to work
+- # fine as they are, but are usually used together with one of the commands
+- # below and not standalone. Hence they're not added to good_commands.
+- good_commands = ('develop', 'sdist', 'build', 'build_ext', 'build_py',
+- 'build_clib', 'build_scripts', 'bdist_wheel', 'bdist_rpm',
+- 'bdist_wininst', 'bdist_msi', 'bdist_mpkg')
+-
+- for command in good_commands:
+- if command in args:
+- return True
+-
+- # The following commands are supported, but we need to show more
+- # useful messages to the user
+- if 'install' in args:
+- print(textwrap.dedent("""
+- Note: for reliable uninstall behaviour and dependency installation
+- and uninstallation, please use pip instead of using
+- `setup.py install`:
+-
+- - `pip install .` (from a git repo or downloaded source
+- release)
+- - `pip install scipy` (last SciPy release on PyPI)
+-
+- """))
+- return True
+-
+- if '--help' in args or '-h' in sys.argv[1]:
+- print(textwrap.dedent("""
+- SciPy-specific help
+- -------------------
+-
+- To install SciPy from here with reliable uninstall, we recommend
+- that you use `pip install .`. To install the latest SciPy release
+- from PyPI, use `pip install scipy`.
+-
+- For help with build/installation issues, please ask on the
+- scipy-user mailing list. If you are sure that you have run
+- into a bug, please report it at https://github.com/scipy/scipy/issues.
+-
+- Setuptools commands help
+- ------------------------
+- """))
+- return False
+-
+-
+- # The following commands aren't supported. They can only be executed when
+- # the user explicitly adds a --force command-line argument.
+- bad_commands = dict(
+- test="""
+- `setup.py test` is not supported. Use one of the following
+- instead:
+-
+- - `python runtests.py` (to build and test)
+- - `python runtests.py --no-build` (to test installed scipy)
+- - `>>> scipy.test()` (run tests for installed scipy
+- from within an interpreter)
+- """,
+- upload="""
+- `setup.py upload` is not supported, because it's insecure.
+- Instead, build what you want to upload and upload those files
+- with `twine upload -s <filenames>` instead.
+- """,
+- upload_docs="`setup.py upload_docs` is not supported",
+- easy_install="`setup.py easy_install` is not supported",
+- clean="""
+- `setup.py clean` is not supported, use one of the following instead:
+-
+- - `git clean -xdf` (cleans all files)
+- - `git clean -Xdf` (cleans all versioned files, doesn't touch
+- files that aren't checked into the git repo)
+- """,
+- check="`setup.py check` is not supported",
+- register="`setup.py register` is not supported",
+- bdist_dumb="`setup.py bdist_dumb` is not supported",
+- bdist="`setup.py bdist` is not supported",
+- flake8="`setup.py flake8` is not supported, use flake8 standalone",
+- build_sphinx="`setup.py build_sphinx` is not supported, see doc/README.md",
+- )
+- bad_commands['nosetests'] = bad_commands['test']
+- for command in ('upload_docs', 'easy_install', 'bdist', 'bdist_dumb',
+- 'register', 'check', 'install_data', 'install_headers',
+- 'install_lib', 'install_scripts', ):
+- bad_commands[command] = "`setup.py %s` is not supported" % command
+-
+- for command in bad_commands.keys():
+- if command in args:
+- print(textwrap.dedent(bad_commands[command]) +
+- "\nAdd `--force` to your command to use it anyway if you "
+- "must (unsupported).\n")
+- sys.exit(1)
+-
+- # Commands that do more than print info, but also don't need Cython and
+- # template parsing.
+- other_commands = ['egg_info', 'install_egg_info', 'rotate']
+- for command in other_commands:
+- if command in args:
+- return False
+-
+- # If we got here, we didn't detect what setup.py command was given
+- warnings.warn("Unrecognized setuptools command ('{}'), proceeding with "
+- "generating Cython sources and expanding templates".format(
+- ' '.join(sys.argv[1:])))
+- return True
+-
+-def check_setuppy_command():
+- run_build = parse_setuppy_commands()
+- if run_build:
+- try:
+- pkgname = 'numpy'
+- import numpy
+- pkgname = 'pybind11'
+- import pybind11
+- except ImportError as exc: # We do not have our build deps installed
+- print(textwrap.dedent(
+- """Error: '%s' must be installed before running the build.
+- """
+- % (pkgname,)))
+- sys.exit(1)
+-
+- return run_build
+-
+-def configuration(parent_package='', top_path=None):
+- from numpy.distutils.system_info import get_info, NotFoundError
+- from numpy.distutils.misc_util import Configuration
+-
+- lapack_opt = get_info('lapack_opt')
+-
+- if not lapack_opt:
+- if sys.platform == "darwin":
+- msg = ('No BLAS/LAPACK libraries found. '
+- 'Note: Accelerate is no longer supported.')
+- else:
+- msg = 'No BLAS/LAPACK libraries found.'
+- msg += ("\n"
+- "To build Scipy from sources, BLAS & LAPACK libraries "
+- "need to be installed.\n"
+- "See site.cfg.example in the Scipy source directory and\n"
+- "https://docs.scipy.org/doc/scipy/dev/contributor/building.html "
+- "for details.")
+- raise NotFoundError(msg)
+-
+- config = Configuration(None, parent_package, top_path)
+- config.set_options(ignore_setup_xxx_py=True,
+- assume_default_configuration=True,
+- delegate_options_to_subpackages=True,
+- quiet=True)
+-
+- config.add_subpackage('scipy')
+- config.add_data_files(('scipy', '*.txt'))
+-
+- config.get_version('scipy/version.py')
+-
+- return config
+-
+-
+-def setup_package():
+- # In maintenance branch, change np_maxversion to N+3 if numpy is at N
+- # Update here, in pyproject.toml, and in scipy/__init__.py
+- # Rationale: SciPy builds without deprecation warnings with N; deprecations
+- # in N+1 will turn into errors in N+3
+- # For Python versions, if releases is (e.g.) <=3.9.x, set bound to 3.10
+- np_minversion = '1.21.6'
+- np_maxversion = '1.28.0'
+- python_minversion = '3.9'
+- python_maxversion = '3.13'
+- if IS_RELEASE_BRANCH:
+- req_np = 'numpy>={},<{}'.format(np_minversion, np_maxversion)
+- req_py = '>={},<{}'.format(python_minversion, python_maxversion)
+- else:
+- req_np = 'numpy>={}'.format(np_minversion)
+- req_py = '>={}'.format(python_minversion)
+-
+- # Rewrite the version file every time
+- write_version_py('.')
+-
+- cmdclass = {'sdist': sdist_checked}
+-
+- metadata = dict(
+- name='scipy',
+- maintainer="SciPy Developers",
+- maintainer_email="scipy-dev@python.org",
+- description=DOCLINES[0],
+- long_description="\n".join(DOCLINES[2:]),
+- url="https://www.scipy.org",
+- download_url="https://github.com/scipy/scipy/releases",
+- project_urls={
+- "Bug Tracker": "https://github.com/scipy/scipy/issues",
+- "Documentation": "https://docs.scipy.org/doc/scipy/reference/",
+- "Source Code": "https://github.com/scipy/scipy",
+- },
+- license='BSD',
+- cmdclass=cmdclass,
+- classifiers=[_f for _f in CLASSIFIERS.split('\n') if _f],
+- platforms=["Windows", "Linux", "Solaris", "Mac OS-X", "Unix"],
+- install_requires=[req_np],
+- python_requires=req_py,
+- zip_safe=False,
+- )
+-
+- if "--force" in sys.argv:
+- run_build = True
+- sys.argv.remove('--force')
+- else:
+- # Raise errors for unsupported commands, improve help output, etc.
+- run_build = check_setuppy_command()
+-
+- # Disable OSX Accelerate, it has too old LAPACK
+- os.environ['ACCELERATE'] = 'None'
+-
+- # This import is here because it needs to be done before importing setup()
+- # from numpy.distutils, but after the MANIFEST removing and sdist import
+- # higher up in this file.
+- from setuptools import setup
+-
+- if run_build:
+- from numpy.distutils.core import setup
+-
+- # Customize extension building
+- cmdclass['build_ext'] = get_build_ext_override()
+- cmdclass['build_clib'] = get_build_clib_override()
+-
+- if not 'sdist' in sys.argv:
+- # Generate Cython sources, unless we're creating an sdist
+- # Cython is a build dependency, and shipping generated .c files
+- # can cause problems (see gh-14199)
+- generate_cython()
+-
+- metadata['configuration'] = configuration
+- else:
+- # Don't import numpy here - non-build actions are required to succeed
+- # without NumPy for example when pip is used to install Scipy when
+- # NumPy is not yet present in the system.
+-
+- # Version number is added to metadata inside configuration() if build
+- # is run.
+- metadata['version'] = get_version_info('.')[0]
+-
+- setup(**metadata)
+-
+-
+-if __name__ == '__main__':
+- setup_package()
+diff '--color=auto' -uNr scipy/setup.py scipy.mod/setup.py
+--- scipy/setup.py 1970-01-01 05:30:00.000000000 +0530
++++ scipy.mod/setup.py 2023-10-30 19:22:02.921729484 +0530
+@@ -0,0 +1,545 @@
++#!/usr/bin/env python
++"""SciPy: Scientific Library for Python
++
++SciPy (pronounced "Sigh Pie") is open-source software for mathematics,
++science, and engineering. The SciPy library
++depends on NumPy, which provides convenient and fast N-dimensional
++array manipulation. The SciPy library is built to work with NumPy
++arrays, and provides many user-friendly and efficient numerical
++routines such as routines for numerical integration and optimization.
++Together, they run on all popular operating systems, are quick to
++install, and are free of charge. NumPy and SciPy are easy to use,
++but powerful enough to be depended upon by some of the world's
++leading scientists and engineers. If you need to manipulate
++numbers on a computer and display or publish the results,
++give SciPy a try!
++
++"""
++
++
++# IMPORTANT:
++#
++# THIS FILE IS INTENTIONALLY RENAMED FROM setup.py TO _setup.py
++# IT IS ONLY KEPT IN THE REPO BECAUSE conda-forge STILL NEEDS IT
++# FOR BUILDING SCIPY ON WINDOWS. IT SHOULD NOT BE USED BY ANYONE
++# ELSE. USE `pip install .` OR ANOTHER INSTALL COMMAND USING A
++# BUILD FRONTEND LIKE pip OR pypa/build TO INSTALL SCIPY FROM SOURCE.
++#
++# SEE http://scipy.github.io/devdocs/building/index.html FOR BUILD
++# INSTRUCTIONS.
++
++
++DOCLINES = (__doc__ or '').split("\n")
++
++import os
++import sys
++import subprocess
++import textwrap
++import warnings
++import sysconfig
++from tools.version_utils import write_version_py, get_version_info
++from tools.version_utils import IS_RELEASE_BRANCH
++import importlib
++
++
++if sys.version_info[:2] < (3, 9):
++ raise RuntimeError("Python version >= 3.9 required.")
++
++import builtins
++
++
++CLASSIFIERS = """\
++Development Status :: 5 - Production/Stable
++Intended Audience :: Science/Research
++Intended Audience :: Developers
++License :: OSI Approved :: BSD License
++Programming Language :: C
++Programming Language :: Python
++Programming Language :: Python :: 3
++Programming Language :: Python :: 3.9
++Programming Language :: Python :: 3.10
++Programming Language :: Python :: 3.11
++Topic :: Software Development :: Libraries
++Topic :: Scientific/Engineering
++Operating System :: Microsoft :: Windows
++Operating System :: POSIX :: Linux
++Operating System :: POSIX
++Operating System :: Unix
++Operating System :: MacOS
++
++"""
++
++
++# BEFORE importing setuptools, remove MANIFEST. Otherwise it may not be
++# properly updated when the contents of directories change (true for distutils,
++# not sure about setuptools).
++if os.path.exists('MANIFEST'):
++ os.remove('MANIFEST')
++
++# This is a bit hackish: we are setting a global variable so that the main
++# scipy __init__ can detect if it is being loaded by the setup routine, to
++# avoid attempting to load components that aren't built yet. While ugly, it's
++# a lot more robust than what was previously being used.
++builtins.__SCIPY_SETUP__ = True
++
++
++def check_submodules():
++ """ verify that the submodules are checked out and clean
++ use `git submodule update --init`; on failure
++ """
++ if not os.path.exists('.git'):
++ return
++ with open('.gitmodules') as f:
++ for l in f:
++ if 'path' in l:
++ p = l.split('=')[-1].strip()
++ if not os.path.exists(p):
++ raise ValueError('Submodule %s missing' % p)
++
++
++ proc = subprocess.Popen(['git', 'submodule', 'status'],
++ stdout=subprocess.PIPE)
++ status, _ = proc.communicate()
++ status = status.decode("ascii", "replace")
++ for line in status.splitlines():
++ if line.startswith('-') or line.startswith('+'):
++ raise ValueError('Submodule not clean: %s' % line)
++
++
++class concat_license_files():
++ """Merge LICENSE.txt and LICENSES_bundled.txt for sdist creation
++
++ Done this way to keep LICENSE.txt in repo as exact BSD 3-clause (see
++ NumPy gh-13447). This makes GitHub state correctly how SciPy is licensed.
++ """
++ def __init__(self):
++ self.f1 = 'LICENSE.txt'
++ self.f2 = 'LICENSES_bundled.txt'
++
++ def __enter__(self):
++ """Concatenate files and remove LICENSES_bundled.txt"""
++ with open(self.f1, 'r') as f1:
++ self.bsd_text = f1.read()
++
++ with open(self.f1, 'a') as f1:
++ with open(self.f2, 'r') as f2:
++ self.bundled_text = f2.read()
++ f1.write('\n\n')
++ f1.write(self.bundled_text)
++
++ def __exit__(self, exception_type, exception_value, traceback):
++ """Restore content of both files"""
++ with open(self.f1, 'w') as f:
++ f.write(self.bsd_text)
++
++
++from distutils.command.sdist import sdist
++class sdist_checked(sdist):
++ """ check submodules on sdist to prevent incomplete tarballs """
++ def run(self):
++ check_submodules()
++ with concat_license_files():
++ sdist.run(self)
++
++
++def get_build_ext_override():
++ """
++ Custom build_ext command to tweak extension building.
++ """
++ from numpy.distutils.command.build_ext import build_ext as npy_build_ext
++ if int(os.environ.get('SCIPY_USE_PYTHRAN', 1)):
++ try:
++ import pythran
++ from pythran.dist import PythranBuildExt
++ except ImportError:
++ BaseBuildExt = npy_build_ext
++ else:
++ BaseBuildExt = PythranBuildExt[npy_build_ext]
++ _pep440 = importlib.import_module('scipy._lib._pep440')
++ if _pep440.parse(pythran.__version__) < _pep440.Version('0.11.0'):
++ raise RuntimeError("The installed `pythran` is too old, >= "
++ "0.11.0 is needed, {} detected. Please "
++ "upgrade Pythran, or use `export "
++ "SCIPY_USE_PYTHRAN=0`.".format(
++ pythran.__version__))
++ else:
++ BaseBuildExt = npy_build_ext
++
++ class build_ext(BaseBuildExt):
++ def finalize_options(self):
++ super().finalize_options()
++
++ # Disable distutils parallel build, due to race conditions
++ # in numpy.distutils (Numpy issue gh-15957)
++ if self.parallel:
++ print("NOTE: -j build option not supported. Set NPY_NUM_BUILD_JOBS=4 "
++ "for parallel build.")
++ self.parallel = None
++
++ def build_extension(self, ext):
++ # When compiling with GNU compilers, use a version script to
++ # hide symbols during linking.
++ if self.__is_using_gnu_linker(ext):
++ export_symbols = self.get_export_symbols(ext)
++ text = '{global: %s; local: *; };' % (';'.join(export_symbols),)
++
++ script_fn = os.path.join(self.build_temp, 'link-version-{}.map'.format(ext.name))
++ with open(script_fn, 'w') as f:
++ f.write(text)
++ # line below fixes gh-8680
++ ext.extra_link_args = [arg for arg in ext.extra_link_args if not "version-script" in arg]
++ ext.extra_link_args.append('-Wl,--version-script=' + script_fn)
++
++ # Allow late configuration
++ hooks = getattr(ext, '_pre_build_hook', ())
++ _run_pre_build_hooks(hooks, (self, ext))
++
++ super().build_extension(ext)
++
++ def __is_using_gnu_linker(self, ext):
++ if not sys.platform.startswith('linux'):
++ return False
++
++ # Fortran compilation with gfortran uses it also for
++ # linking. For the C compiler, we detect gcc in a similar
++ # way as distutils does it in
++ # UnixCCompiler.runtime_library_dir_option
++ if ext.language == 'f90':
++ is_gcc = (self._f90_compiler.compiler_type in ('gnu', 'gnu95'))
++ elif ext.language == 'f77':
++ is_gcc = (self._f77_compiler.compiler_type in ('gnu', 'gnu95'))
++ else:
++ is_gcc = False
++ if self.compiler.compiler_type == 'unix':
++ cc = sysconfig.get_config_var("CC")
++ if not cc:
++ cc = ""
++ compiler_name = os.path.basename(cc.split(" ")[0])
++ is_gcc = "gcc" in compiler_name or "g++" in compiler_name
++ return is_gcc and sysconfig.get_config_var('GNULD') == 'yes'
++
++ return build_ext
++
++
++def get_build_clib_override():
++ """
++ Custom build_clib command to tweak library building.
++ """
++ from numpy.distutils.command.build_clib import build_clib as old_build_clib
++
++ class build_clib(old_build_clib):
++ def finalize_options(self):
++ super().finalize_options()
++
++ # Disable parallelization (see build_ext above)
++ self.parallel = None
++
++ def build_a_library(self, build_info, lib_name, libraries):
++ # Allow late configuration
++ hooks = build_info.get('_pre_build_hook', ())
++ _run_pre_build_hooks(hooks, (self, build_info))
++ old_build_clib.build_a_library(self, build_info, lib_name, libraries)
++
++ return build_clib
++
++
++def _run_pre_build_hooks(hooks, args):
++ """Call a sequence of pre-build hooks, if any"""
++ if hooks is None:
++ hooks = ()
++ elif not hasattr(hooks, '__iter__'):
++ hooks = (hooks,)
++ for hook in hooks:
++ hook(*args)
++
++
++def generate_cython():
++ cwd = os.path.abspath(os.path.dirname(__file__))
++ print("Cythonizing sources")
++ p = subprocess.call([sys.executable,
++ os.path.join(cwd, 'tools', 'cythonize.py'),
++ 'scipy'],
++ cwd=cwd)
++ if p != 0:
++ # Could be due to a too old pip version and build isolation, check that
++ try:
++ # Note, pip may not be installed or not have been used
++ import pip
++ except (ImportError, ModuleNotFoundError):
++ raise RuntimeError("Running cythonize failed!")
++ else:
++ _pep440 = importlib.import_module('scipy._lib._pep440')
++ if _pep440.parse(pip.__version__) < _pep440.Version('18.0.0'):
++ raise RuntimeError("Cython not found or too old. Possibly due "
++ "to `pip` being too old, found version {}, "
++ "needed is >= 18.0.0.".format(
++ pip.__version__))
++ else:
++ raise RuntimeError("Running cythonize failed!")
++
++
++def parse_setuppy_commands():
++ """Check the commands and respond appropriately. Disable broken commands.
++
++ Return a boolean value for whether or not to run the build or not (avoid
++ parsing Cython and template files if False).
++ """
++ args = sys.argv[1:]
++
++ if not args:
++ # User forgot to give an argument probably, let setuptools handle that.
++ return True
++
++ info_commands = ['--help-commands', '--name', '--version', '-V',
++ '--fullname', '--author', '--author-email',
++ '--maintainer', '--maintainer-email', '--contact',
++ '--contact-email', '--url', '--license', '--description',
++ '--long-description', '--platforms', '--classifiers',
++ '--keywords', '--provides', '--requires', '--obsoletes']
++
++ for command in info_commands:
++ if command in args:
++ return False
++
++ # Note that 'alias', 'saveopts' and 'setopt' commands also seem to work
++ # fine as they are, but are usually used together with one of the commands
++ # below and not standalone. Hence they're not added to good_commands.
++ good_commands = ('develop', 'sdist', 'build', 'build_ext', 'build_py',
++ 'build_clib', 'build_scripts', 'bdist_wheel', 'bdist_rpm',
++ 'bdist_wininst', 'bdist_msi', 'bdist_mpkg')
++
++ for command in good_commands:
++ if command in args:
++ return True
++
++ # The following commands are supported, but we need to show more
++ # useful messages to the user
++ if 'install' in args:
++ print(textwrap.dedent("""
++ Note: for reliable uninstall behaviour and dependency installation
++ and uninstallation, please use pip instead of using
++ `setup.py install`:
++
++ - `pip install .` (from a git repo or downloaded source
++ release)
++ - `pip install scipy` (last SciPy release on PyPI)
++
++ """))
++ return True
++
++ if '--help' in args or '-h' in sys.argv[1]:
++ print(textwrap.dedent("""
++ SciPy-specific help
++ -------------------
++
++ To install SciPy from here with reliable uninstall, we recommend
++ that you use `pip install .`. To install the latest SciPy release
++ from PyPI, use `pip install scipy`.
++
++ For help with build/installation issues, please ask on the
++ scipy-user mailing list. If you are sure that you have run
++ into a bug, please report it at https://github.com/scipy/scipy/issues.
++
++ Setuptools commands help
++ ------------------------
++ """))
++ return False
++
++
++ # The following commands aren't supported. They can only be executed when
++ # the user explicitly adds a --force command-line argument.
++ bad_commands = dict(
++ test="""
++ `setup.py test` is not supported. Use one of the following
++ instead:
++
++ - `python runtests.py` (to build and test)
++ - `python runtests.py --no-build` (to test installed scipy)
++ - `>>> scipy.test()` (run tests for installed scipy
++ from within an interpreter)
++ """,
++ upload="""
++ `setup.py upload` is not supported, because it's insecure.
++ Instead, build what you want to upload and upload those files
++ with `twine upload -s <filenames>` instead.
++ """,
++ upload_docs="`setup.py upload_docs` is not supported",
++ easy_install="`setup.py easy_install` is not supported",
++ clean="""
++ `setup.py clean` is not supported, use one of the following instead:
++
++ - `git clean -xdf` (cleans all files)
++ - `git clean -Xdf` (cleans all versioned files, doesn't touch
++ files that aren't checked into the git repo)
++ """,
++ check="`setup.py check` is not supported",
++ register="`setup.py register` is not supported",
++ bdist_dumb="`setup.py bdist_dumb` is not supported",
++ bdist="`setup.py bdist` is not supported",
++ flake8="`setup.py flake8` is not supported, use flake8 standalone",
++ build_sphinx="`setup.py build_sphinx` is not supported, see doc/README.md",
++ )
++ bad_commands['nosetests'] = bad_commands['test']
++ for command in ('upload_docs', 'easy_install', 'bdist', 'bdist_dumb',
++ 'register', 'check', 'install_data', 'install_headers',
++ 'install_lib', 'install_scripts', ):
++ bad_commands[command] = "`setup.py %s` is not supported" % command
++
++ for command in bad_commands.keys():
++ if command in args:
++ print(textwrap.dedent(bad_commands[command]) +
++ "\nAdd `--force` to your command to use it anyway if you "
++ "must (unsupported).\n")
++ sys.exit(1)
++
++ # Commands that do more than print info, but also don't need Cython and
++ # template parsing.
++ other_commands = ['egg_info', 'install_egg_info', 'rotate']
++ for command in other_commands:
++ if command in args:
++ return False
++
++ # If we got here, we didn't detect what setup.py command was given
++ warnings.warn("Unrecognized setuptools command ('{}'), proceeding with "
++ "generating Cython sources and expanding templates".format(
++ ' '.join(sys.argv[1:])))
++ return True
++
++def check_setuppy_command():
++ run_build = parse_setuppy_commands()
++ if run_build:
++ try:
++ pkgname = 'numpy'
++ import numpy
++ pkgname = 'pybind11'
++ import pybind11
++ except ImportError as exc: # We do not have our build deps installed
++ print(textwrap.dedent(
++ """Error: '%s' must be installed before running the build.
++ """
++ % (pkgname,)))
++ sys.exit(1)
++
++ return run_build
++
++def configuration(parent_package='', top_path=None):
++ from numpy.distutils.system_info import get_info, NotFoundError
++ from numpy.distutils.misc_util import Configuration
++
++ lapack_opt = get_info('lapack_opt')
++
++ if not lapack_opt:
++ if sys.platform == "darwin":
++ msg = ('No BLAS/LAPACK libraries found. '
++ 'Note: Accelerate is no longer supported.')
++ else:
++ msg = 'No BLAS/LAPACK libraries found.'
++ msg += ("\n"
++ "To build Scipy from sources, BLAS & LAPACK libraries "
++ "need to be installed.\n"
++ "See site.cfg.example in the Scipy source directory and\n"
++ "https://docs.scipy.org/doc/scipy/dev/contributor/building.html "
++ "for details.")
++ raise NotFoundError(msg)
++
++ config = Configuration(None, parent_package, top_path)
++ config.set_options(ignore_setup_xxx_py=True,
++ assume_default_configuration=True,
++ delegate_options_to_subpackages=True,
++ quiet=True)
++
++ config.add_subpackage('scipy')
++ config.add_data_files(('scipy', '*.txt'))
++
++ config.get_version('scipy/version.py')
++
++ return config
++
++
++def setup_package():
++ # In maintenance branch, change np_maxversion to N+3 if numpy is at N
++ # Update here, in pyproject.toml, and in scipy/__init__.py
++ # Rationale: SciPy builds without deprecation warnings with N; deprecations
++ # in N+1 will turn into errors in N+3
++ # For Python versions, if releases is (e.g.) <=3.9.x, set bound to 3.10
++ np_minversion = '1.21.6'
++ np_maxversion = '1.28.0'
++ python_minversion = '3.9'
++ python_maxversion = '3.13'
++ if IS_RELEASE_BRANCH:
++ req_np = 'numpy>={},<{}'.format(np_minversion, np_maxversion)
++ req_py = '>={},<{}'.format(python_minversion, python_maxversion)
++ else:
++ req_np = 'numpy>={}'.format(np_minversion)
++ req_py = '>={}'.format(python_minversion)
++
++ # Rewrite the version file every time
++ write_version_py('.')
++
++ cmdclass = {'sdist': sdist_checked}
++
++ metadata = dict(
++ name='scipy',
++ maintainer="SciPy Developers",
++ maintainer_email="scipy-dev@python.org",
++ description=DOCLINES[0],
++ long_description="\n".join(DOCLINES[2:]),
++ url="https://www.scipy.org",
++ download_url="https://github.com/scipy/scipy/releases",
++ project_urls={
++ "Bug Tracker": "https://github.com/scipy/scipy/issues",
++ "Documentation": "https://docs.scipy.org/doc/scipy/reference/",
++ "Source Code": "https://github.com/scipy/scipy",
++ },
++ license='BSD',
++ cmdclass=cmdclass,
++ classifiers=[_f for _f in CLASSIFIERS.split('\n') if _f],
++ platforms=["Windows", "Linux", "Solaris", "Mac OS-X", "Unix"],
++ install_requires=[req_np],
++ python_requires=req_py,
++ zip_safe=False,
++ )
++
++ if "--force" in sys.argv:
++ run_build = True
++ sys.argv.remove('--force')
++ else:
++ # Raise errors for unsupported commands, improve help output, etc.
++ run_build = check_setuppy_command()
++
++ # Disable OSX Accelerate, it has too old LAPACK
++ os.environ['ACCELERATE'] = 'None'
++
++ # This import is here because it needs to be done before importing setup()
++ # from numpy.distutils, but after the MANIFEST removing and sdist import
++ # higher up in this file.
++ from setuptools import setup
++
++ if run_build:
++ from numpy.distutils.core import setup
++
++ # Customize extension building
++ cmdclass['build_ext'] = get_build_ext_override()
++ cmdclass['build_clib'] = get_build_clib_override()
++
++ if not 'sdist' in sys.argv:
++ # Generate Cython sources, unless we're creating an sdist
++ # Cython is a build dependency, and shipping generated .c files
++ # can cause problems (see gh-14199)
++ generate_cython()
++
++ metadata['configuration'] = configuration
++ else:
++ # Don't import numpy here - non-build actions are required to succeed
++ # without NumPy for example when pip is used to install Scipy when
++ # NumPy is not yet present in the system.
++
++ # Version number is added to metadata inside configuration() if build
++ # is run.
++ metadata['version'] = get_version_info('.')[0]
++
++ setup(**metadata)
++
++
++if __name__ == '__main__':
++ setup_package()
From 436d5a93282dcfcc7027d19d70963c6773544434 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Mon, 20 Nov 2023 20:56:25 +0100
Subject: [PATCH 012/158] Update prerequisites (`Dockerfile`,
`prerequisites.py`, docs) (#2923)
* Update Dockerfile to reflect latest requirements
* Increase JDK version on macOS prerequisites
* Force usage of a specific JDK version on macOS
* Update docs to reflect the current status
* Re-order deps in quickstart and Dockerfile
---
Dockerfile | 62 ++++++++++++---------------
doc/source/quickstart.rst | 71 +++++++++++++++----------------
pythonforandroid/prerequisites.py | 16 +++----
3 files changed, 70 insertions(+), 79 deletions(-)
diff --git a/Dockerfile b/Dockerfile
index 3b48c508c9..b5b2c597ff 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -18,7 +18,7 @@
# If platform is not specified, by default the target platform of the build request is used.
# This is not what we want, as Google doesn't provide a linux/arm64 compatible NDK.
# See: https://docs.docker.com/engine/reference/builder/#from
-FROM --platform=linux/amd64 ubuntu:20.04
+FROM --platform=linux/amd64 ubuntu:22.04
# configure locale
RUN apt -y update -qq > /dev/null \
@@ -48,43 +48,37 @@ ENV HOME_DIR="/home/${USER}"
ENV WORK_DIR="${HOME_DIR}/app" \
PATH="${HOME_DIR}/.local/bin:${PATH}" \
ANDROID_HOME="${HOME_DIR}/.android" \
- JAVA_HOME="/usr/lib/jvm/java-13-openjdk-amd64"
+ JAVA_HOME="/usr/lib/jvm/java-17-openjdk-amd64"
# install system dependencies
-RUN dpkg --add-architecture i386 \
- && ${RETRY} apt -y update -qq > /dev/null \
+RUN ${RETRY} apt -y update -qq > /dev/null \
&& ${RETRY} DEBIAN_FRONTEND=noninteractive apt install -qq --yes --no-install-recommends \
- autoconf \
- automake \
- autopoint \
- build-essential \
- ccache \
- cmake \
- gettext \
- git \
- lbzip2 \
- libffi-dev \
- libgtk2.0-0:i386 \
- libidn11:i386 \
- libltdl-dev \
- libncurses5:i386 \
- libssl-dev \
- libstdc++6:i386 \
- libtool \
- openjdk-13-jdk \
- patch \
- pkg-config \
- python3 \
- python3-dev \
- python3-pip \
- python3-venv \
- sudo \
- unzip \
- wget \
- zip \
- zlib1g-dev \
- zlib1g:i386 \
+ ant \
+ autoconf \
+ automake \
+ ccache \
+ cmake \
+ g++ \
+ gcc \
+ git \
+ lbzip2 \
+ libffi-dev \
+ libltdl-dev \
+ libtool \
+ libssl-dev \
+ make \
+ openjdk-17-jdk \
+ patch \
+ pkg-config \
+ python3 \
+ python3-dev \
+ python3-pip \
+ python3-venv \
+ sudo \
+ unzip \
+ wget \
+ zip \
&& apt -y autoremove \
&& apt -y clean \
&& rm -rf /var/lib/apt/lists/*
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 46612e5ad5..95c4eefc4e 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -56,48 +56,45 @@ You can also test the master branch from Github using::
pip install git+https://github.com/kivy/python-for-android.git
-Installing Dependencies
+Installing Prerequisites
~~~~~~~~~~~~~~~~~~~~~~~
-p4a has several dependencies that must be installed:
-
-- ant
-- autoconf (for libffi and other recipes)
-- automake
-- ccache (optional)
-- cmake (required for some native code recipes like jpeg's recipe)
-- cython (can be installed via pip)
-- gcc
-- git
-- libncurses (including 32 bit)
-- libtool (for libffi and recipes)
-- libssl-dev (for TLS/SSL support on hostpython3 and recipe)
-- openjdk-8
-- patch
-- python3
-- unzip
-- virtualenv (can be installed via pip)
-- zlib (including 32 bit)
-- zip
-
-On recent versions of Ubuntu and its derivatives you may be able to
-install most of these with::
-
- sudo dpkg --add-architecture i386
- sudo apt-get update
- sudo apt-get install -y build-essential ccache git zlib1g-dev python3 python3-dev libncurses5:i386 libstdc++6:i386 zlib1g:i386 openjdk-8-jdk unzip ant ccache autoconf libtool libssl-dev
-
-On Arch Linux you should be able to run the following to
-install most of the dependencies (note: this list may not be
-complete)::
+p4a requires a few dependencies to be installed on your system to work
+properly. While we're working on a way to automate pre-requisites checks,
+suggestions and installation on all platforms (macOS is already supported),
+on Linux distros you'll need to install them manually.
- sudo pacman -S core/autoconf core/automake core/gcc core/make core/patch core/pkgconf extra/cmake extra/jdk8-openjdk extra/python-pip extra/unzip extra/zip
+On recent versions of Ubuntu and its derivatives you can easily install them via
+the following command (re-adapted from the `Dockerfile` we use to perform CI builds)::
-On macOS::
+ sudo apt-get update
+ sudo apt-get install -y \
+ ant \
+ autoconf \
+ automake \
+ ccache \
+ cmake \
+ g++ \
+ gcc \
+ git \
+ lbzip2 \
+ libffi-dev \
+ libltdl-dev \
+ libtool \
+ libssl-dev \
+ make \
+ openjdk-17-jdk \
+ patch \
+ pkg-config \
+ python3 \
+ python3-dev \
+ python3-pip \
+ python3-venv \
+ sudo \
+ unzip \
+ wget \
+ zip
- brew install autoconf automake libtool openssl pkg-config
- brew tap homebrew/cask-versions
- brew install --cask homebrew/cask-versions/adoptopenjdk8
Installing Android SDK
~~~~~~~~~~~~~~~~~~~~~~
diff --git a/pythonforandroid/prerequisites.py b/pythonforandroid/prerequisites.py
index d5e013f11d..e85991948f 100644
--- a/pythonforandroid/prerequisites.py
+++ b/pythonforandroid/prerequisites.py
@@ -151,7 +151,7 @@ class JDKPrerequisite(Prerequisite):
name = "JDK"
mandatory = dict(linux=False, darwin=True)
installer_is_supported = dict(linux=False, darwin=True)
- min_supported_version = 11
+ supported_version = 17
def darwin_checker(self):
if "JAVA_HOME" in os.environ:
@@ -197,21 +197,21 @@ def _darwin_jdk_is_supported(self, jdk_path):
res = _stdout_res.strip().decode()
major_version = int(res.split(" ")[-1].split(".")[0])
- if major_version >= self.min_supported_version:
+ if major_version == self.supported_version:
info(f"Found a valid JDK at {jdk_path}")
return True
else:
- error(f"JDK {self.min_supported_version} or higher is required")
+ error(f"JDK version {major_version} is not supported")
return False
def darwin_helper(self):
info(
- "python-for-android requires a JDK 11 or higher to be installed on macOS,"
+ f"python-for-android requires a JDK {self.supported_version} to be installed on macOS,"
"but seems like you don't have one installed."
)
info(
"If you think that a valid JDK is already installed, please verify that "
- "you have a JDK 11 or higher installed and that `/usr/libexec/java_home` "
+ f"you have a JDK {self.supported_version} installed and that `/usr/libexec/java_home` "
"shows the correct path."
)
info(
@@ -221,12 +221,12 @@ def darwin_helper(self):
def darwin_installer(self):
info(
- "Looking for a JDK 11 or higher installation which is not the default one ..."
+ f"Looking for a JDK {self.supported_version} installation which is not the default one ..."
)
- jdk_path = self._darwin_get_libexec_jdk_path(version="11+")
+ jdk_path = self._darwin_get_libexec_jdk_path(version=str(self.supported_version))
if not self._darwin_jdk_is_supported(jdk_path):
- info("We're unlucky, there's no JDK 11 or higher installation available")
+ info(f"We're unlucky, there's no JDK {self.supported_version} or higher installation available")
base_url = "https://github.com/adoptium/temurin17-binaries/releases/download/jdk-17.0.2%2B8/"
if platform.machine() == "arm64":
From 0df46c8ccce849da6eb11739e5063653f21352cb Mon Sep 17 00:00:00 2001
From: shyamnathp <shyamnath94.mec@gmail.com>
Date: Fri, 8 Dec 2023 13:41:48 +0100
Subject: [PATCH 013/158] Remove redundant append into `WHITELIST_PATTERNS`
(#2935)
- pyconfig.h was appended twice into WHITELIST_PATTERNS
- Regression from commit a636b88
Co-authored-by: Shyamnath Premnadh <Shyamnath.Premnadh@qt.io>
---
pythonforandroid/bootstraps/common/build/build.py | 2 --
1 file changed, 2 deletions(-)
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index 59c38b609c..eced5886c8 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -1052,6 +1052,4 @@ def _read_configuration():
if __name__ == "__main__":
- if get_bootstrap_name() in ('sdl2', 'webview', 'service_only'):
- WHITELIST_PATTERNS.append('pyconfig.h')
parse_args_and_make_package()
From bb63f7684b177eab26da3dac1660a5fe526020bd Mon Sep 17 00:00:00 2001
From: Julian <github@somethinkodd.com>
Date: Sun, 17 Dec 2023 02:18:10 +1100
Subject: [PATCH 014/158] Update Contributing Guidelines and Readme (#2922)
* Update Contributing Guidelines and Readme
### Contributing Guidelines
* Refer to Kivy for the heavy-lifting, and discuss the *differences* with Kivy.
* Made wording more consistent with other sources.
* Removed empty promise of a release every 4 weeks. That isn't a sustainable rate.
* Very mild copy-edit.
### ReadMe
* Refer to Kivy for details of support, rather than repeating.
* Refer to p4a Contributing Guidelines, rather than repeating
* Refer to Quickstart rather than repeating
* Increased space for all the current sponsors and provisioned a far way into the future.
* Could have used an SVG here, but the comment promised we would include a link :-(
* Update for review comments
Use nicer alternatives for URLs.
Move History section from README to Contributing Guidelines.
* Make doc structure consistent and up-to-date
This is part of an effort to make the Kivy sibling projects' documentation
structure consistent and up-to-date.
CHECKLIST
* CONTRIBUTING.md
[x] If repo takes user contributions, is present
[x] In root dir (not .github dir)
[x] Explains relationship to Kivy, if unclear.
[x] Refers to kivy/kivy Contribution Guidelines.
MORE TO GO HERE
* LICENSE
[x] If repo takes user contributions, is present.
[x] Acknowledges the range of years to 2023.
[x] Acknowledges Kivy Team and other contributors
[x] Mentions it is an MIT License.
* CODE_OF_CONDUCT.md
[x] If repo takes user contributions or hosts discussions, is present.
[x] Refers to kivy/kivy CODE_OF_CONDUCT.md
* CONTACT.md
[x] Refers to kivy/kivy CONTACT.md
* FAQ.md
[x] If repo is big enough for RST documentation, is present.
* README:
[x] Is a Markdown file.
[x] Describes the project.
[x] Describes its place in the Kivy sibling projects.
[x] If Documentation exists, mention it.
[x] If CONTRIBUTING exists, mentions it.
[x] If LICENSE exists, mentions it.
[x] If CODE_OF_CONDUCT exists, mentions it.
[x] Mentions kivy/kivy CONTACT.md
[NA] Uses Python syntax colouring for embedded Python code.
[x] Uses badges to display current status, including:
[x] Backers
[x] Sponsors
[x] GitHub contributors
[x] Contributor Covenant
[x] PyPI Version
[x] PyPI Python Version
[x] Build/Test status
[x] Displays all contributors to the repo.
[x] Displays backers
[x] Displays top sponsors.
* RST documentation, if present
[x] Describes the project.
[x] Describes its place in the Kivy sibling projects.
[x] Mentions (Kivy/Kivy) Contact Us link.
[x] Mentions LICENSE.
[x] Mentions CONTRIBUTING
[x] Mentions FAQ
[x] conf.py mentioned Kivy Team and other contributors
- copyright, latex_documents, man_pages, texinfo documents
* WORKFLOWS
[x] NO_RESPONSE.yml is present if the repo has awaiting_reply tag.
[x] NO_RESPONSE uses latest script versions.
[x] NO_RESPONSE runs every day, and skips if forked.
[x] SUPPORT.yml is present if the repo has a `support` tag.
[x] SUPPORT.yml refers to repo's CONTACT.md
* setup.py/cfg, if present and on PyPI
[x] Supplies description to PyPI
[x] Supplies Python versions to PyPI
[x] Supplies Documentation, if any, to PyPI
* PEP8 fixes
* Rewrote description of binary formats
Also cleared merge artifact.
This was a response to review comments.
---
.github/workflows/no-response.yml | 32 ++++
.github/workflows/support.yml | 33 ++--
CODE_OF_CONDUCT.md | 8 +
CONTACT.md | 6 +
CONTRIBUTING.md | 250 ++++++++++++++++++++++++++++++
FAQ.md | 51 +++++-
LICENSE | 4 +-
README.md | 192 ++++++++++-------------
doc/source/conf.py | 6 +-
doc/source/contact.rst | 6 +
doc/source/contribute.rst | 240 ++--------------------------
doc/source/faq.rst | 5 +
doc/source/index.rst | 63 +++++---
doc/source/troubleshooting.rst | 87 +----------
setup.py | 35 +++--
15 files changed, 556 insertions(+), 462 deletions(-)
create mode 100644 .github/workflows/no-response.yml
create mode 100644 CODE_OF_CONDUCT.md
create mode 100644 CONTACT.md
create mode 100644 CONTRIBUTING.md
create mode 100644 doc/source/contact.rst
create mode 100644 doc/source/faq.rst
diff --git a/.github/workflows/no-response.yml b/.github/workflows/no-response.yml
new file mode 100644
index 0000000000..343d344560
--- /dev/null
+++ b/.github/workflows/no-response.yml
@@ -0,0 +1,32 @@
+name: No Response
+
+# Both `issue_comment` and `scheduled` event types are required for this Action
+# to work properly.
+on:
+ issue_comment:
+ types: [created]
+ schedule:
+ # Schedule for an arbitrary time (5am) once every day
+ - cron: '* 5 * * *'
+
+jobs:
+ noResponse:
+ # Don't run if in a fork
+ if: github.repository_owner == 'kivy'
+ runs-on: ubuntu-latest
+ steps:
+ - uses: lee-dohm/no-response@9bb0a4b5e6a45046f00353d5de7d90fb8bd773bb
+ # This commit hash targets release v0.5.0 of lee-dohm/no-response.
+ # Targeting a commit hash instead of a tag has been done for security reasons.
+ # Please be aware that the commit hash specifically targets the "Automatic compilation"
+ # done by `github-actions[bot]` as the `no-response` Action needs to be compiled.
+ with:
+ token: ${{ github.token }}
+ daysUntilClose: 42
+ responseRequiredLabel: 'awaiting-reply'
+ closeComment: >
+ This issue has been automatically closed because there has been no response
+ to our request for more information from the original author. With only the
+ information that is currently in the issue, we don't have the means
+ to take action. Please reach out if you have or find the answers we need so
+ that we can investigate further.
diff --git a/.github/workflows/support.yml b/.github/workflows/support.yml
index 2907000010..a0693e7632 100644
--- a/.github/workflows/support.yml
+++ b/.github/workflows/support.yml
@@ -1,3 +1,8 @@
+# When a user creates an issue that is actually a support request, it should
+# be closed with a friendly comment.
+#
+# This triggers on an issue being labelled with the `support` tag.
+
name: 'Support Requests'
on:
@@ -11,22 +16,30 @@ jobs:
action:
runs-on: ubuntu-latest
steps:
- - uses: dessant/support-requests@v2
+ - uses: dessant/support-requests@v4
with:
github-token: ${{ github.token }}
support-label: 'support'
issue-comment: >
- 👋 We use the issue tracker exclusively for bug reports and feature requests.
- However, this issue appears to be a support request. Please use our
- [support channels](https://github.com/kivy/python-for-android/blob/master/README.md#support)
- to get help with the project.
-
-
- If you're having trouble installing or using python-for-android,
- maybe you could be interested in our [quickstart](https://python-for-android.readthedocs.io/en/latest/quickstart) guide.
+ 👋 @{issue-author},
+ Sorry to hear you are having difficulties with Kivy's python-for-android; Kivy unites a number of different technologies, so building apps can be temperamental.
+
+ We try to use GitHub issues only to track work for developers to do to fix bugs and add new features to python-for-android.
+ However, this issue appears to be a support request. Please use our
+ [support channels](https://github.com/kivy/python-for-android/blob/master/CONTACT.md)
+ to get help with the project.
+
+ If you're having trouble installing python-for-android,
+ please see our [quickstart](https://python-for-android.readthedocs.io/en/latest/quickstart) guide.
+
+ If you're having trouble using python-for-android,
+ please see our [troubleshooting guide](https://python-for-android.readthedocs.io/en/latest/troubleshooting)
+ and [FAQ](https://github.com/kivy/python-for-android/blob/master/FAQ.md).
+
Let us know if this comment was made in error, and we'll be happy
to reopen the issue.
+
close-issue: true
- lock-issue: false
\ No newline at end of file
+ lock-issue: false
diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md
new file mode 100644
index 0000000000..29d18faa24
--- /dev/null
+++ b/CODE_OF_CONDUCT.md
@@ -0,0 +1,8 @@
+In the interest of fostering an open and welcoming community, we as
+contributors and maintainers need to ensure participation in our project and
+our sister projects is a harassment-free and positive experience for everyone.
+It is vital that all interaction is conducted in a manner conveying respect,
+open-mindedness and gratitude.
+
+Please consult the [latest Kivy Code of Conduct](https://github.com/kivy/kivy/blob/master/CODE_OF_CONDUCT.md).
+
diff --git a/CONTACT.md b/CONTACT.md
new file mode 100644
index 0000000000..c778a77a17
--- /dev/null
+++ b/CONTACT.md
@@ -0,0 +1,6 @@
+# Contacting the Kivy Team
+
+If you are looking to contact the Kivy Team (who are responsible for managing
+the python-for-android project), including looking for support, please see our
+latest [Contact Us](https://github.com/kivy/kivy/blob/master/CONTACT.md)
+document.
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
new file mode 100644
index 0000000000..be864e06f0
--- /dev/null
+++ b/CONTRIBUTING.md
@@ -0,0 +1,250 @@
+# Contribution Guidelines
+
+python-for-android is part of the [Kivy](https://kivy.org) ecosystem - a large group of
+products used by many thousands of developers for free, but it
+is built entirely by the contributions of volunteers. We welcome (and rely on)
+users who want to give back to the community by contributing to the project.
+
+Contributions can come in many forms. See the latest
+[Contribution Guidelines](https://github.com/kivy/kivy/blob/master/CONTRIBUTING.md)
+for how you can help us.
+
+.. warning::
+ The python-for-android process differs in small but important ways from the
+ Kivy framework's process. See below.
+
+## Development model
+
+Unlike the Kivy framework, python-for-android is developed using the following
+model:
+
+- The `master` branch always represents the latest stable release.
+- The `develop` branch is the most up to date with new contributions.
+- Releases happen periodically, and consist of merging the current `develop`
+ branch into `master`.
+
+This means pull requests for python-for-android code and documentation
+submissions should be made to the `develop` branch, not the `master` branch.
+
+For reference, this is based on a
+[Git flow](https://nvie.com/posts/a-successful-git-branching-model/) model,
+although we don't follow this religiously.
+
+## Versioning
+
+python-for-android releases currently use
+[calendar versioning](https://calver.org/). Release numbers are of the form
+YYYY.MM.DD.
+
+We use calendar versioning because in practice, changes in
+python-for-android are often driven by updates or adjustments in the
+Android build tools. It's usually best for users to be working from
+the latest release. We try to maintain backwards compatibility even
+while internals are changing.
+
+## History
+
+In 2015, these tools were rewritten to provide a new, easier-to-use and
+easier-to-extend interface. If you'd like to browse the old toolchain,
+its status is
+[recorded for posterity](https://github.com/kivy/python-for-android/tree/old_toolchain).
+
+In the last quarter of 2018, the Python recipes were changed. The
+new recipe for Python3 (3.7.1) had a new build system which was
+applied to the ancient Python recipe, allowing us to bump the Python2
+version number to 2.7.15. This change unified the build process for
+both Python recipes, and probably solved various issues detected over the
+years. These **unified Python recipes** require a **minimum target api level of 21**,
+*Android 5.0 - Lollipop*. If you need to build targeting an
+api level below 21, you should use an older version of python-for-android
+(<=0.7.1).
+
+On March 2020, we dropped support for creating apps that use Python 2. The
+latest python-for-android release that supported building Python 2 was version
+2019.10.6.
+
+On August 2021, we added support for Android App Bundle (aab). As a
+collateral benefit, we now support multi-arch apk.
+
+## Creating a new release
+
+(These instructions are for core developers, not casual contributors.)
+
+New releases follow these steps:
+
+- Create a new branch `release-YYYY.MM.DD` based on the `develop` branch.
+ - `git checkout -b release-YYYY.MM.DD develop`
+- Create a Github pull request to merge `release-YYYY.MM.DD` into `master`.
+- Complete all steps in the [release checklist](#Release_checklist),
+ and document this in the pull request (copy the checklist into the PR text)
+
+At this point, wait for reviewer approval and conclude any discussion that
+arises. To complete the release:
+
+- Merge the release branch to the `master` branch.
+- Also merge the release branch to the `develop` branch.
+- Tag the release commit in `master`, with tag `vYYYY.MM.DD`. Include a short
+ summary of the changes.
+- Release distributions and PyPI upload should be
+ [handled by the CI](https://github.com/kivy/python-for-android/blob/v2020.04.29/.travis.yml#L60-L70).
+- Add to the GitHub release page (see e.g. [this example](https://github.com/kivy/python-for-android/releases/tag/v2019.06.06):
+ - The python-for-android README summary
+ - A short list of major changes in this release, if any
+ - A changelog summarising merge commits since the last release
+ - The release sdist and wheel(s)
+
+## Release checklist
+
+ - [ ] Check that the builds are passing
+ - [ ] [GitHub Action](https://github.com/kivy/python-for-android/actions)
+ - [ ] Run the tests locally via `tox`: this performs some long-running tests that are skipped on github-actions.
+ - [ ] Build and run the [on_device_unit_tests](https://github.com/kivy/python-for-android/tree/master/testapps/on_device_unit_tests) app using buildozer. Check that they all pass.
+ - [ ] Build (or download from github actions) and run the following [testapps](https://github.com/kivy/python-for-android/tree/master/testapps/on_device_unit_tests) for arch `armeabi-v7a` and `arm64-v8a`:
+ - [ ] on_device_unit_tests
+ - [ ] `armeabi-v7a` (`cd testapps/on_device_unit_tests && PYTHONPATH=.:../../ python3 setup.py apk --ndk-dir=<your-ndk-dir> --sdk-dir=<your-sdk-dir> --arch=armeabi-v7a --debug`)
+ - [ ] `arm64-v8a` (`cd testapps/on_device_unit_tests && PYTHONPATH=.:../../ python3 setup.py apk --ndk-dir=<your-ndk-dir> --sdk-dir=<your-sdk-dir> --arch=arm64-v8a --debug`)
+ - [ ] Check that the version number is correct
+
+## How python-for-android uses `pip`
+
+*Last update: July 2019*
+
+This section is meant to provide a quick summary how
+python-for-android uses pip and Python packages in
+its build process.
+**It is written for a Python
+packager's point of view, not for regular end users or
+contributors,** to assist with making pip developers and
+other packaging experts aware of p4a's packaging needs.
+
+Please note this section just attempts to neutrally list the
+current mechanisms, so some of this isn't necessarily meant
+to stay but just how things work inside p4a in
+this very moment.
+
+
+### Basic concepts
+
+*(This part repeats other parts of the docs, for the sake of
+making this a more independent read)*
+
+p4a builds & packages a Python application for use on Android.
+It does this by providing a Java wrapper, and for graphical applications
+an SDL2-based wrapper which can be used with the Kivy framework if
+desired (or alternatively just plain PySDL2). Any such the Python application
+will likely have further library dependencies to do its work.
+
+p4a supports two types of package dependencies for a project:
+
+**Recipe:** Install a script in custom p4a format. Can either install
+C/C++ or other software that cannot be pulled in via pip, or software
+that can be installed via pip but break on Android by default.
+These are maintained primarily inside the p4a source tree by p4a
+contributors and interested folks.
+
+**Python package:** any random pip python package can be directly
+installed if it doesn't need adjustments to work for Android.
+
+p4a will map any dependency to an internal recipe if present, and
+otherwise use pip to obtain it regularly from whatever external source.
+
+
+### Install process regarding packages
+
+The install/build process of a p4a project, as triggered by the
+`p4a apk` command, roughly works as follows in regards to Python
+packages:
+
+1. The user has specified a project folder to install. This is either
+ just a folder with Python scripts and a `main.py`, or it may
+ also have a `pyproject.toml` for a more standardized install.
+
+2. Dependencies are collected: they can be either specified via
+ ``--requirements`` as a list of names or pip-style URLs, or p4a
+ can optionally scan them from a project folder via the
+ pep517 library (if there is a `pyproject.toml` or `setup.py`).
+
+3. The collected dependencies are mapped to p4a's recipes if any are
+ available for them, otherwise they're kept around as external
+ regular package references.
+
+4. All the dependencies mapped to recipes are built via p4a's internal
+ mechanisms to build these recipes. (This may or may not indirectly
+ use pip, depending on whether the recipe wraps a python package
+ or not and uses pip to install or not.)
+
+5. **If the user has specified to install the project in standardized
+ ways,** then the `setup.py`/whatever build system
+ of the project will be run. This happens with cross compilation set up
+ (`CC`/`CFLAGS`/... set to use the
+ proper toolchain) and a custom site-packages location.
+ The actual comand is a simple `pip install .` in the project folder
+ with some extra options: e.g. all dependencies that were already
+ installed by recipes will be pinned with a `-c` constraints file
+ to make sure pip won't install them, and build isolation will be
+ disabled via ``--no-build-isolation`` so pip doesn't reinstall
+ recipe-packages on its own.
+
+ **If the user has not specified to use standardized build approaches**,
+ p4a will simply install all the remaining dependencies that weren't
+ mapped to recipes directly and just plain copy in the user project
+ without installing. Any `setup.py` or `pyproject.toml` of the user
+ project will then be ignored in this step.
+
+6. Google's gradle is invoked to package it all up into an `.apk`.
+
+
+### Overall process / package relevant notes for p4a
+
+Here are some common things worth knowing about python-for-android's
+dealing with python packages:
+
+- Packages will work fine without a recipe if:
+
+ * they would also build on Linux ARM,
+ * don't use any API not available in the NDK if they use native code, and
+ * don't use any weird compiler flags the toolchain doesn't like if they use native code.
+ * works with cross compilation.
+
+- There is currently no easy way for a package to know it is being
+ cross-compiled (at least that we know of) other than examining the
+ `CC` compiler that was set, or that it is being cross-compiled for
+ Android specifically. If that breaks a package, it currently needs
+ to be worked around with a recipe.
+
+- If a package does **not** work, p4a developers will often create a
+ recipe instead of getting upstream to fix it because p4a simply
+ is too niche.
+
+- Most packages without native code will just work out of the box.
+ Many with native code tend not to, especially if complex, e.g. numpy.
+
+- Anything mapped to a p4a recipe cannot be just reinstalled by pip,
+ specifically also not inside build isolation as a dependency.
+ (It *may* work if the patches of the recipe are just relevant
+ to fix runtime issues.)
+ Therefore as of now, the best way to deal with this limitation seems
+ to be to keep build isolation always off.
+
+
+### Ideas for the future regarding packaging
+
+- We in overall prefer to use the recipe mechanism less if we can.
+ Overall, the recipes are just a collection of workarounds.
+ It may look quite hacky from the outside, since p4a
+ version pins recipe-wrapped packages usually to make the patches reliably
+ apply. This creates work for the recipes to be kept up-to-date, and
+ obviously this approach doesn't scale too well. However, it has ended
+ up as a quite practical interim solution until better ways are found.
+
+- Obviously, it would be nice if packages could know they are being
+ cross-compiled, and for Android specifically. We aren't currently aware
+ of any good mechanism for that.
+
+- If pip could actually run the recipes (instead of p4a wrapping pip and
+ doing so) then this might even allow build isolation to work - but
+ this might be too complex to get working. It might be more practical
+ to just gradually reduce the reliance on recipes instead and make
+ more packages work out of the box. This has been done e.g. with
+ improvements to the cross-compile environment being set up automatically,
+ and we're open for any ideas on how to improve this.
diff --git a/FAQ.md b/FAQ.md
index 9e37084375..7450a6c44d 100644
--- a/FAQ.md
+++ b/FAQ.md
@@ -2,10 +2,8 @@
## Introduction
-[python-for-android](https://python-for-android.readthedocs.io/) is an open
-source build tool to let you package Python code into standalone Android artifacts (APKs, AABs, AARs).
-These can be passed around, installed, or uploaded to marketplaces such as the
-Play Store just like any other Android app.
+python-for-android (p4a) is a development tool that packages Python apps into
+binaries that can run on Android devices.
### Sibling Projects:
@@ -22,7 +20,7 @@ number of target platforms, using a specification file to define the build.
### Is it possible to have a kiosk app on Android?
Thomas Hansen wrote a detailed answer
-on [the kivy-users mailing list](https://groups.google.com/d/msg/kivy-users/QKoCekAR1c0/yV-85Y_iAwoJ)
+on [the (old) kivy-users mailing list](https://groups.google.com/d/msg/kivy-users/QKoCekAR1c0/yV-85Y_iAwoJ)
Basically, you need to root the device, remove the SystemUI package, add some
lines to the xml configuration, and you're done.
@@ -71,3 +69,46 @@ On macOS:
brew install openssl
p4a clean builds # or with: buildozer `buildozer android clean
+
+
+#### AttributeError: 'AnsiCodes' object has no attribute 'LIGHTBLUE_EX'
+
+This occurs if your version of `colorama` is too low, install version
+0.3.3 or higher.
+
+If you install python-for-android with `pip` or via `setup.py`, this
+dependency should be taken care of automatically.
+
+#### AttributeError: 'Context' object has no attribute 'hostpython'
+
+This is a known bug in some releases. To work around it, add your
+python requirement explicitly,
+e.g. :code:`--requirements=python3,kivy`. This also applies when using
+buildozer, in which case add python3 to your buildozer.spec requirements.
+
+#### linkname too long
+
+This can happen when you try to include a very long filename, which
+doesn't normally happen but can occur accidentally if the p4a
+directory contains a .buildozer directory that is not excluded from
+the build (e.g. if buildozer was previously used). Removing this
+directory should fix the problem, and is desirable anyway since you
+don't want it in the APK.
+
+#### Requested API target 19 is not available, install it with the SDK android tool
+
+This means that your SDK is missing the required platform tools. You
+need to install the ``platforms;android-19`` package in your SDK,
+using the ``android`` or ``sdkmanager`` tools (depending on SDK
+version).
+
+If using buildozer this should be done automatically, but as a
+workaround you can run these from
+``~/.buildozer/android/platform/android-sdk-20/tools/android``.
+
+#### ModuleNotFoundError: No module named '_ctypes'
+
+You do not have the libffi headers available to python-for-android, so you need to install them. On Ubuntu and derivatives these come from the `libffi-dev` package.
+
+After installing the headers, clean the build (`p4a clean builds`, or with buildozer delete the `.buildozer` directory within your app directory) and run python-for-android again.
+
diff --git a/LICENSE b/LICENSE
index d5d6b13c85..4e3506010a 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,4 +1,6 @@
-Copyright (c) 2010-2017 Kivy Team and other contributors
+MIT License
+
+Copyright (c) 2010-2023 Kivy Team and other contributors
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
diff --git a/README.md b/README.md
index 764ea7b7a7..c7cbb96c9a 100644
--- a/README.md
+++ b/README.md
@@ -1,144 +1,124 @@
-python-for-android
-==================
+# python-for-android
+
+python-for-android (p4a) is a development tool that packages Python apps into
+binaries that can run on Android devices.
+
+It can generate:
+
+* [Android Package](https://en.wikipedia.org/wiki/Apk_(file_format)) (APK)
+ files, ready to install locally on a device, especially for testing. This format
+ is used by many [app stores](https://en.wikipedia.org/wiki/List_of_Android_app_stores)
+ but not [Google Play Store](https://play.google.com/store/).
+* [Android App Bundle](https://developer.android.com/guide/app-bundle/faq)
+ (AAB) files which can be shared on [Google Play Store](https://play.google.com/store/).
+* [Android Archive](https://developer.android.com/studio/projects/android-library)
+ (AAR) files which can be used as a re-usable bundle of resources for other
+ projects.
+
+It supports multiple CPU architectures.
+
+It supports apps developed with [Kivy framework](http://kivy.org), but was
+built to be flexible about the backend libraries (through "bootstraps"), and
+also supports [PySDL2](https://pypi.org/project/PySDL2/), and a
+[WebView](https://developer.android.com/reference/android/webkit/WebView) with
+a Python web server.
+
+It automatically supports dependencies on most pure Python packages. For other
+packages, including those that depend on C code, a special "recipe" must be
+written to support cross-compiling. python-for-android comes with recipes for
+many of the mosty popular libraries (e.g. numpy and sqlalchemy) built in.
+
+python-for-android works by cross-compiling the Python interpreter and its
+dependencies for Android devices, and bundling it with the app's python code
+and dependencies. The Python code is then interpreted on the Android device.
+
+It is recommended that python-for-android be used via
+[Buildozer](https://buildozer.readthedocs.io/), which ensures the correct
+dependencies are pre-installed, and centralizes the configuration. However,
+python-for-android is not limited to being used with Buildozer.
-[](https://github.com/kivy/python-for-android/actions?query=workflow%3A%22Unit+tests+%26+build+apps%22)
-[](https://coveralls.io/github/kivy/python-for-android?branch=develop)
[](#backers)
[](#sponsors)
+[](https://github.com/kivy/python-for-android/graphs/contributors)
+[](CODE_OF_CONDUCT.md)
-python-for-android is a packaging tool for Python apps on Android. You can
-create your own Python distribution including the modules and
-dependencies you want, and bundle it in an APK or AAB along with your own code.
-
-Features include:
-
-- Different app backends including Kivy, PySDL2, and a WebView with
- Python webserver.
-- Automatic support for most pure Python modules, and built in support
- for many others, including popular dependencies such as numpy and
- sqlalchemy.
-- Multiple architecture targets, for APKs optimised on any given
- device.
-- AAB: Android App Bundle support.
+
+
-For documentation and support, see:
-
-- Website: http://python-for-android.readthedocs.io
-- Mailing list: https://groups.google.com/forum/#!forum/kivy-users or
- https://groups.google.com/forum/#!forum/python-android.
+[](https://github.com/kivy/python-for-android/actions?query=workflow%3A%22Unit+tests+%26+build+apps%22)
+[](https://coveralls.io/github/kivy/python-for-android?branch=develop)
## Documentation
-Follow the [quickstart
-instructions](<https://python-for-android.readthedocs.org/en/latest/quickstart/>)
-to install and begin creating APKs and AABs.
-
-**Quick instructions**: install python-for-android with:
-
- pip install python-for-android
-
-(for the develop branch: `pip install git+https://github.com/kivy/python-for-android.git`)
-
-Test that the install works with:
-
- p4a --version
-
-To build any actual apps, **set up the Android SDK and NDK**
-as described in the [quickstart](
-<https://python-for-android.readthedocs.org/en/latest/quickstart/#installing-android-sdk>).
-**Use the SDK/NDK API level & NDK version as in the quickstart,**
-other API levels may not work.
-
-With everything installed, build an APK with SDL2 with e.g.:
-
- p4a apk --private PATH_TO_YOUR_APP_CODE --package=org.example.myapp --name "My application" --version 0.1 --bootstrap=sdl2 --requirements=python3,kivy
-
-**If you need to deploy your app on Google Play, Android App Bundle (aab) is required since 1 August 2021:**
+More information is available in the
+[online documentation](https://python-for-android.readthedocs.io) including a
+[quickstart guide](https://python-for-android.readthedocs.io/en/latest/quickstart/).
-**For full instructions and parameter options,** see [the
-documentation](https://python-for-android.readthedocs.io/en/latest/quickstart/#usage).
+python-for-android is managed by the [Kivy team](https://kivy.org).
## Support
-If you need assistance, you can ask for help on our mailing list:
+Are you having trouble using python-for-android or any of its related projects
+in the Kivy ecosystem?
+Is there an error you don’t understand? Are you trying to figure out how to use
+it? We have volunteers who can help!
-- User Group: https://groups.google.com/group/kivy-users
-- Email: kivy-users@googlegroups.com
+The best channels to contact us for support are listed in the latest
+[Contact Us](https://github.com/kivy/pyton-for-android/blob/master/CONTACT.md)
+document.
-We also have [#support Discord channel](https://chat.kivy.org/).
+## Code of Conduct
-## Contributing
+In the interest of fostering an open and welcoming community, we as
+contributors and maintainers need to ensure participation in our project and
+our sister projects is a harassment-free and positive experience for everyone.
+It is vital that all interaction is conducted in a manner conveying respect,
+open-mindedness and gratitude.
-We love pull requests and discussing novel ideas. Check out the Kivy
-project [contribution guide](https://kivy.org/doc/stable/contribute.html) and
-feel free to improve python-for-android.
-
-See [our
-documentation](https://python-for-android.readthedocs.io/en/latest/contribute/)
-for more information about the python-for-android development and
-release model, but don't worry about the details. You just need to
-make a pull request, we'll take care of the rest.
-
-The following mailing list and IRC channel are used exclusively for
-discussions about developing the Kivy framework and its sister projects:
-
-- Dev Group: https://groups.google.com/group/kivy-dev
-- Email: kivy-dev@googlegroups.com
-
-We also have [#dev Discord channel](https://chat.kivy.org/).
-
-## License
-
-python-for-android is released under the terms of the MIT License.
-Please refer to the LICENSE file.
-
-## History
-
-In 2015 these tools were rewritten to provide a new, easier-to-use and
-easier-to-extend interface. If you'd like to browse the old toolchain, its
-status is recorded for posterity at
-https://github.com/kivy/python-for-android/tree/old_toolchain.
-
-In the last quarter of 2018 the python recipes were changed. The
-new recipe for python3 (3.7.1) had a new build system which was
-applied to the ancient python recipe, allowing us to bump the python2
-version number to 2.7.15. This change unified the build process for
-both python recipes, and probably solved various issues detected over the
-years. These **unified python recipes** require a **minimum target api level of 21**,
-*Android 5.0 - Lollipop*. If you need to build targeting an
-api level below 21, you should use an older version of python-for-android
-(<=0.7.1).
-
-On March of 2020 we dropped support for creating apps that use Python 2. The latest
-python-for-android release that supported building Python 2 was version 2019.10.6.
-
-On August of 2021, we added support for Android App Bundle (aab). As a collateral,
-now We support multi-arch apk.
+Please consult the [latest Code of Conduct](https://github.com/kivy/python-for-android/blob/master/CODE_OF_CONDUCT.md).
## Contributors
-This project exists thanks to all the people who contribute. [[Contribute](https://kivy.org/doc/stable/contribute.html)].
-<a href="https://github.com/kivy/python-for-android/graphs/contributors"><img src="https://opencollective.com/kivy/contributors.svg?width=890&button=false" /></a>
+This project exists thanks to
+[all the people who contribute](https://github.com/kivy/python-for-android/graphs/contributors).
+[[Become a contributor](CONTRIBUTING.md)].
+<img src="https://contrib.nn.ci/api?repo=kivy/python-for-android&pages=5&no_bot=true&radius=22&cols=18">
## Backers
-Thank you to all our backers! 🙏 [[Become a backer](https://opencollective.com/kivy#backer)]
-
-<a href="https://opencollective.com/kivy#backers" target="_blank"><img src="https://opencollective.com/kivy/backers.svg?width=890"></a>
+Thank you to [all of our backers](https://opencollective.com/kivy)!
+🙏 [[Become a backer](https://opencollective.com/kivy#backer)]
+<img src="https://opencollective.com/kivy/backers.svg?width=890&avatarHeight=44&button=false">
## Sponsors
-Support this project by becoming a sponsor. Your logo will show up here with a link to your website. [[Become a sponsor](https://opencollective.com/kivy#sponsor)]
+Special thanks to
+[all of our sponsors, past and present](https://opencollective.com/kivy).
+Support this project by
+[[becoming a sponsor](https://opencollective.com/kivy#sponsor)].
+Here are our top current sponsors. Please click through to see their websites,
+and support them as they support us.
+
+<!--- See https://github.com/orgs/kivy/discussions/15 for explanation of this code. -->
<a href="https://opencollective.com/kivy/sponsor/0/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/0/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/1/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/1/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/2/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/2/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/3/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/3/avatar.svg"></a>
+
<a href="https://opencollective.com/kivy/sponsor/4/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/4/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/5/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/5/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/6/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/6/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/7/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/7/avatar.svg"></a>
+
<a href="https://opencollective.com/kivy/sponsor/8/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/8/avatar.svg"></a>
<a href="https://opencollective.com/kivy/sponsor/9/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/9/avatar.svg"></a>
+<a href="https://opencollective.com/kivy/sponsor/10/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/10/avatar.svg"></a>
+<a href="https://opencollective.com/kivy/sponsor/11/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/11/avatar.svg"></a>
+
+<a href="https://opencollective.com/kivy/sponsor/12/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/12/avatar.svg"></a>
+<a href="https://opencollective.com/kivy/sponsor/13/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/13/avatar.svg"></a>
+<a href="https://opencollective.com/kivy/sponsor/14/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/14/avatar.svg"></a>
+<a href="https://opencollective.com/kivy/sponsor/15/website" target="_blank"><img src="https://opencollective.com/kivy/sponsor/15/avatar.svg"></a>
diff --git a/doc/source/conf.py b/doc/source/conf.py
index a355505748..773083f980 100644
--- a/doc/source/conf.py
+++ b/doc/source/conf.py
@@ -58,7 +58,7 @@
author = 'Kivy Team and other contributors'
-copyright = f'{_today.year}, {author}'
+copyright = f'2015-{_today.year}, {author}'
# The version info for the project you're documenting, acts as replacement for
# |version| and |release|, also used in various other places throughout the
@@ -290,7 +290,9 @@ def get_version():
# dir menu entry, description, category)
texinfo_documents = [
(master_doc, 'python-for-android', u'python-for-android Documentation',
- author, 'python-for-android', 'One line description of project.',
+ author, 'python-for-android',
+ 'A development tool that packages Python apps into binaries that can run on '
+ 'Android devices',
'Miscellaneous'),
]
diff --git a/doc/source/contact.rst b/doc/source/contact.rst
new file mode 100644
index 0000000000..fedc04b47f
--- /dev/null
+++ b/doc/source/contact.rst
@@ -0,0 +1,6 @@
+Contact Us
+==========
+
+If you are looking to contact the Kivy Team (who are responsible for managing the
+python-for-android project), including looking for support, please see our
+`latest contact details <https://github.com/kivy/python-for-android/blob/master/CONTACT.md>`_.
\ No newline at end of file
diff --git a/doc/source/contribute.rst b/doc/source/contribute.rst
index c66cd096ea..653c670081 100644
--- a/doc/source/contribute.rst
+++ b/doc/source/contribute.rst
@@ -1,231 +1,19 @@
-Development and Contributing
-============================
+.. _contributing:
-The development of python-for-android is managed by the Kivy team `via
-Github <https://github.com/kivy/python-for-android>`_.
+.. _contribute:
-Issues and pull requests are welcome via the integrated `issue tracker
-<https://github.com/kivy/python-for-android/issues>`_.
+Contribution Guidelines
+=======================
-Read on for more information about how we manage development and
-releases, but don't worry about the details! Pull requests are welcome
-and we'll deal with the rest.
+Buildozer is part of the `Kivy <https://kivy.org>`_ ecosystem - a large group of
+products used by many thousands of developers for free, but it
+is built entirely by the contributions of volunteers. We welcome (and rely on)
+users who want to give back to the community by contributing to the project.
-Development model
------------------
-
-python-for-android is developed using the following model:
-
-- The ``master`` branch always represents the latest stable release.
-- The ``develop`` branch is the most up to date with new contributions.
-- Releases happen periodically, and consist of merging the current ``develop`` branch into ``master``.
-
-For reference, this is based on a `Git flow
-<https://nvie.com/posts/a-successful-git-branching-model/>`__ model,
-although we don't follow this religiously.
-
-Versioning
-----------
-
-python-for-android releases currently use `calendar versioning
-<https://calver.org/>`__. Release numbers are of the form
-YYYY.MM.DD. We aim to create a new release every four weeks, but more
-frequent releases are also possible.
-
-We use calendar versioning because in practice, changes in
-python-for-android are often driven by updates or adjustments in the
-Android build tools. It's usually best for users to be working from
-the latest release. We try to maintain backwards compatibility even
-while internals are changing.
-
-
-Creating a new release
-----------------------
-
-New releases follow these steps:
-
-- Create a new branch ``release-YYYY.MM.DD`` based on the ``develop`` branch.
- - ``git checkout -b release-YYYY.MM.DD develop``
-- Create a Github pull request to merge ``release-YYYY.MM.DD`` into ``master``.
-- Complete all steps in the `release checklist <release_checklist_>`_,
- and document this in the pull request (copy the checklist into the PR text)
-
-At this point, wait for reviewer approval and conclude any discussion that arises. To complete the release:
-
-- Merge the release branch to the ``master`` branch.
-- Also merge the release branch to the ``develop`` branch.
-- Tag the release commit in ``master``, with tag ``vYYYY.MM.DD``. Include a short summary of the changes.
-- Release distributions and PyPI upload should be `handled by the CI
- <https://github.com/kivy/python-for-android/blob/v2020.04.29/.travis.yml#L60-L70>`_.
-- Add to the Github release page (see e.g. `this example <https://github.com/kivy/python-for-android/releases/tag/v2019.06.06>`__):
- - The python-for-android README summary
- - A short list of major changes in this release, if any
- - A changelog summarising merge commits since the last release
- - The release sdist and wheel(s)
-
-.. _release_checklist:
-
-Release checklist
-~~~~~~~~~~~~~~~~~
-::
-
- - [ ] Check that the builds are passing
- - [ ] [GitHub Action](https://github.com/kivy/python-for-android/actions)
- - [ ] Run the tests locally via `tox`: this performs some long-running tests that are skipped on github-actions.
- - [ ] Build and run the [on_device_unit_tests](https://github.com/kivy/python-for-android/tree/master/testapps/on_device_unit_tests) app using buildozer. Check that they all pass.
- - [ ] Build (or download from github actions) and run the following [testapps](https://github.com/kivy/python-for-android/tree/master/testapps/on_device_unit_tests) for arch `armeabi-v7a` and `arm64-v8a`:
- - [ ] on_device_unit_tests
- - [ ] `armeabi-v7a` (`cd testapps/on_device_unit_tests && PYTHONPATH=.:../../ python3 setup.py apk --ndk-dir=<your-ndk-dir> --sdk-dir=<your-sdk-dir> --arch=armeabi-v7a --debug`)
- - [ ] `arm64-v8a` (`cd testapps/on_device_unit_tests && PYTHONPATH=.:../../ python3 setup.py apk --ndk-dir=<your-ndk-dir> --sdk-dir=<your-sdk-dir> --arch=arm64-v8a --debug`)
- - [ ] Check that the version number is correct
-
-
-
-How python-for-android uses `pip`
----------------------------------
-
-*Last update: July 2019*
-
-This section is meant to provide a quick summary how
-p4a (=python-for-android) uses pip and python packages in
-its build process.
-**It is written for a python
-packagers point of view, not for regular end users or
-contributors,** to assist with making pip developers and
-other packaging experts aware of p4a's packaging needs.
-
-Please note this section just attempts to neutrally list the
-current mechanisms, so some of this isn't necessarily meant
-to stay but just how things work inside p4a in
-this very moment.
-
-
-Basic concepts
-~~~~~~~~~~~~~~
-
-*(This part repeats other parts of the docs, for the sake of
-making this a more independent read)*
-
-p4a builds & packages a python application for use on Android.
-It does this by providing a Java wrapper, and for graphical applications
-an SDL2-based wrapper which can be used with the kivy UI toolkit if
-desired (or alternatively just plain PySDL2). Any such python application
-will of course have further library dependencies to do its work.
-
-p4a supports two types of package dependencies for a project:
-
-**Recipe:** install script in custom p4a format. Can either install
-C/C++ or other things that cannot be pulled in via pip, or things
-that can be installed via pip but break on android by default.
-These are maintained primarily inside the p4a source tree by p4a
-contributors and interested folks.
-
-**Python package:** any random pip python package can be directly
-installed if it doesn't need adjustments to work for Android.
-
-p4a will map any dependency to an internal recipe if present, and
-otherwise use pip to obtain it regularly from whatever external source.
-
-
-Install process regarding packages
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-The install/build process of a p4a project, as triggered by the
-`p4a apk` command, roughly works as follows in regards to python
-packages:
-
-1. The user has specified a project folder to install. This is either
- just a folder with python scripts and a `main.py`, or it may
- also have a `pyproject.toml` for a more standardized install.
-
-2. Dependencies are collected: they can be either specified via
- ``--requirements`` as a list of names or pip-style URLs, or p4a
- can optionally scan them from a project folder via the
- pep517 library (if there is a `pyproject.toml` or `setup.py`).
-
-3. The collected dependencies are mapped to p4a's recipes if any are
- available for them, otherwise they're kept around as external
- regular package references.
-
-4. All the dependencies mapped to recipes are built via p4a's internal
- mechanisms to build these recipes. (This may or may not indirectly
- use pip, depending on whether the recipe wraps a python package
- or not and uses pip to install or not.)
-
-5. **If the user has specified to install the project in standardized
- ways,** then the `setup.py`/whatever build system
- of the project will be run. This happens with cross compilation set up
- (`CC`/`CFLAGS`/... set to use the
- proper toolchain) and a custom site-packages location.
- The actual comand is a simple `pip install .` in the project folder
- with some extra options: e.g. all dependencies that were already
- installed by recipes will be pinned with a `-c` constraints file
- to make sure pip won't install them, and build isolation will be
- disabled via ``--no-build-isolation`` so pip doesn't reinstall
- recipe-packages on its own.
-
- **If the user has not specified to use standardized build approaches**,
- p4a will simply install all the remaining dependencies that weren't
- mapped to recipes directly and just plain copy in the user project
- without installing. Any `setup.py` or `pyproject.toml` of the user
- project will then be ignored in this step.
-
-6. Google's gradle is invoked to package it all up into an `.apk`.
-
-
-Overall process / package relevant notes for p4a
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-Here are some common things worth knowing about python-for-android's
-dealing with python packages:
-
-- Packages will work fine without a recipe if they would also build
- on Linux ARM, don't use any API not available in the NDK if they
- use native code, and don't use any weird compiler flags the toolchain
- doesn't like if they use native code. The package also needs to
- work with cross compilation.
-
-- There is currently no easy way for a package to know it is being
- cross-compiled (at least that we know of) other than examining the
- `CC` compiler that was set, or that it is being cross-compiled for
- Android specifically. If that breaks a package it currently needs
- to be worked around with a recipe.
-
-- If a package does **not** work, p4a developers will often create a
- recipe instead of getting upstream to fix it because p4a simply
- is too niche.
-
-- Most packages without native code will just work out of the box.
- Many with native code tend not to, especially if complex, e.g. numpy.
-
-- Anything mapped to a p4a recipe cannot be just reinstalled by pip,
- specifically also not inside build isolation as a dependency.
- (It *may* work if the patches of the recipe are just relevant
- to fix runtime issues.)
- Therefore as of now, the best way to deal with this limitation seems
- to be to keep build isolation always off.
-
-
-Ideas for the future regarding packaging
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-- We in overall prefer to use the recipe mechanism less if we can.
- In overall the recipes are just a collection of workarounds.
- It may look quite hacky from the outside, since p4a
- version pins recipe-wrapped packages usually to make the patches reliably
- apply. This creates work for the recipes to be kept up-to-date, and
- obviously this approach doesn't scale too well. However, it has ended
- up as a quite practical interims solution until better ways are found.
-
-- Obviously, it would be nice if packages could know they are being
- cross-compiled, and for Android specifically. We aren't currently aware
- of a good mechanism for that.
-
-- If pip could actually run the recipes (instead of p4a wrapping pip and
- doing so) then this might even allow build isolation to work - but
- this might be too complex to get working. It might be more practical
- to just gradually reduce the reliance on recipes instead and make
- more packages work out of the box. This has been done e.g. with
- improvements to the cross-compile environment being set up automatically,
- and we're open for any ideas on how to improve this.
+Contributions can come in many forms. See the latest
+`Contribution Guidelines <https://github.com/kivy/python-for-android/blob/master/CONTRIBUTING.md>`_
+for general guidelines of how you can help us, and specific instructions for python-for-android
+development.
+.. warning::
+ The python-for-android process differs in small but important ways from the Kivy framework's process.
diff --git a/doc/source/faq.rst b/doc/source/faq.rst
new file mode 100644
index 0000000000..6527cf3c29
--- /dev/null
+++ b/doc/source/faq.rst
@@ -0,0 +1,5 @@
+FAQ
+===
+
+python-for-android has an `online FAQ <https://github.com/kivy/python-for-android/blob/master/FAQ.md>`_. It contains
+the answers to questions that repeatedly come up.
diff --git a/doc/source/index.rst b/doc/source/index.rst
index 5faf83da49..929fa384c7 100644
--- a/doc/source/index.rst
+++ b/doc/source/index.rst
@@ -1,23 +1,48 @@
python-for-android
==================
-python-for-android is an open source build tool to let you package
-Python code into standalone android APKs. These can be passed around,
-installed, or uploaded to marketplaces such as the Play Store just
-like any other Android app. This tool was originally developed for the
-`Kivy cross-platform graphical framework <https://kivy.org/>`_,
-but now supports multiple bootstraps and can be easily extended to
-package other types of Python apps for Android.
-
-python-for-android supports two major operations; first, it can
-compile the Python interpreter, its dependencies, backend libraries
-and python code for Android devices. This stage is fully customisable:
-you can install as many or few components as you like. The result is
-a standalone Android project which can be used to generate any number
-of different APKs, even with different names, icons, Python code etc.
-The second function of python-for-android is to provide a simple
-interface to these distributions, to generate from such a project a
-Python APK with build parameters and Python code to taste.
+python-for-android (p4a) is a development tool that packages Python apps into
+binaries that can run on Android devices.
+
+It can generate:
+
+* `Android Package <https://en.wikipedia.org/wiki/Apk_(file_format)>`_ (APK)
+ files, ready to install locally on a device, especially for testing. This format
+ is used by many `app stores <https://en.wikipedia.org/wiki/List_of_Android_app_stores>`_
+ but not `Google Play Store <https://play.google.com/store/>`_.
+* `Android App Bundle <https://developer.android.com/guide/app-bundle/faq>`_
+ (AAB) files which can be shared on `Google Play Store <https://play.google.com/store/>`_.
+* `Android Archive <https://developer.android.com/studio/projects/android-library>`_
+ (AAR) files which can be used as a re-usable bundle of resources for other
+ projects.
+
+It supports multiple CPU architectures.
+
+It supports apps developed with `Kivy framework <http://kivy.org>`_, but was
+built to be flexible about the backend libraries (through "bootstraps"), and
+also supports `PySDL2 <https://pypi.org/project/PySDL2/>`_, and a
+`WebView <https://developer.android.com/reference/android/webkit/WebView>`_ with
+a Python web server.
+
+It automatically supports dependencies on most pure Python packages. For other
+packages, including those that depend on C code, a special "recipe" must be
+written to support cross-compiling. python-for-android comes with recipes for
+many of the mosty popular libraries (e.g. numpy and sqlalchemy) built in.
+
+python-for-android works by cross-compiling the Python interpreter and its
+dependencies for Android devices, and bundling it with the app's python code
+and dependencies. The Python code is then interpreted on the Android device.
+
+It is recommended that python-for-android be used via
+`Buildozer <https://buildozer.readthedocs.io/>`_, which ensures the correct
+dependencies are pre-installed, and centralizes the configuration. However,
+python-for-android is not limited to being used with Buildozer.
+
+Buildozer is released and distributed under the terms of the MIT license. You
+should have received a
+copy of the MIT license alongside your distribution. Our
+`latest license <https://github.com/kivy/python-for-android/blob/master/LICENSE>`_
+is also available.
Contents
@@ -36,8 +61,10 @@ Contents
services
troubleshooting
docker
- contribute
testing_pull_requests
+ faq
+ contribute
+ contact
Indices and tables
diff --git a/doc/source/troubleshooting.rst b/doc/source/troubleshooting.rst
index d9f309db9e..25f8cbc04b 100644
--- a/doc/source/troubleshooting.rst
+++ b/doc/source/troubleshooting.rst
@@ -15,20 +15,6 @@ include this full log, e.g. via a `pastebin
<https://pastebin.ubuntu.com/>`_ or `Github gist
<https://gist.github.com/>`_.
-Getting help
-------------
-
-python-for-android is managed by the Kivy Organisation, and you can
-get help with any problems using the same channels as Kivy itself:
-
-- by email to the `kivy-users Google group
- <https://groups.google.com/forum/#!forum/kivy-users>`_
-- on `#support Discord channel <https://chat.kivy.org/>`_
-
-If you find a bug, you can also post an issue on the
-`python-for-android Github page
-<https://github.com/kivy/python-for-android>`_.
-
Debugging on Android
--------------------
@@ -114,73 +100,8 @@ For example, the Python installation for ``arm64-v8a`` is available in ``lib/arm
$ ls
modules site-packages stdlib.zip
+FAQ
+---
-
-Common errors
--------------
-
-The following are common problems and resolutions that users have reported.
-
-
-AttributeError: 'AnsiCodes' object has no attribute 'LIGHTBLUE_EX'
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-This occurs if your version of colorama is too low, install version
-0.3.3 or higher.
-
-If you install python-for-android with pip or via setup.py, this
-dependency should be taken care of automatically.
-
-AttributeError: 'Context' object has no attribute 'hostpython'
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-This is a known bug in some releases. To work around it, add your
-python requirement explicitly,
-e.g. :code:`--requirements=python3,kivy`. This also applies when using
-buildozer, in which case add python3 to your buildozer.spec requirements.
-
-linkname too long
-~~~~~~~~~~~~~~~~~
-
-This can happen when you try to include a very long filename, which
-doesn't normally happen but can occur accidentally if the p4a
-directory contains a .buildozer directory that is not excluded from
-the build (e.g. if buildozer was previously used). Removing this
-directory should fix the problem, and is desirable anyway since you
-don't want it in the APK.
-
-Requested API target 19 is not available, install it with the SDK android tool
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-This means that your SDK is missing the required platform tools. You
-need to install the ``platforms;android-19`` package in your SDK,
-using the ``android`` or ``sdkmanager`` tools (depending on SDK
-version).
-
-If using buildozer this should be done automatically, but as a
-workaround you can run these from
-``~/.buildozer/android/platform/android-sdk-20/tools/android``.
-
-ModuleNotFoundError: No module named '_ctypes'
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-You do not have the libffi headers available to python-for-android, so you need to install them. On Ubuntu and derivatives these come from the `libffi-dev` package.
-
-After installing the headers, clean the build (`p4a clean builds`, or with buildozer delete the `.buildozer` directory within your app directory) and run python-for-android again.
-
-SSLError("Can't connect to HTTPS URL because the SSL module is not available.")
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-Your `hostpython3` was compiled without SSL support. You need to install the SSL development files before rebuilding the `hostpython3` recipe.
-Remember to always clean the build before rebuilding (`p4a clean builds`, or with buildozer `buildozer android clean`).
-
-On Ubuntu and derivatives::
-
- apt install libssl-dev
- p4a clean builds # or with: buildozer `buildozer android clean
-
-On macOS::
-
- brew install openssl
- sudo ln -sfn /usr/local/opt/openssl /usr/local/ssl
- p4a clean builds # or with: buildozer `buildozer android clean
+Check out the `online FAQ <https://github.com/kivy/python-for-android/blob/master/FAQ.md>`_ for common
+errors.
\ No newline at end of file
diff --git a/setup.py b/setup.py
index 6351cdcf8a..badce08e82 100644
--- a/setup.py
+++ b/setup.py
@@ -34,7 +34,8 @@
def recursively_include(results, directory, patterns):
for root, subfolders, files in walk(directory):
for fn in files:
- if not any(glob.fnmatch.fnmatch(fn, pattern) for pattern in patterns):
+ if not any(
+ glob.fnmatch.fnmatch(fn, pattern) for pattern in patterns):
continue
filename = join(root, fn)
directory = 'pythonforandroid'
@@ -47,9 +48,12 @@ def recursively_include(results, directory, patterns):
['*.patch', 'Setup*', '*.pyx', '*.py', '*.c', '*.h',
'*.mk', '*.jam', '*.diff', ])
recursively_include(package_data, 'pythonforandroid/bootstraps',
- ['*.properties', '*.xml', '*.java', '*.tmpl', '*.txt', '*.png',
- '*.mk', '*.c', '*.h', '*.py', '*.sh', '*.jpg', '*.aidl',
- '*.gradle', '.gitkeep', 'gradlew*', '*.jar', "*.patch", ])
+ [
+ '*.properties', '*.xml', '*.java', '*.tmpl', '*.txt',
+ '*.png', '*.mk', '*.c', '*.h', '*.py', '*.sh', '*.jpg',
+ '*.aidl', '*.gradle', '.gitkeep', 'gradlew*', '*.jar',
+ '*.patch',
+ ])
recursively_include(package_data, 'pythonforandroid/bootstraps',
['sdl-config', ])
recursively_include(package_data, 'pythonforandroid/bootstraps/webview',
@@ -59,8 +63,7 @@ def recursively_include(results, directory, patterns):
with open(join(dirname(__file__), 'README.md'),
encoding="utf-8",
- errors="replace",
- ) as fileh:
+ errors="replace", ) as fileh:
long_description = fileh.read()
init_filen = join(dirname(__file__), 'pythonforandroid', '__init__.py')
@@ -68,8 +71,7 @@ def recursively_include(results, directory, patterns):
try:
with open(init_filen,
encoding="utf-8",
- errors="replace"
- ) as fileh:
+ errors="replace") as fileh:
lines = fileh.readlines()
except IOError:
pass
@@ -82,15 +84,19 @@ def recursively_include(results, directory, patterns):
version = matches[0].strip("'").strip('"')
break
if version is None:
- raise Exception('Error: version could not be loaded from {}'.format(init_filen))
+ raise Exception(
+ 'Error: version could not be loaded from {}'.format(init_filen))
setup(name='python-for-android',
version=version,
- description='Android APK packager for Python scripts and apps',
+ description=(
+ 'A development tool that packages Python apps into '
+ 'binaries that can run on Android devices.'
+ ),
long_description=long_description,
long_description_content_type='text/markdown',
python_requires=">=3.7.0",
- author='The Kivy team',
+ author='Kivy Team and other contributors',
author_email='kivy-dev@googlegroups.com',
url='https://github.com/kivy/python-for-android',
license='MIT',
@@ -121,9 +127,16 @@ def recursively_include(results, directory, patterns):
'Programming Language :: Python :: 3.8',
'Programming Language :: Python :: 3.9',
'Programming Language :: Python :: 3.10',
+ 'Programming Language :: Python :: 3.11',
'Topic :: Software Development',
'Topic :: Utilities',
],
packages=packages,
package_data=package_data,
+ project_urls={
+ 'Documentation': "https://python-for-android.readthedocs.io",
+ 'Source': "https://github.com/kivy/python-for-android",
+ 'Bug Reports': "https://github.com/kivy/python-for-android/issues",
+ },
+
)
From d35c70ae149748a213abb5bbe499ef4b6ba64d88 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sun, 17 Dec 2023 09:37:38 +0100
Subject: [PATCH 015/158] `sourceCompatibility` 1.7 and `targetCompatibility`
1.7 are obsolete, use 1.8 by default (#2942)
---
.../bootstraps/common/build/templates/build.tmpl.gradle | 5 -----
1 file changed, 5 deletions(-)
diff --git a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
index b46c9030ff..750a435d99 100644
--- a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
+++ b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
@@ -82,13 +82,8 @@ android {
}
compileOptions {
- {% if args.enable_androidx %}
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
- {% else %}
- sourceCompatibility JavaVersion.VERSION_1_7
- targetCompatibility JavaVersion.VERSION_1_7
- {% endif %}
{%- for option in args.compile_options %}
{{option}}
{%- endfor %}
From 66ba3e516ad7682e25918a45fd81b85e1ac5a90e Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sun, 17 Dec 2023 15:11:55 +0100
Subject: [PATCH 016/158] Update sdl2 deps to reflect the same targeted in
kivy/kivy (#2927)
* Update sdl2 deps to reflect the same targeted in kivy/kivy
* Update md5 sum
* Create new patch and set new include dir for sdl_image
* Set pygame as broken recipe
* sdl_image 2.8.0 has been released
---
ci/constants.py | 2 +
.../build/src/patches/SDLActivity.java.patch | 56 ++++++++++---------
pythonforandroid/recipes/kivy/__init__.py | 3 +-
pythonforandroid/recipes/pygame/__init__.py | 7 ++-
pythonforandroid/recipes/sdl2/__init__.py | 6 +-
.../recipes/sdl2/sdl-orientation-pr-6984.diff | 27 ---------
.../recipes/sdl2_image/__init__.py | 7 ++-
.../recipes/sdl2_mixer/__init__.py | 2 +-
pythonforandroid/recipes/sdl2_ttf/__init__.py | 2 +-
9 files changed, 49 insertions(+), 63 deletions(-)
delete mode 100644 pythonforandroid/recipes/sdl2/sdl-orientation-pr-6984.diff
diff --git a/ci/constants.py b/ci/constants.py
index 52a996c90f..cc1d9ea70a 100644
--- a/ci/constants.py
+++ b/ci/constants.py
@@ -41,6 +41,8 @@ class TargetPython(Enum):
'libtorrent',
# pybind11 build fails on macos
'pybind11',
+ # pygame (likely need to be updated) is broken with newer SDL2 versions
+ 'pygame',
])
BROKEN_RECIPES = {
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLActivity.java.patch b/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLActivity.java.patch
index 71d2537e7c..434be4e8ba 100644
--- a/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLActivity.java.patch
+++ b/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLActivity.java.patch
@@ -1,18 +1,19 @@
--- a/src/main/java/org/libsdl/app/SDLActivity.java
+++ b/src/main/java/org/libsdl/app/SDLActivity.java
-@@ -222,6 +222,8 @@ public class SDLActivity extends Activity implements View.OnSystemUiVisibilityCh
+@@ -221,6 +221,8 @@
+
// This is what SDL runs in. It invokes SDL_main(), eventually
protected static Thread mSDLThread;
-
-+ public static int keyboardInputType = InputType.TYPE_CLASS_TEXT | InputType.TYPE_TEXT_VARIATION_VISIBLE_PASSWORD;
+
++ public static int keyboardInputType = InputType.TYPE_CLASS_TEXT | InputType.TYPE_TEXT_VARIATION_VISIBLE_PASSWORD;
+
protected static SDLGenericMotionListener_API12 getMotionListener() {
if (mMotionListener == null) {
- if (Build.VERSION.SDK_INT >= 26) {
-@@ -324,6 +326,15 @@ public class SDLActivity extends Activity implements View.OnSystemUiVisibilityCh
+@@ -323,6 +325,15 @@
+ Log.v(TAG, "Model: " + Build.MODEL);
Log.v(TAG, "onCreate()");
super.onCreate(savedInstanceState);
-
++
+ SDLActivity.initialize();
+ // So we can call stuff from static callbacks
+ mSingleton = this;
@@ -21,11 +22,10 @@
+ // We don't do this in onCreate because we unpack and load the app data on a thread
+ // and we can't run setup tasks until that thread completes.
+ protected void finishLoad() {
-+
+
try {
Thread.currentThread().setName("SDLActivity");
- } catch (Exception e) {
-@@ -835,7 +846,7 @@ public class SDLActivity extends Activity implements View.OnSystemUiVisibilityCh
+@@ -837,7 +848,7 @@
Handler commandHandler = new SDLCommandHandler();
// Send a message from the SDLMain thread
@@ -34,28 +34,30 @@
Message msg = commandHandler.obtainMessage();
msg.arg1 = command;
msg.obj = data;
-@@ -1384,6 +1395,20 @@ public class SDLActivity extends Activity implements View.OnSystemUiVisibilityCh
+@@ -1385,7 +1396,22 @@
+ return null;
+ }
return SDLActivity.mSurface.getNativeSurface();
++ }
++
++ /**
++ * Calls turnActive() on singleton to keep loading screen active
++ */
++ public static void triggerAppConfirmedActive() {
++ mSingleton.appConfirmedActive();
}
-
-+ /**
-+ * Calls turnActive() on singleton to keep loading screen active
-+ */
-+ public static void triggerAppConfirmedActive() {
-+ mSingleton.appConfirmedActive();
-+ }
++
++ /**
++ * Trick needed for loading screen, overridden by PythonActivity
++ * to keep loading screen active
++ */
++ public void appConfirmedActive() {
++ }
+
-+ /**
-+ * Trick needed for loading screen, overridden by PythonActivity
-+ * to keep loading screen active
-+ */
-+ public void appConfirmedActive() {
-+ }
-+
+
// Input
- /**
-@@ -1878,6 +1903,7 @@ class SDLMain implements Runnable {
+@@ -1881,6 +1907,7 @@
Log.v("SDL", "Running main function " + function + " from library " + library);
@@ -63,7 +65,7 @@
SDLActivity.nativeRunMain(library, function, arguments);
Log.v("SDL", "Finished main function");
-@@ -1935,8 +1961,7 @@ class DummyEdit extends View implements View.OnKeyListener {
+@@ -1938,8 +1965,7 @@
public InputConnection onCreateInputConnection(EditorInfo outAttrs) {
ic = new SDLInputConnection(this, true);
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index ebf7b29e82..a2dc8798ae 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -66,9 +66,10 @@ def get_recipe_env(self, arch):
env['USE_SDL2'] = '1'
env['KIVY_SPLIT_EXAMPLES'] = '1'
sdl2_mixer_recipe = self.get_recipe('sdl2_mixer', self.ctx)
+ sdl2_image_recipe = self.get_recipe('sdl2_image', self.ctx)
env['KIVY_SDL2_PATH'] = ':'.join([
join(self.ctx.bootstrap.build_dir, 'jni', 'SDL', 'include'),
- join(self.ctx.bootstrap.build_dir, 'jni', 'SDL2_image'),
+ *sdl2_image_recipe.get_include_dirs(arch),
*sdl2_mixer_recipe.get_include_dirs(arch),
join(self.ctx.bootstrap.build_dir, 'jni', 'SDL2_ttf'),
])
diff --git a/pythonforandroid/recipes/pygame/__init__.py b/pythonforandroid/recipes/pygame/__init__.py
index 99124deff7..b77240e1b9 100644
--- a/pythonforandroid/recipes/pygame/__init__.py
+++ b/pythonforandroid/recipes/pygame/__init__.py
@@ -42,13 +42,18 @@ def prebuild_arch(self, arch):
for include_dir in sdl2_mixer_recipe.get_include_dirs(arch):
sdl_mixer_includes += f"-I{include_dir} "
+ sdl2_image_includes = ""
+ sdl2_image_recipe = self.get_recipe('sdl2_image', self.ctx)
+ for include_dir in sdl2_image_recipe.get_include_dirs(arch):
+ sdl2_image_includes += f"-I{include_dir} "
+
setup_file = setup_template.format(
sdl_includes=(
" -I" + join(self.ctx.bootstrap.build_dir, 'jni', 'SDL', 'include') +
" -L" + join(self.ctx.bootstrap.build_dir, "libs", str(arch)) +
" -L" + png_lib_dir + " -L" + jpeg_lib_dir + " -L" + arch.ndk_lib_dir_versioned),
sdl_ttf_includes="-I"+join(self.ctx.bootstrap.build_dir, 'jni', 'SDL2_ttf'),
- sdl_image_includes="-I"+join(self.ctx.bootstrap.build_dir, 'jni', 'SDL2_image'),
+ sdl_image_includes=sdl2_image_includes,
sdl_mixer_includes=sdl_mixer_includes,
jpeg_includes="-I"+jpeg_inc_dir,
png_includes="-I"+png_inc_dir,
diff --git a/pythonforandroid/recipes/sdl2/__init__.py b/pythonforandroid/recipes/sdl2/__init__.py
index e04458a793..8d5fbc2dc2 100644
--- a/pythonforandroid/recipes/sdl2/__init__.py
+++ b/pythonforandroid/recipes/sdl2/__init__.py
@@ -6,16 +6,14 @@
class LibSDL2Recipe(BootstrapNDKRecipe):
- version = "2.26.1"
+ version = "2.28.5"
url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL2-{version}.tar.gz"
- md5sum = 'fba211fe2c67609df6fa3cf55d3c74dc'
+ md5sum = 'a344eb827a03045c9b399e99af4af13d'
dir_name = 'SDL'
depends = ['sdl2_image', 'sdl2_mixer', 'sdl2_ttf']
- patches = ['sdl-orientation-pr-6984.diff']
-
def get_recipe_env(self, arch=None, with_flags_in_cc=True, with_python=True):
env = super().get_recipe_env(
arch=arch, with_flags_in_cc=with_flags_in_cc, with_python=with_python)
diff --git a/pythonforandroid/recipes/sdl2/sdl-orientation-pr-6984.diff b/pythonforandroid/recipes/sdl2/sdl-orientation-pr-6984.diff
deleted file mode 100644
index bbe2ca2243..0000000000
--- a/pythonforandroid/recipes/sdl2/sdl-orientation-pr-6984.diff
+++ /dev/null
@@ -1,27 +0,0 @@
-diff --git a/android-project/app/src/main/java/org/libsdl/app/SDLActivity.java b/android-project/app/src/main/java/org/libsdl/app/SDLActivity.java
-index 2d7d69b76a25..edb42fb55461 100644
---- a/android-project/app/src/main/java/org/libsdl/app/SDLActivity.java
-+++ b/android-project/app/src/main/java/org/libsdl/app/SDLActivity.java
-@@ -971,15 +971,18 @@ public void setOrientationBis(int w, int h, boolean resizable, String hint)
- /* If set, hint "explicitly controls which UI orientations are allowed". */
- if (hint.contains("LandscapeRight") && hint.contains("LandscapeLeft")) {
- orientation_landscape = ActivityInfo.SCREEN_ORIENTATION_SENSOR_LANDSCAPE;
-- } else if (hint.contains("LandscapeRight")) {
-- orientation_landscape = ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
- } else if (hint.contains("LandscapeLeft")) {
-+ orientation_landscape = ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
-+ } else if (hint.contains("LandscapeRight")) {
- orientation_landscape = ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE;
- }
-
-- if (hint.contains("Portrait") && hint.contains("PortraitUpsideDown")) {
-+ /* exact match to 'Portrait' to distinguish with PortraitUpsideDown */
-+ boolean contains_Portrait = hint.contains("Portrait ") || hint.endsWith("Portrait");
-+
-+ if (contains_Portrait && hint.contains("PortraitUpsideDown")) {
- orientation_portrait = ActivityInfo.SCREEN_ORIENTATION_SENSOR_PORTRAIT;
-- } else if (hint.contains("Portrait")) {
-+ } else if (contains_Portrait) {
- orientation_portrait = ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
- } else if (hint.contains("PortraitUpsideDown")) {
- orientation_portrait = ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT;
diff --git a/pythonforandroid/recipes/sdl2_image/__init__.py b/pythonforandroid/recipes/sdl2_image/__init__.py
index a91c6f1bc2..b3ac504fbf 100644
--- a/pythonforandroid/recipes/sdl2_image/__init__.py
+++ b/pythonforandroid/recipes/sdl2_image/__init__.py
@@ -6,12 +6,17 @@
class LibSDL2Image(BootstrapNDKRecipe):
- version = '2.6.2'
+ version = '2.8.0'
url = 'https://github.com/libsdl-org/SDL_image/releases/download/release-{version}/SDL2_image-{version}.tar.gz'
dir_name = 'SDL2_image'
patches = ['enable-webp.patch']
+ def get_include_dirs(self, arch):
+ return [
+ os.path.join(self.ctx.bootstrap.build_dir, "jni", "SDL2_image", "include")
+ ]
+
def prebuild_arch(self, arch):
# We do not have a folder for each arch on BootstrapNDKRecipe, so we
# need to skip the external deps download if we already have done it.
diff --git a/pythonforandroid/recipes/sdl2_mixer/__init__.py b/pythonforandroid/recipes/sdl2_mixer/__init__.py
index 0f02c4c3a4..a00c267d49 100644
--- a/pythonforandroid/recipes/sdl2_mixer/__init__.py
+++ b/pythonforandroid/recipes/sdl2_mixer/__init__.py
@@ -4,7 +4,7 @@
class LibSDL2Mixer(BootstrapNDKRecipe):
- version = '2.6.2'
+ version = '2.6.3'
url = 'https://github.com/libsdl-org/SDL_mixer/releases/download/release-{version}/SDL2_mixer-{version}.tar.gz'
dir_name = 'SDL2_mixer'
diff --git a/pythonforandroid/recipes/sdl2_ttf/__init__.py b/pythonforandroid/recipes/sdl2_ttf/__init__.py
index 4934bd4a67..9f97ae441c 100644
--- a/pythonforandroid/recipes/sdl2_ttf/__init__.py
+++ b/pythonforandroid/recipes/sdl2_ttf/__init__.py
@@ -2,7 +2,7 @@
class LibSDL2TTF(BootstrapNDKRecipe):
- version = '2.20.1'
+ version = '2.20.2'
url = 'https://github.com/libsdl-org/SDL_ttf/releases/download/release-{version}/SDL2_ttf-{version}.tar.gz'
dir_name = 'SDL2_ttf'
From 75be0186e437f6d198f3d51c46361ce46edc1f2a Mon Sep 17 00:00:00 2001
From: shyamnathp <shyamnath94.mec@gmail.com>
Date: Thu, 11 Jan 2024 18:53:31 +0100
Subject: [PATCH 017/158] Initial support for PySide6 and Qt bootstrap (#2918)
* Initial support for PySide6
- Add a new bootstrap for Qt
- This bootstrap will be used by `pyside6-android-deploy` tool shipped
with PySide6, which interally calls pythonforandroid using buildozer.
- The Qt bootstrap depends on recipes PySide6 and shiboken6 among
other mandatory recipes. The recipes for PySide6 and shiboken6
resides in the PySide repository -
https://code.qt.io/cgit/pyside/pyside-setup.git/tree/sources/pyside-tools/deploy_lib/android/recipes
- The PythonActivity entrypoint class is derived from QtActivity class
which is the main acitivty class when a Qt C++ application is
packaged for Android. The jar containing QtActivity class is supplied
through buildozer `android.add_jars` option.
- The C wrapper binary to the application main.py is named as
`main_{abi_name}` instead of just `main` for other bootstraps.
- Multi architecture deployment is not supported at the moment.
- Adapt tests based on the new Qt bootstrap
* Add Qt boostrap to docs
- update the docs to include sections depicting the Qt boostrap.
* Tweak gradle build properties for Qt bootstrap
- Sometimes a flaky Java heap out of memory error is throw. By,
tweaking the memory setting we can get rid of that error.
* Fix bug - check for main.py
- Qt boostrap removed from comparison in the changed line because
its expects a value is args.launcher, which is not applicable for
Qt boostrap. Hence, it exits with an value not found exception.
- Removing Qt boostrap from the comparison leads to checking
for main.py or --private, which is to be done for the Qt boostrap.
* Make --init-classes truly optional
- check if empty, otherwise store empty string
* Add a non-gui test app build to CI that uses Qt bootstrap
- for the purpose of testing, the pyside6 and shiboken6 wheels, the
extra .jar files needed and the recipes for pyside6 and shiboken6
are manually added into testapps/on_device_unit_tests/test_qt.
These files are normally generated by the `pyside6-android-deploy`
tool that is shipped with PySide.
Generating the wheels and the .jar files belongs to the scope of
PySide and not python-for-android. Hence, they are not done here.
This also reduces the load on the CI which will otherwise have to
cross-compile CPython and PySide.
- The Android aarch64 wheels for testing are downloaded from Qt servers.
These wheels are for testing purposes only and the download link will
be updated when official PySide6 Android wheels are generated.
- Tests were added in test_requirements.py so that when running the apk
the current date and time are printed on the terminal. The tests also
checks shiboken6 and PySide6 module imports.
* Remove superfluous whitespace removal
- This was introduced by a VSCode setting and is unrealated to this
patch. Although this might be good, this has to be introduced through
a different patch.
- pyside6 recipe typo adapted in buildoptions.rst
* Add aab generation to test app with Qt bootstrap
* Fix typo in doc/source/buildoptions.rst
Co-authored-by: Mirko Galimberti <me@mirkogalimberti.com>
---------
Co-authored-by: Shyamnath Premnadh <Shyamnath.Premnadh@qt.io>
Co-authored-by: Mirko Galimberti <me@mirkogalimberti.com>
---
.github/workflows/push.yml | 2 +
Makefile | 20 ++
doc/source/buildoptions.rst | 43 +++
doc/source/commands.rst | 8 +-
doc/source/quickstart.rst | 7 +-
.../bootstraps/common/build/build.py | 34 ++-
.../build/templates/gradle.tmpl.properties | 5 +
pythonforandroid/bootstraps/qt/__init__.py | 53 ++++
.../bootstraps/qt/build/.gitignore | 14 +
.../bootstraps/qt/build/blacklist.txt | 70 +++++
.../bootstraps/qt/build/jni/Application.mk | 8 +
.../qt/build/jni/application/src/Android.mk | 18 ++
.../jni/application/src/Android_static.mk | 9 +
.../jni/application/src/bootstrap_name.h | 4 +
.../qt/build/src/main/assets/.gitkeep | 0
.../qt/build/src/main/java/.gitkeep | 0
.../java/org/kivy/android/PythonActivity.java | 245 ++++++++++++++++++
.../qt/build/src/main/jniLibs/.gitkeep | 0
.../qt/build/src/main/libs/.gitkeep | 0
.../qt/build/src/main/res/drawable/.gitkeep | 0
.../qt/build/src/main/res/mipmap/.gitkeep | 0
.../build/templates/AndroidManifest.tmpl.xml | 106 ++++++++
.../qt/build/templates/libs.tmpl.xml | 27 ++
.../qt/build/templates/strings.tmpl.xml | 6 +
pythonforandroid/recipe.py | 5 +-
pythonforandroid/recipes/android/__init__.py | 2 +-
.../recipes/genericndkbuild/__init__.py | 2 +
.../on_device_unit_tests/test_app/main.py | 1 +
.../test_app/tests/test_requirements.py | 22 ++
.../test_qt/jar/PySide6/jar/Qt6Android.jar | Bin 0 -> 134543 bytes
.../jar/PySide6/jar/Qt6AndroidBindings.jar | Bin 0 -> 23527 bytes
.../test_qt/recipes/PySide6/__init__.py | 63 +++++
.../test_qt/recipes/shiboken6/__init__.py | 39 +++
tests/test_bootstrap.py | 20 +-
34 files changed, 819 insertions(+), 14 deletions(-)
create mode 100644 pythonforandroid/bootstraps/qt/__init__.py
create mode 100644 pythonforandroid/bootstraps/qt/build/.gitignore
create mode 100644 pythonforandroid/bootstraps/qt/build/blacklist.txt
create mode 100644 pythonforandroid/bootstraps/qt/build/jni/Application.mk
create mode 100644 pythonforandroid/bootstraps/qt/build/jni/application/src/Android.mk
create mode 100644 pythonforandroid/bootstraps/qt/build/jni/application/src/Android_static.mk
create mode 100644 pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/assets/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/java/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/jniLibs/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/libs/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/res/drawable/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/src/main/res/mipmap/.gitkeep
create mode 100644 pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
create mode 100644 pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
create mode 100644 pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
create mode 100644 testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6Android.jar
create mode 100644 testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6AndroidBindings.jar
create mode 100644 testapps/on_device_unit_tests/test_qt/recipes/PySide6/__init__.py
create mode 100644 testapps/on_device_unit_tests/test_qt/recipes/shiboken6/__init__.py
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index d7f789aaaf..4eff5d0f61 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -70,6 +70,8 @@ jobs:
target: testapps-webview
- name: service_library
target: testapps-service_library-aar
+ - name: qt
+ target: testapps-qt
steps:
- name: Checkout python-for-android
uses: actions/checkout@v4
diff --git a/Makefile b/Makefile
index 9747c6c43d..b894bed3ea 100644
--- a/Makefile
+++ b/Makefile
@@ -82,6 +82,26 @@ testapps-service_library-aar: virtualenv
--requirements python3 \
--arch=arm64-v8a --arch=x86 --release
+testapps-qt: testapps-qt/debug/apk testapps-qt/release/aab
+
+# testapps-webview/MODE/ARTIFACT
+testapps-qt/%: virtualenv
+ $(eval MODE := $(word 2, $(subst /, ,$@)))
+ $(eval ARTIFACT := $(word 3, $(subst /, ,$@)))
+ @echo Building testapps-qt for $(MODE) mode and $(ARTIFACT) artifact
+ . $(ACTIVATE) && cd testapps/on_device_unit_tests/ && \
+ python setup.py $(ARTIFACT) --$(MODE) --sdk-dir $(ANDROID_SDK_HOME) --ndk-dir $(ANDROID_NDK_HOME) \
+ --bootstrap qt \
+ --requirements python3,shiboken6,pyside6 \
+ --arch=arm64-v8a \
+ --local-recipes ./test_qt/recipes \
+ --qt-libs Core \
+ --load-local-libs plugins_platforms_qtforandroid \
+ --add-jar ./test_qt/jar/PySide6/jar/Qt6Android.jar \
+ --add-jar ./test_qt/jar/PySide6/jar/Qt6AndroidBindings.jar \
+ --permission android.permission.WRITE_EXTERNAL_STORAGE \
+ --permission android.permission.INTERNET
+
testapps/%: virtualenv
$(eval $@_APP_ARCH := $(shell basename $*))
. $(ACTIVATE) && cd testapps/on_device_unit_tests/ && \
diff --git a/doc/source/buildoptions.rst b/doc/source/buildoptions.rst
index 44bae679c1..782bd67295 100644
--- a/doc/source/buildoptions.rst
+++ b/doc/source/buildoptions.rst
@@ -204,6 +204,49 @@ systems and frameworks.
include multiple jar files, pass this argument multiple times.
- ``add-source``: Add a source directory to the app's Java code.
+Qt
+~~
+
+This bootstrap can be used with ``--bootstrap=qt`` or by including the ``PySide6`` or
+``shiboken6`` recipe, e.g. ``--requirements=pyside6,shiboken6``. Currently, the only way
+to use this bootstrap is through `pyside6-android-deploy <https://www.qt.io/blog/taking-qt-for-python-to-android>`__
+tool shipped with ``PySide6``, as the recipes for ``PySide6`` and ``shiboken6`` are created
+dynamically. The tool builds ``PySide6`` and ``shiboken6`` wheels for a specific Android platform
+and the recipes simply unpack the built wheels. You can see the recipes `here <https://code.qt.io/cgit/pyside/pyside-setup.git/tree/sources/pyside-tools/deploy_lib/android/recipes>`__.
+
+.. note::
+ The ``pyside6-android-deploy`` tool and hence the Qt bootstrap does not support multi-architecture
+ builds currently.
+
+What are Qt and PySide?
+%%%%%%%%%%%%%%%%%%%%%%%%
+
+`Qt <https://www.qt.io/>`__ is a popularly used cross-platform C++ framework for developing
+GUI applications. `PySide6 <https://doc.qt.io/qtforpython-6/quickstart.html>`__ refers to the
+Python bindings for Qt6, and enables the Python developers access to the Qt6 API.
+`Shiboken6 <https://doc.qt.io/qtforpython-6/shiboken6/index.html>`__ is the binding generator
+tool used for generating the Python bindings from C++ code.
+
+.. note:: The `shiboken6` recipe is for the `Shiboken Python module <https://doc.qt.io/qtforpython-6/shiboken6/shibokenmodule.html>`__
+ which includes a couple of utility functions for inspecting and debugging PySide6 code.
+
+Build Options
+%%%%%%%%%%%%%
+
+``pyside6-android-deploy`` works by generating a ``buildozer.spec`` file and thereby using
+`buildozer <https://buildozer.readthedocs.io/en/latest/>`__ to control the build options used by
+``python-for-android`` with the Qt bootstrap. Apart from the general build options that works
+across all the other bootstraps, the Qt bootstrap introduces the following 3 new build options.
+
+- ``--qt-libs``: list of Qt libraries(modules) to be loaded.
+- ``--load-local-libs``: list of Qt plugin libraries to be loaded.
+- ``--init-classes``: list of Java class names to the loaded from the Qt jar files supplied through
+ the ``--add-jar`` option.
+
+These build options are automatically populated by the ``pyside6-android-deploy`` tool, but can be
+modified by updating the ``buildozer.spec`` file. Apart from the above 3 build options, the tool
+also automatically identifies the values to be fed into the cli options ``--permission``, ``--add-jar``
+depending on the PySide6 modules used by the applicaiton.
Requirements blacklist (APK size optimization)
----------------------------------------------
diff --git a/doc/source/commands.rst b/doc/source/commands.rst
index 5a0884aa5e..dda644e47a 100644
--- a/doc/source/commands.rst
+++ b/doc/source/commands.rst
@@ -26,7 +26,7 @@ behaviour, though not all commands make use of them.
``--debug``
Print extra debug information about the build, including all compilation output.
-
+
``--sdk_dir``
The filepath where the Android SDK is installed. This can
alternatively be set in several other ways.
@@ -34,7 +34,7 @@ behaviour, though not all commands make use of them.
``--android_api``
The Android API level to target; python-for-android will check if
the platform tools for this level are installed.
-
+
``--ndk_dir``
The filepath where the Android NDK is installed. This can
alternatively be set in several other ways.
@@ -74,12 +74,12 @@ supply those that you need.
The architecture to build for. You can specify multiple architectures to build for
at the same time. As an example ``p4a ... --arch arm64-v8a --arch armeabi-v7a ...``
will build a distribution for both ``arm64-v8a`` and ``armeabi-v7a``.
-
+
``--bootstrap BOOTSTRAP``
The Java bootstrap to use for your application. You mostly don't
need to worry about this or set it manually, as an appropriate
bootstrap will be chosen from your ``--requirements``. Current
- choices are ``sdl2`` (used with Kivy and most other apps) or ``webview``.
+ choices are ``sdl2`` (used with Kivy and most other apps), ``webview`` or ``qt``.
.. note:: These options are preliminary. Others will include toggles
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 95c4eefc4e..987a00cddb 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -26,8 +26,8 @@ Concepts
- **bootstrap:** A bootstrap is the app backend that will start your
application. The default for graphical applications is SDL2.
- You can also use e.g. the webview for web apps, or service_only/service_library for
- background services. Different bootstraps have different additional
+ You can also use e.g. the webview for web apps, or service_only/service_library for
+ background services, or qt for PySide6 apps. Different bootstraps have different additional
build options.
*Advanced:*
@@ -281,7 +281,7 @@ Recipe management
You can see the list of the available recipes with::
p4a recipes
-
+
If you are contributing to p4a and want to test a recipes again,
you need to clean the build and rebuild your distribution::
@@ -295,7 +295,6 @@ it (edit the ``__init__.py``)::
mkdir -p p4a-recipes/myrecipe
touch p4a-recipes/myrecipe/__init__.py
-
Distribution management
~~~~~~~~~~~~~~~~~~~~~~~
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index eced5886c8..29d16ea9f2 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -83,7 +83,7 @@ def get_bootstrap_name():
if PYTHON is not None and not exists(PYTHON):
PYTHON = None
-if _bootstrap_name in ('sdl2', 'webview', 'service_only'):
+if _bootstrap_name in ('sdl2', 'webview', 'service_only', 'qt'):
WHITELIST_PATTERNS.append('pyconfig.h')
environment = jinja2.Environment(loader=jinja2.FileSystemLoader(
@@ -543,6 +543,7 @@ def make_package(args):
}
if get_bootstrap_name() == "sdl2":
render_args["url_scheme"] = url_scheme
+
render(
'AndroidManifest.tmpl.xml',
manifest_path,
@@ -571,7 +572,8 @@ def make_package(args):
render(
'gradle.tmpl.properties',
'gradle.properties',
- args=args)
+ args=args,
+ bootstrap_name=get_bootstrap_name())
# ant build templates
render(
@@ -601,6 +603,26 @@ def make_package(args):
join(res_dir, 'values/strings.xml'),
**render_args)
+ # Library resources from Qt
+ # These are referred by QtLoader.java in Qt6AndroidBindings.jar
+ # qt_libs and load_local_libs are loaded at App startup
+ if get_bootstrap_name() == "qt":
+ qt_libs = args.qt_libs.split(",")
+ load_local_libs = args.load_local_libs.split(",")
+ init_classes = args.init_classes
+ if init_classes:
+ init_classes = init_classes.split(",")
+ init_classes = ":".join(init_classes)
+ arch = get_dist_info_for("archs")[0]
+ render(
+ 'libs.tmpl.xml',
+ join(res_dir, 'values/libs.xml'),
+ qt_libs=qt_libs,
+ load_local_libs=load_local_libs,
+ init_classes=init_classes,
+ arch=arch
+ )
+
if exists(join("templates", "custom_rules.tmpl.xml")):
render(
'custom_rules.tmpl.xml',
@@ -951,6 +973,14 @@ def create_argument_parser():
help='Use that parameter if you need to implement your own PythonServive Java class')
ap.add_argument('--activity-class-name', dest='activity_class_name', default=DEFAULT_PYTHON_ACTIVITY_JAVA_CLASS,
help='The full java class name of the main activity')
+ if get_bootstrap_name() == "qt":
+ ap.add_argument('--qt-libs', dest='qt_libs', required=True,
+ help='comma separated list of Qt libraries to be loaded')
+ ap.add_argument('--load-local-libs', dest='load_local_libs', required=True,
+ help='comma separated list of Qt plugin libraries to be loaded')
+ ap.add_argument('--init-classes', dest='init_classes', default='',
+ help='comma separated list of java class names to be loaded from the Qt jar files, '
+ 'specified through add_jar cli option')
return ap
diff --git a/pythonforandroid/bootstraps/common/build/templates/gradle.tmpl.properties b/pythonforandroid/bootstraps/common/build/templates/gradle.tmpl.properties
index f99dd5a052..cea16375d2 100644
--- a/pythonforandroid/bootstraps/common/build/templates/gradle.tmpl.properties
+++ b/pythonforandroid/bootstraps/common/build/templates/gradle.tmpl.properties
@@ -1,3 +1,8 @@
+{% if bootstrap_name == "qt" %}
+# For tweaking memory settings. Otherwise, a p4a session with Qt bootstrap and PySide6 recipe
+# terminates with a Java out of memory exception
+org.gradle.jvmargs=-Xmx2500m -XX:MaxMetaspaceSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
+{% endif %}
{% if args.enable_androidx %}
android.useAndroidX=true
android.enableJetifier=true
diff --git a/pythonforandroid/bootstraps/qt/__init__.py b/pythonforandroid/bootstraps/qt/__init__.py
new file mode 100644
index 0000000000..9a6e03f064
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/__init__.py
@@ -0,0 +1,53 @@
+import sh
+from os.path import join
+from pythonforandroid.toolchain import (
+ Bootstrap, current_directory, info, info_main, shprint)
+from pythonforandroid.util import ensure_dir, rmdir
+
+
+class QtBootstrap(Bootstrap):
+ name = 'qt'
+ recipe_depends = ['python3', 'genericndkbuild', 'PySide6', 'shiboken6']
+ # this is needed because the recipes PySide6 and shiboken6 resides in the PySide Qt repository
+ # - https://code.qt.io/cgit/pyside/pyside-setup.git/
+ # Without this some tests will error because it cannot find the recipes within pythonforandroid
+ # repository
+ can_be_chosen_automatically = False
+
+ def assemble_distribution(self):
+ info_main("# Creating Android project using Qt bootstrap")
+
+ rmdir(self.dist_dir)
+ info("Copying gradle build")
+ shprint(sh.cp, '-r', self.build_dir, self.dist_dir)
+
+ with current_directory(self.dist_dir):
+ with open('local.properties', 'w') as fileh:
+ fileh.write('sdk.dir={}'.format(self.ctx.sdk_dir))
+
+ arch = self.ctx.archs[0]
+ if len(self.ctx.archs) > 1:
+ raise ValueError("Trying to build for more than one arch. Qt bootstrap cannot handle that yet")
+
+ info(f"Bootstrap running with arch {arch}")
+
+ with current_directory(self.dist_dir):
+ info("Copying Python distribution")
+
+ self.distribute_libs(arch, [self.ctx.get_libs_dir(arch.arch)])
+ self.distribute_aars(arch)
+ self.distribute_javaclasses(self.ctx.javaclass_dir,
+ dest_dir=join("src", "main", "java"))
+
+ python_bundle_dir = join(f'_python_bundle__{arch.arch}', '_python_bundle')
+ ensure_dir(python_bundle_dir)
+ site_packages_dir = self.ctx.python_recipe.create_python_bundle(
+ join(self.dist_dir, python_bundle_dir), arch)
+
+ if not self.ctx.with_debug_symbols:
+ self.strip_libraries(arch)
+ self.fry_eggs(site_packages_dir)
+ super().assemble_distribution()
+
+
+bootstrap = QtBootstrap()
diff --git a/pythonforandroid/bootstraps/qt/build/.gitignore b/pythonforandroid/bootstraps/qt/build/.gitignore
new file mode 100644
index 0000000000..a1fc39c070
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/.gitignore
@@ -0,0 +1,14 @@
+.gradle
+/build/
+
+# Ignore Gradle GUI config
+gradle-app.setting
+
+# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
+!gradle-wrapper.jar
+
+# Cache of project
+.gradletasknamecache
+
+# # Work around https://youtrack.jetbrains.com/issue/IDEA-116898
+# gradle/wrapper/gradle-wrapper.properties
diff --git a/pythonforandroid/bootstraps/qt/build/blacklist.txt b/pythonforandroid/bootstraps/qt/build/blacklist.txt
new file mode 100644
index 0000000000..65f6e4df2e
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/blacklist.txt
@@ -0,0 +1,70 @@
+# prevent user to include invalid extensions
+*.apk
+*.aab
+*.apks
+*.pxd
+
+# eggs
+*.egg-info
+
+# unit test
+unittest/*
+
+# python config
+config/makesetup
+
+# unused encodings
+lib-dynload/*codec*
+encodings/cp*.pyo
+encodings/tis*
+encodings/shift*
+encodings/bz2*
+encodings/iso*
+encodings/undefined*
+encodings/johab*
+encodings/p*
+encodings/m*
+encodings/euc*
+encodings/k*
+encodings/unicode_internal*
+encodings/quo*
+encodings/gb*
+encodings/big5*
+encodings/hp*
+encodings/hz*
+
+# unused python modules
+bsddb/*
+wsgiref/*
+hotshot/*
+pydoc_data/*
+tty.pyo
+anydbm.pyo
+nturl2path.pyo
+LICENCE.txt
+macurl2path.pyo
+dummy_threading.pyo
+audiodev.pyo
+antigravity.pyo
+dumbdbm.pyo
+sndhdr.pyo
+__phello__.foo.pyo
+sunaudio.pyo
+os2emxpath.pyo
+multiprocessing/dummy*
+
+# unused binaries python modules
+lib-dynload/termios.so
+lib-dynload/_lsprof.so
+lib-dynload/*audioop.so
+lib-dynload/_hotshot.so
+lib-dynload/_heapq.so
+lib-dynload/_json.so
+lib-dynload/grp.so
+lib-dynload/resource.so
+lib-dynload/pyexpat.so
+lib-dynload/_ctypes_test.so
+lib-dynload/_testcapi.so
+
+# odd files
+plat-linux3/regen
diff --git a/pythonforandroid/bootstraps/qt/build/jni/Application.mk b/pythonforandroid/bootstraps/qt/build/jni/Application.mk
new file mode 100644
index 0000000000..e3d23e5be1
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/jni/Application.mk
@@ -0,0 +1,8 @@
+
+# Uncomment this if you're using STL in your project
+# See CPLUSPLUS-SUPPORT.html in the NDK documentation for more information
+# APP_STL := stlport_static
+
+# APP_ABI := armeabi armeabi-v7a x86
+APP_ABI := $(ARCH)
+APP_PLATFORM := $(NDK_API)
diff --git a/pythonforandroid/bootstraps/qt/build/jni/application/src/Android.mk b/pythonforandroid/bootstraps/qt/build/jni/application/src/Android.mk
new file mode 100644
index 0000000000..aebe3f623b
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/jni/application/src/Android.mk
@@ -0,0 +1,18 @@
+LOCAL_PATH := $(call my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_MODULE := main_$(PREFERRED_ABI)
+
+# Add your application source files here...
+LOCAL_SRC_FILES := start.c
+
+LOCAL_CFLAGS += -I$(PYTHON_INCLUDE_ROOT) $(EXTRA_CFLAGS)
+
+LOCAL_SHARED_LIBRARIES := python_shared
+
+LOCAL_LDLIBS := -llog $(EXTRA_LDLIBS)
+
+LOCAL_LDFLAGS += -L$(PYTHON_LINK_ROOT) $(APPLICATION_ADDITIONAL_LDFLAGS)
+
+include $(BUILD_SHARED_LIBRARY)
diff --git a/pythonforandroid/bootstraps/qt/build/jni/application/src/Android_static.mk b/pythonforandroid/bootstraps/qt/build/jni/application/src/Android_static.mk
new file mode 100644
index 0000000000..1bb58cb76d
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/jni/application/src/Android_static.mk
@@ -0,0 +1,9 @@
+LOCAL_PATH := $(call my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_MODULE := main_$(PREFERRED_ABI)
+
+LOCAL_SRC_FILES := start.c
+
+include $(BUILD_SHARED_LIBRARY)
diff --git a/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
new file mode 100644
index 0000000000..8a4d8aa464
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
@@ -0,0 +1,4 @@
+
+#define BOOTSTRAP_USES_NO_SDL_HEADERS
+
+const char bootstrap_name[] = "qt";
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/assets/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/assets/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/java/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/java/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java b/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
new file mode 100644
index 0000000000..81cad01616
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
@@ -0,0 +1,245 @@
+package org.kivy.android;
+
+import android.os.SystemClock;
+
+import java.io.File;
+import java.util.Collections;
+import java.util.Iterator;
+import java.util.List;
+import java.util.ArrayList;
+
+import android.app.Activity;
+import android.app.AlertDialog;
+import android.content.DialogInterface;
+import android.content.Intent;
+import android.view.KeyEvent;
+import android.util.Log;
+import android.widget.Toast;
+import android.os.Bundle;
+import android.os.PowerManager;
+import android.content.Context;
+import android.content.pm.PackageManager;
+
+import org.qtproject.qt.android.bindings.QtActivity;
+import org.qtproject.qt.android.QtNative;
+
+public class PythonActivity extends QtActivity {
+
+ private static final String TAG = "PythonActivity";
+
+ public static PythonActivity mActivity = null;
+
+ private Bundle mMetaData = null;
+ private PowerManager.WakeLock mWakeLock = null;
+
+ public String getAppRoot() {
+ String app_root = getFilesDir().getAbsolutePath() + "/app";
+ return app_root;
+ }
+
+ public String getEntryPoint(String search_dir) {
+ /* Get the main file (.pyc|.py) depending on if we
+ * have a compiled version or not.
+ */
+ List<String> entryPoints = new ArrayList<String>();
+ entryPoints.add("main.pyc"); // python 3 compiled files
+ for (String value : entryPoints) {
+ File mainFile = new File(search_dir + "/" + value);
+ if (mainFile.exists()) {
+ return value;
+ }
+ }
+ return "main.py";
+ }
+
+ @Override
+ public void onCreate(Bundle savedInstanceState) {
+ this.mActivity = this;
+ Log.v(TAG, "Ready to unpack");
+ File app_root_file = new File(getAppRoot());
+ PythonUtil.unpackAsset(mActivity, "private", app_root_file, true);
+ PythonUtil.unpackPyBundle(mActivity, getApplicationInfo().nativeLibraryDir + "/" + "libpybundle", app_root_file, false);
+
+ Log.v("Python", "Device: " + android.os.Build.DEVICE);
+ Log.v("Python", "Model: " + android.os.Build.MODEL);
+
+ // Set up the Python environment
+ String app_root_dir = getAppRoot();
+ String mFilesDirectory = mActivity.getFilesDir().getAbsolutePath();
+ String entry_point = getEntryPoint(app_root_dir);
+
+ Log.v(TAG, "Setting env vars for start.c and Python to use");
+ QtNative.setEnvironmentVariable("ANDROID_ENTRYPOINT", entry_point);
+ QtNative.setEnvironmentVariable("ANDROID_ARGUMENT", app_root_dir);
+ QtNative.setEnvironmentVariable("ANDROID_APP_PATH", app_root_dir);
+ QtNative.setEnvironmentVariable("ANDROID_PRIVATE", mFilesDirectory);
+ QtNative.setEnvironmentVariable("ANDROID_UNPACK", app_root_dir);
+ QtNative.setEnvironmentVariable("PYTHONHOME", app_root_dir);
+ QtNative.setEnvironmentVariable("PYTHONPATH", app_root_dir + ":" + app_root_dir + "/lib");
+ QtNative.setEnvironmentVariable("PYTHONOPTIMIZE", "2");
+
+ Log.v(TAG, "About to do super onCreate");
+ super.onCreate(savedInstanceState);
+ Log.v(TAG, "Did super onCreate");
+
+ this.mActivity = this;
+ try {
+ Log.v(TAG, "Access to our meta-data...");
+ mActivity.mMetaData = mActivity.getPackageManager().getApplicationInfo(
+ mActivity.getPackageName(), PackageManager.GET_META_DATA).metaData;
+
+ PowerManager pm = (PowerManager) mActivity.getSystemService(Context.POWER_SERVICE);
+ if ( mActivity.mMetaData.getInt("wakelock") == 1 ) {
+ mActivity.mWakeLock = pm.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "Screen On");
+ mActivity.mWakeLock.acquire();
+ }
+ } catch (PackageManager.NameNotFoundException e) {
+ }
+ }
+
+ @Override
+ public void onDestroy() {
+ Log.i("Destroy", "end of app");
+ super.onDestroy();
+
+ // make sure all child threads (python_thread) are stopped
+ android.os.Process.killProcess(android.os.Process.myPid());
+ }
+
+ long lastBackClick = SystemClock.elapsedRealtime();
+ @Override
+ public boolean onKeyDown(int keyCode, KeyEvent event) {
+ // If it wasn't the Back key or there's no web page history, bubble up to the default
+ // system behavior (probably exit the activity)
+ if (SystemClock.elapsedRealtime() - lastBackClick > 2000){
+ lastBackClick = SystemClock.elapsedRealtime();
+ Toast.makeText(this, "Click again to close the app",
+ Toast.LENGTH_LONG).show();
+ return true;
+ }
+
+ lastBackClick = SystemClock.elapsedRealtime();
+ return super.onKeyDown(keyCode, event);
+ }
+
+
+ //----------------------------------------------------------------------------
+ // Listener interface for onNewIntent
+ //
+
+ public interface NewIntentListener {
+ void onNewIntent(Intent intent);
+ }
+
+ private List<NewIntentListener> newIntentListeners = null;
+
+ public void registerNewIntentListener(NewIntentListener listener) {
+ if ( this.newIntentListeners == null )
+ this.newIntentListeners = Collections.synchronizedList(new ArrayList<NewIntentListener>());
+ this.newIntentListeners.add(listener);
+ }
+
+ public void unregisterNewIntentListener(NewIntentListener listener) {
+ if ( this.newIntentListeners == null )
+ return;
+ this.newIntentListeners.remove(listener);
+ }
+
+ @Override
+ protected void onNewIntent(Intent intent) {
+ if ( this.newIntentListeners == null )
+ return;
+ this.onResume();
+ synchronized ( this.newIntentListeners ) {
+ Iterator<NewIntentListener> iterator = this.newIntentListeners.iterator();
+ while ( iterator.hasNext() ) {
+ (iterator.next()).onNewIntent(intent);
+ }
+ }
+ }
+
+ //----------------------------------------------------------------------------
+ // Listener interface for onActivityResult
+ //
+
+ public interface ActivityResultListener {
+ void onActivityResult(int requestCode, int resultCode, Intent data);
+ }
+
+ private List<ActivityResultListener> activityResultListeners = null;
+
+ public void registerActivityResultListener(ActivityResultListener listener) {
+ if ( this.activityResultListeners == null )
+ this.activityResultListeners = Collections.synchronizedList(new ArrayList<ActivityResultListener>());
+ this.activityResultListeners.add(listener);
+ }
+
+ public void unregisterActivityResultListener(ActivityResultListener listener) {
+ if ( this.activityResultListeners == null )
+ return;
+ this.activityResultListeners.remove(listener);
+ }
+
+ @Override
+ protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
+ if ( this.activityResultListeners == null )
+ return;
+ this.onResume();
+ synchronized ( this.activityResultListeners ) {
+ Iterator<ActivityResultListener> iterator = this.activityResultListeners.iterator();
+ while ( iterator.hasNext() )
+ (iterator.next()).onActivityResult(requestCode, resultCode, intent);
+ }
+ }
+
+ public static void start_service(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument
+ ) {
+ _do_start_service(
+ serviceTitle, serviceDescription, pythonServiceArgument, true
+ );
+ }
+
+ public static void start_service_not_as_foreground(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument
+ ) {
+ _do_start_service(
+ serviceTitle, serviceDescription, pythonServiceArgument, false
+ );
+ }
+
+ public static void _do_start_service(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument,
+ boolean showForegroundNotification
+ ) {
+ Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
+ String argument = PythonActivity.mActivity.getFilesDir().getAbsolutePath();
+ String app_root_dir = PythonActivity.mActivity.getAppRoot();
+ String entry_point = PythonActivity.mActivity.getEntryPoint(app_root_dir + "/service");
+ serviceIntent.putExtra("androidPrivate", argument);
+ serviceIntent.putExtra("androidArgument", app_root_dir);
+ serviceIntent.putExtra("serviceEntrypoint", "service/" + entry_point);
+ serviceIntent.putExtra("pythonName", "python");
+ serviceIntent.putExtra("pythonHome", app_root_dir);
+ serviceIntent.putExtra("pythonPath", app_root_dir + ":" + app_root_dir + "/lib");
+ serviceIntent.putExtra("serviceStartAsForeground",
+ (showForegroundNotification ? "true" : "false")
+ );
+ serviceIntent.putExtra("serviceTitle", serviceTitle);
+ serviceIntent.putExtra("serviceDescription", serviceDescription);
+ serviceIntent.putExtra("pythonServiceArgument", pythonServiceArgument);
+ PythonActivity.mActivity.startService(serviceIntent);
+ }
+
+ public static void stop_service() {
+ Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
+ PythonActivity.mActivity.stopService(serviceIntent);
+ }
+
+}
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/jniLibs/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/jniLibs/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/libs/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/libs/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/res/drawable/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/res/drawable/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/res/mipmap/.gitkeep b/pythonforandroid/bootstraps/qt/build/src/main/res/mipmap/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
new file mode 100644
index 0000000000..057794e4ed
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
@@ -0,0 +1,106 @@
+<?xml version="1.0" encoding="utf-8"?>
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+ android:versionCode="{{ args.numeric_version }}"
+ android:versionName="{{ args.version }}"
+ android:installLocation="auto">
+
+ <supports-screens
+ android:smallScreens="true"
+ android:normalScreens="true"
+ android:largeScreens="true"
+ android:anyDensity="true"
+ {% if args.min_sdk_version >= 9 %}
+ android:xlargeScreens="true"
+ {% endif %}
+ />
+
+ <uses-sdk android:minSdkVersion="{{ args.min_sdk_version }}" android:targetSdkVersion="{{ android_api }}" />
+ <uses-feature android:glEsVersion="0x00020000" />
+
+ <!-- Set permissions -->
+ {% for perm in args.permissions %}
+ <uses-permission android:name="{{ perm.name }}"{% if perm.maxSdkVersion %} android:maxSdkVersion="{{ perm.maxSdkVersion }}"{% endif %}{% if perm.usesPermissionFlags %} android:usesPermissionFlags="{{ perm.usesPermissionFlags }}"{% endif %} />
+ {% endfor %}
+
+ {% if args.wakelock %}
+ <uses-permission android:name="android.permission.WAKE_LOCK" />
+ {% endif %}
+
+ {% if args.billing_pubkey %}
+ <uses-permission android:name="com.android.vending.BILLING" />
+ {% endif %}
+
+ {{ args.extra_manifest_xml }}
+
+ <application android:name="org.qtproject.qt.android.bindings.QtApplication"
+ android:label="@string/app_name"
+ {% if debug %}android:debuggable="true"{% endif %}
+ android:icon="@mipmap/icon"
+ android:allowBackup="{{ args.allow_backup }}"
+ android:fullBackupOnly="false"
+ {% if args.backup_rules %}android:fullBackupContent="@xml/{{ args.backup_rules }}"{% endif %}
+ {{ args.extra_manifest_application_arguments }}
+ android:theme="{{args.android_apptheme}}{% if not args.window %}.Fullscreen{% endif %}"
+ android:hardwareAccelerated="true"
+ >
+ <!--
+ android:extractNativeLibs="true" = needed for smaller apk size
+ android:requestLegacyExternalStorage="true"
+ android:allowNativeHeapPointerTagging="false"
+ -->
+ {% for l in args.android_used_libs %}
+ <uses-library android:name="{{ l }}" />
+ {% endfor %}
+
+ {% for m in args.meta_data %}
+ <meta-data android:name="{{ m.split('=', 1)[0] }}" android:value="{{ m.split('=', 1)[-1] }}"/>{% endfor %}
+ <meta-data android:name="wakelock" android:value="{% if args.wakelock %}1{% else %}0{% endif %}"/>
+
+ <activity android:name="{{args.android_entrypoint}}"
+ android:label="@string/app_name"
+ android:configChanges="mcc|mnc|locale|touchscreen|keyboard|keyboardHidden|navigation|orientation|screenLayout|fontScale|uiMode{% if args.min_sdk_version >= 8 %}|uiMode{% endif %}{% if args.min_sdk_version >= 13 %}|screenSize|smallestScreenSize{% endif %}{% if args.min_sdk_version >= 17 %}|layoutDirection{% endif %}{% if args.min_sdk_version >= 24 %}|density{% endif %}"
+ android:screenOrientation="{{ args.manifest_orientation }}"
+ android:exported="true"
+ {% if args.activity_launch_mode %}
+ android:launchMode="{{ args.activity_launch_mode }}"
+ {% endif %}
+ >
+
+ <intent-filter>
+ <action android:name="android.intent.action.MAIN" />
+ <category android:name="android.intent.category.LAUNCHER" />
+ </intent-filter>
+
+ {%- if args.intent_filters -%}
+ {{- args.intent_filters -}}
+ {%- endif -%}
+
+ <!-- ToDo: Need more meta-data. Adapt accordingly -->
+ <!-- https://doc.qt.io/qt-6/android-manifest-file-configuration.html#qt-specific-meta-data -->
+ <meta-data
+ android:name="android.app.lib_name"
+ android:value="main"/>
+
+ <meta-data
+ android:name="android.app.extract_android_style"
+ android:value="minimal" />
+ </activity>
+
+ {% if service or args.launcher %}
+ <service android:name="{{ args.service_class_name }}"
+ android:process=":pythonservice" />
+ {% endif %}
+ {% for name in service_names %}
+ <service android:name="{{ args.package }}.Service{{ name|capitalize }}"
+ android:process=":service_{{ name }}" />
+ {% endfor %}
+ {% for name in native_services %}
+ <service android:name="{{ name }}" />
+ {% endfor %}
+
+ {% for a in args.add_activity %}
+ <activity android:name="{{ a }}"></activity>
+ {% endfor %}
+ </application>
+
+</manifest>
diff --git a/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
new file mode 100644
index 0000000000..d423f4152b
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
@@ -0,0 +1,27 @@
+<?xml version='1.0' encoding='utf-8'?>
+<resources>
+ <array name="qt_libs">
+ <item>{{ arch }};c++_shared</item>
+ {%- for qt_lib in qt_libs %}
+ <item>{{ arch }};Qt6{{ qt_lib }}_{{ arch }}</item>
+ {%- endfor -%}
+ </array>
+
+ <array name="load_local_libs">
+ {%- for load_local_lib in load_local_libs %}
+ <item>{{ arch }};lib{{ load_local_lib }}_{{ arch }}.so</item>
+ {%- endfor -%}
+ <item>{{ arch }};libshiboken6.abi3.so</item>
+ <item>{{ arch }};libpyside6.abi3.so</item>
+ {%- for qt_lib in qt_libs %}
+ <item>{{ arch }};Qt{{ qt_lib }}.abi3.so</item>
+ {% if qt_lib == "Qml" -%}
+ <item>{{ arch }};libpyside6qml.abi3.so</item>
+ {% endif %}
+ {%- endfor -%}
+ </array>
+
+ <string name="static_init_classes">{{ init_classes }}</string>
+ <string name="use_local_qt_libs">1</string>
+ <string name="bundle_local_qt_libs">1</string>
+</resources>
diff --git a/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
new file mode 100644
index 0000000000..41c20ac663
--- /dev/null
+++ b/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
@@ -0,0 +1,6 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <string name="app_name">{{ args.name }}</string>
+ <string name="private_version">{{ private_version }}</string>
+ <string name="presplash_color">{{ args.presplash_color }}</string>
+</resources>
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index dc53fd4337..bbd61e603d 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -449,7 +449,7 @@ def unpack(self, arch):
extraction_filename = join(
self.ctx.packages_path, self.name, filename)
if isfile(extraction_filename):
- if extraction_filename.endswith('.zip'):
+ if extraction_filename.endswith(('.zip', '.whl')):
try:
sh.unzip(extraction_filename)
except (sh.ErrorReturnCode_1, sh.ErrorReturnCode_2):
@@ -1171,6 +1171,9 @@ def reduce_object_file_names(self, dirn):
parts = file_basename.split('.')
if len(parts) <= 2:
continue
+ # PySide6 libraries end with .abi3.so
+ if parts[1] == "abi3":
+ continue
move(filen, join(file_dirname, parts[0] + '.so'))
diff --git a/pythonforandroid/recipes/android/__init__.py b/pythonforandroid/recipes/android/__init__.py
index e568ac8d9e..608d9ee738 100644
--- a/pythonforandroid/recipes/android/__init__.py
+++ b/pythonforandroid/recipes/android/__init__.py
@@ -35,7 +35,7 @@ def prebuild_arch(self, arch):
ctx_bootstrap = ctx_bootstrap.decode('utf-8')
bootstrap = bootstrap_name = ctx_bootstrap
is_sdl2 = (bootstrap_name == "sdl2")
- if bootstrap_name in ["sdl2", "webview", "service_only", "service_library"]:
+ if bootstrap_name in ["sdl2", "webview", "service_only", "service_library", "qt"]:
java_ns = u'org.kivy.android'
jni_ns = u'org/kivy/android'
else:
diff --git a/pythonforandroid/recipes/genericndkbuild/__init__.py b/pythonforandroid/recipes/genericndkbuild/__init__.py
index 901f208986..8b2a9c26a2 100644
--- a/pythonforandroid/recipes/genericndkbuild/__init__.py
+++ b/pythonforandroid/recipes/genericndkbuild/__init__.py
@@ -21,6 +21,8 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True, with_python=True):
with_python=with_python,
)
env['APP_ALLOW_MISSING_DEPS'] = 'true'
+ # required for Qt bootstrap
+ env['PREFERRED_ABI'] = arch.arch
return env
def build_arch(self, arch):
diff --git a/testapps/on_device_unit_tests/test_app/main.py b/testapps/on_device_unit_tests/test_app/main.py
index 48bf0dc33d..31422e8939 100644
--- a/testapps/on_device_unit_tests/test_app/main.py
+++ b/testapps/on_device_unit_tests/test_app/main.py
@@ -13,6 +13,7 @@
- A kivy unittest app (sdl2 bootstrap)
- A unittest app (webview bootstrap)
- A non-gui unittests app
+ - A non-gui Qt app (qt bootstrap)
If you install/build this app via the `setup.py` file, a file named
`app_requirements.txt` will be generated which will contain the requirements
diff --git a/testapps/on_device_unit_tests/test_app/tests/test_requirements.py b/testapps/on_device_unit_tests/test_app/tests/test_requirements.py
index e4104f8300..451f9fbf0c 100644
--- a/testapps/on_device_unit_tests/test_app/tests/test_requirements.py
+++ b/testapps/on_device_unit_tests/test_app/tests/test_requirements.py
@@ -251,3 +251,25 @@ def test_run_module(self):
import libtorrent as lt
print('Imported libtorrent version {}'.format(lt.version))
+
+
+class Pyside6TestCase(PythonTestMixIn, TestCase):
+ module_import = 'PySide6'
+
+ def test_run_module(self):
+ import PySide6
+ from PySide6.QtCore import QDateTime
+ from PySide6 import QtWidgets
+
+ print(f"Imported PySide6 version {PySide6.__version__}")
+ print(f"Current date and time obtained from PySide6 : {QDateTime.currentDateTime().toString()}")
+
+
+class Shiboken6TestCase(PythonTestMixIn, TestCase):
+ module_import = 'shiboken6'
+
+ def test_run_module(self):
+ import shiboken6
+ from shiboken6 import Shiboken
+
+ print('Imported shiboken6 version {}'.format(shiboken6.__version__))
diff --git a/testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6Android.jar b/testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6Android.jar
new file mode 100644
index 0000000000000000000000000000000000000000..c09f18fa6bed03ca8d2e778f3d834eaf68812a7c
GIT binary patch
literal 134543
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%0AWv;R>(VFxG?KgJ+
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
REWy7zP6*`X5*TzL{|~M8<s<+A
literal 0
HcmV?d00001
diff --git a/testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6AndroidBindings.jar b/testapps/on_device_unit_tests/test_qt/jar/PySide6/jar/Qt6AndroidBindings.jar
new file mode 100644
index 0000000000000000000000000000000000000000..d656c34d7a6cec39924ca3f56042de2e6c462aa5
GIT binary patch
literal 23527
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
literal 0
HcmV?d00001
diff --git a/testapps/on_device_unit_tests/test_qt/recipes/PySide6/__init__.py b/testapps/on_device_unit_tests/test_qt/recipes/PySide6/__init__.py
new file mode 100644
index 0000000000..6c795746ee
--- /dev/null
+++ b/testapps/on_device_unit_tests/test_qt/recipes/PySide6/__init__.py
@@ -0,0 +1,63 @@
+# Copyright (C) 2023 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+
+import shutil
+import zipfile
+from os.path import join
+from pathlib import Path
+
+from pythonforandroid.logger import info
+from pythonforandroid.recipe import PythonRecipe
+
+
+class PySideRecipe(PythonRecipe):
+ version = '6.6.0a1'
+ # This will download the aarch64 wheel from the Qt servers.
+ # This wheel is only for testing purposes. This test will be update when PySide releases
+ # official PySide6 Android wheels.
+ url = ("https://download.qt.io/snapshots/ci/pyside/test/Android/aarch64/"
+ "PySide6-6.6.0a1-6.6.0-cp37-abi3-android_aarch64.whl")
+ wheel_name = 'PySide6-6.6.0a1-6.6.0-cp37-abi3-android_aarch64.whl'
+ depends = ["shiboken6"]
+ call_hostpython_via_targetpython = False
+ install_in_hostpython = False
+
+ def build_arch(self, arch):
+ """Unzip the wheel and copy into site-packages of target"""
+
+ self.wheel_path = join(self.ctx.packages_path, self.name, self.wheel_name)
+ info("Copying libc++_shared.so from SDK to be loaded on startup")
+ libcpp_path = f"{self.ctx.ndk.sysroot_lib_dir}/{arch.command_prefix}/libc++_shared.so"
+ shutil.copyfile(libcpp_path, Path(self.ctx.get_libs_dir(arch.arch)) / "libc++_shared.so")
+
+ info(f"Installing {self.name} into site-packages")
+ with zipfile.ZipFile(self.wheel_path, "r") as zip_ref:
+ info("Unzip wheels and copy into {}".format(self.ctx.get_python_install_dir(arch.arch)))
+ zip_ref.extractall(self.ctx.get_python_install_dir(arch.arch))
+
+ lib_dir = Path(f"{self.ctx.get_python_install_dir(arch.arch)}/PySide6/Qt/lib")
+
+ info("Copying Qt libraries to be loaded on startup")
+ shutil.copytree(lib_dir, self.ctx.get_libs_dir(arch.arch), dirs_exist_ok=True)
+ shutil.copyfile(lib_dir.parent.parent / "libpyside6.abi3.so",
+ Path(self.ctx.get_libs_dir(arch.arch)) / "libpyside6.abi3.so")
+
+ shutil.copyfile(lib_dir.parent.parent / "QtCore.abi3.so",
+ Path(self.ctx.get_libs_dir(arch.arch)) / "QtCore.abi3.so")
+
+ shutil.copyfile(lib_dir.parent.parent / "QtWidgets.abi3.so",
+ Path(self.ctx.get_libs_dir(arch.arch)) / "QtWidgets.abi3.so")
+
+ shutil.copyfile(lib_dir.parent.parent / "QtGui.abi3.so",
+ Path(self.ctx.get_libs_dir(arch.arch)) / "QtGui.abi3.so")
+
+ plugin_path = (lib_dir.parent / "plugins" / "platforms" /
+ f"libplugins_platforms_qtforandroid_{arch.arch}.so")
+
+ if plugin_path.exists():
+ shutil.copyfile(plugin_path,
+ (Path(self.ctx.get_libs_dir(arch.arch)) /
+ f"libplugins_platforms_qtforandroid_{arch.arch}.so"))
+
+
+recipe = PySideRecipe()
diff --git a/testapps/on_device_unit_tests/test_qt/recipes/shiboken6/__init__.py b/testapps/on_device_unit_tests/test_qt/recipes/shiboken6/__init__.py
new file mode 100644
index 0000000000..71feb18d27
--- /dev/null
+++ b/testapps/on_device_unit_tests/test_qt/recipes/shiboken6/__init__.py
@@ -0,0 +1,39 @@
+# Copyright (C) 2023 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+
+import shutil
+import zipfile
+from os.path import join
+from pathlib import Path
+
+from pythonforandroid.logger import info
+from pythonforandroid.recipe import PythonRecipe
+
+
+class ShibokenRecipe(PythonRecipe):
+ version = '6.6.0a1'
+ # This will download the aarch64 wheel from the Qt servers.
+ # This wheel is only for testing purposes. This test will be update when PySide releases
+ # official shiboken6 Android wheels.
+ url = ("https://download.qt.io/snapshots/ci/pyside/test/Android/aarch64/"
+ "shiboken6-6.6.0a1-6.6.0-cp37-abi3-android_aarch64.whl")
+ wheel_name = 'shiboken6-6.6.0a1-6.6.0-cp37-abi3-android_aarch64.whl'
+
+ call_hostpython_via_targetpython = False
+ install_in_hostpython = False
+
+ def build_arch(self, arch):
+ ''' Unzip the wheel and copy into site-packages of target'''
+
+ self.wheel_path = join(self.ctx.packages_path, self.name, self.wheel_name)
+ info('Installing {} into site-packages'.format(self.name))
+ with zipfile.ZipFile(self.wheel_path, 'r') as zip_ref:
+ info('Unzip wheels and copy into {}'.format(self.ctx.get_python_install_dir(arch.arch)))
+ zip_ref.extractall(self.ctx.get_python_install_dir(arch.arch))
+
+ lib_dir = Path(f"{self.ctx.get_python_install_dir(arch.arch)}/shiboken6")
+ shutil.copyfile(lib_dir / "libshiboken6.abi3.so",
+ Path(self.ctx.get_libs_dir(arch.arch)) / "libshiboken6.abi3.so")
+
+
+recipe = ShibokenRecipe()
diff --git a/tests/test_bootstrap.py b/tests/test_bootstrap.py
index 1e17a9bd18..742ea0ba73 100644
--- a/tests/test_bootstrap.py
+++ b/tests/test_bootstrap.py
@@ -144,9 +144,9 @@ def test_all_bootstraps(self):
"""A test which will initialize a bootstrap and will check if the
method :meth:`~pythonforandroid.bootstrap.Bootstrap.all_bootstraps `
returns the expected values, which should be: `empty", `service_only`,
- `webview` and `sdl2`
+ `webview`, `sdl2` and `qt`
"""
- expected_bootstraps = {"empty", "service_only", "service_library", "webview", "sdl2"}
+ expected_bootstraps = {"empty", "service_only", "service_library", "webview", "sdl2", "qt"}
set_of_bootstraps = Bootstrap.all_bootstraps()
self.assertEqual(
expected_bootstraps, expected_bootstraps & set_of_bootstraps
@@ -344,6 +344,7 @@ def bootstrap_name(self):
name of the bootstrap to test"""
raise NotImplementedError("Not implemented in GenericBootstrapTest")
+ @mock.patch("pythonforandroid.bootstraps.qt.open", create=True)
@mock.patch("pythonforandroid.bootstraps.service_only.open", create=True)
@mock.patch("pythonforandroid.bootstraps.webview.open", create=True)
@mock.patch("pythonforandroid.bootstraps.sdl2.open", create=True)
@@ -370,6 +371,7 @@ def test_assemble_distribution(
mock_open_sdl2_files,
mock_open_webview_files,
mock_open_service_only_files,
+ mock_open_qt_files
):
"""
A test for any overwritten method of
@@ -410,6 +412,7 @@ def test_assemble_distribution(
"sdl2": mock_open_sdl2_files,
"webview": mock_open_webview_files,
"service_only": mock_open_service_only_files,
+ "qt": mock_open_qt_files
}
expected_open_calls = {
"sdl2": [
@@ -418,6 +421,7 @@ def test_assemble_distribution(
],
"webview": [mock.call("local.properties", "w")],
"service_only": [mock.call("local.properties", "w")],
+ "qt": [mock.call("local.properties", "w")]
}
mock_open_bs = mock_open_bootstraps[self.bootstrap_name]
# test that the expected calls has been called
@@ -659,3 +663,15 @@ def test_assemble_distribution(self, *args):
with self.assertRaises(SystemExit) as e:
bs.assemble_distribution()
self.assertEqual(e.exception.args[0], 1)
+
+
+class TestBootstrapQt(GenericBootstrapTest, unittest.TestCase):
+ """
+ An inherited class of `GenericBootstrapTest` and `unittest.TestCase` which
+ will be used to perform tests for
+ :class:`~pythonforandroid.bootstraps.qt.BootstrapQt`.
+ """
+
+ @property
+ def bootstrap_name(self):
+ return "qt"
From 704523095f4157dd6418ef62d867f58584af050a Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Jan 2024 11:25:38 +0530
Subject: [PATCH 018/158] recipes: new `materialyoucolor` recipe
---
.../recipes/materialyoucolor/__init__.py | 13 +++++++++++++
1 file changed, 13 insertions(+)
create mode 100644 pythonforandroid/recipes/materialyoucolor/__init__.py
diff --git a/pythonforandroid/recipes/materialyoucolor/__init__.py b/pythonforandroid/recipes/materialyoucolor/__init__.py
new file mode 100644
index 0000000000..6ae3ad2caf
--- /dev/null
+++ b/pythonforandroid/recipes/materialyoucolor/__init__.py
@@ -0,0 +1,13 @@
+from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+
+
+class MaterialyoucolorRecipe(CompiledComponentsPythonRecipe):
+ need_stl_shared = True
+ stl_lib_name = "c++_shared"
+ version = "2.0.5"
+ url = "https://github.com/T-Dynamos/materialyoucolor-pyhton/archive/refs/tags/v{version}.tar.gz"
+ depends = ["setuptools"]
+ call_hostpython_via_targetpython = False
+
+
+recipe = MaterialyoucolorRecipe()
From f06e5e39a86076389179508ea35b6e5c0c0bb6f1 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sat, 20 Jan 2024 10:48:58 +0100
Subject: [PATCH 019/158] Bump Kivy version to `2.3.0` (#2952)
---
pythonforandroid/recipes/kivy/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index a2dc8798ae..5cb56611e7 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -22,7 +22,7 @@ def is_kivy_affected_by_deadlock_issue(recipe=None, arch=None):
class KivyRecipe(CythonRecipe):
- version = '2.2.1'
+ version = '2.3.0'
url = 'https://github.com/kivy/kivy/archive/{version}.zip'
name = 'kivy'
From 4dcb61ea5976b2c048a8bc533a8868995f116f88 Mon Sep 17 00:00:00 2001
From: prolenodev <89631771+prolenodev@users.noreply.github.com>
Date: Sun, 21 Jan 2024 22:49:48 +0800
Subject: [PATCH 020/158] Update OpenSSL version to `1.1.1w` (#2958)
---
pythonforandroid/recipes/openssl/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/openssl/__init__.py b/pythonforandroid/recipes/openssl/__init__.py
index 520fe6da1b..766c10e361 100644
--- a/pythonforandroid/recipes/openssl/__init__.py
+++ b/pythonforandroid/recipes/openssl/__init__.py
@@ -47,7 +47,7 @@ class OpenSSLRecipe(Recipe):
version = '1.1'
'''the major minor version used to link our recipes'''
- url_version = '1.1.1m'
+ url_version = '1.1.1w'
'''the version used to download our libraries'''
url = 'https://www.openssl.org/source/openssl-{url_version}.tar.gz'
From f47fed606157bde6d1e82d58bfde339efdc3a65b Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sun, 21 Jan 2024 16:06:09 +0100
Subject: [PATCH 021/158] Update CHANGELOG.md and bump version to 2024.01.21
---
CHANGELOG.md | 158 +++++++++++++++++++++++++++++++++++
pythonforandroid/__init__.py | 2 +-
2 files changed, 159 insertions(+), 1 deletion(-)
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 2ce4711cbb..5686bcbb88 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,163 @@
# Changelog
+## [v2024.01.21](https://github.com/kivy/python-for-android/tree/v2024.01.21)
+
+[Full Changelog](https://github.com/kivy/python-for-android/compare/v2023.09.16...v2024.01.21)
+
+**Fixed bugs:**
+
+- Update documentation copyright [\#2921](https://github.com/kivy/python-for-android/issues/2921)
+- Support mail address is broken [\#2899](https://github.com/kivy/python-for-android/issues/2899)
+- doc/macos/jdk: invalid brew install command provided. [\#2896](https://github.com/kivy/python-for-android/issues/2896)
+- pyzmq recipe build fail [\#2818](https://github.com/kivy/python-for-android/issues/2818)
+- Existing distribution not detected due to pip package casing mismatch [\#2494](https://github.com/kivy/python-for-android/issues/2494)
+- unknown argument "fp-model" and strict is not a directory or a file [\#2359](https://github.com/kivy/python-for-android/issues/2359)
+- Copy past is not working on kivy mobile app [\#2270](https://github.com/kivy/python-for-android/issues/2270)
+- Flaky test failure in blacklist\(?\) - investigation needed [\#1781](https://github.com/kivy/python-for-android/issues/1781)
+- Problem with loding gevent: BadZipfile: File is not a zip file [\#1739](https://github.com/kivy/python-for-android/issues/1739)
+- ImportError when importing files containing \N{name} escape sequence [\#1060](https://github.com/kivy/python-for-android/issues/1060)
+- Error with permission specification via setup.cfg [\#985](https://github.com/kivy/python-for-android/issues/985)
+
+**Closed issues:**
+
+- Build failed: Could not find `android` or `sdkmanager` binaries in Android SDK [\#2956](https://github.com/kivy/python-for-android/issues/2956)
+- Libffi - configure: error: C compiler cannot create executables \(WSL 2\) [\#2953](https://github.com/kivy/python-for-android/issues/2953)
+- G [\#2951](https://github.com/kivy/python-for-android/issues/2951)
+- Hh [\#2949](https://github.com/kivy/python-for-android/issues/2949)
+- Can't build for Android on macOS on M2 [\#2947](https://github.com/kivy/python-for-android/issues/2947)
+- BroadcastReceiver does not invoke the callback [\#2946](https://github.com/kivy/python-for-android/issues/2946)
+- Add pdf2docx library recipe [\#2941](https://github.com/kivy/python-for-android/issues/2941)
+- use build aar in kotlin app ,can't load /lib/arm64/libpybundle.so file [\#2940](https://github.com/kivy/python-for-android/issues/2940)
+- Feature Request: Pymssql [\#2936](https://github.com/kivy/python-for-android/issues/2936)
+- LXML v4.8.0 fails to build. [\#2928](https://github.com/kivy/python-for-android/issues/2928)
+- Tryin to apply a plugin fails [\#2926](https://github.com/kivy/python-for-android/issues/2926)
+- ModuleNotFoundError: No module named '\_sysconfigdata\_\_darwin\_darwin' [\#2925](https://github.com/kivy/python-for-android/issues/2925)
+- ReadTheDocs version is unclear. [\#2920](https://github.com/kivy/python-for-android/issues/2920)
+- How to get real file path from uri [\#2911](https://github.com/kivy/python-for-android/issues/2911)
+- And [\#2910](https://github.com/kivy/python-for-android/issues/2910)
+- ModuleNotFoundError: No module named 'backports'
+ [\#2909](https://github.com/kivy/python-for-android/issues/2909)
+- not able to acess files unless connected to adb once [\#2907](https://github.com/kivy/python-for-android/issues/2907)
+- opening files in other apps [\#2906](https://github.com/kivy/python-for-android/issues/2906)
+- ImportError: dlopen failed: cannot locate symbol "\_ZTVSt9bad\_alloc" [\#2903](https://github.com/kivy/python-for-android/issues/2903)
+- Fails to build pyjnius [\#2902](https://github.com/kivy/python-for-android/issues/2902)
+- Kivy app crashes on startup [\#2895](https://github.com/kivy/python-for-android/issues/2895)
+- aar file does not import properly in version v2023.09.16 [\#2894](https://github.com/kivy/python-for-android/issues/2894)
+- App is crashing with Pyrebase4 [\#2893](https://github.com/kivy/python-for-android/issues/2893)
+- shared libs builds with 32 bit arch instaead of 64 bit [\#2888](https://github.com/kivy/python-for-android/issues/2888)
+- liblzma download error [\#2885](https://github.com/kivy/python-for-android/issues/2885)
+- Misconfiguration causing failure in compilation. [\#2879](https://github.com/kivy/python-for-android/issues/2879)
+- cygrpc.so is for EM\_X86\_64 \(62\) instead of EM\_AARCH64 \(183\) [\#2853](https://github.com/kivy/python-for-android/issues/2853)
+- Are you able to build cffi==1.15.1? [\#2847](https://github.com/kivy/python-for-android/issues/2847)
+- java.lang.IllegalStateException [\#2844](https://github.com/kivy/python-for-android/issues/2844)
+- \[BUG\]: ctypes: AttributeError: undefined symbol: PyCapsule\_New [\#2840](https://github.com/kivy/python-for-android/issues/2840)
+- kivy cant load image in requesturl android [\#2832](https://github.com/kivy/python-for-android/issues/2832)
+- Feature Request: Add Python `3.11` support [\#2798](https://github.com/kivy/python-for-android/issues/2798)
+- Error Build APK FIle using Flask [\#2783](https://github.com/kivy/python-for-android/issues/2783)
+- macOS: gwadlew fails at build tools stage \(newest build tools is 34.0.0-rc3, brew/openjdk@20\). [\#2781](https://github.com/kivy/python-for-android/issues/2781)
+- Kivy python Error loading video on some android device [\#2780](https://github.com/kivy/python-for-android/issues/2780)
+- buildozer/p4a.prerequisites: enable automation build with no questions asked. [\#2778](https://github.com/kivy/python-for-android/issues/2778)
+- \_python\_bundle does not exist...this not looks good, all python recipes should have this folder, should we expect a crash soon? [\#2773](https://github.com/kivy/python-for-android/issues/2773)
+- Background service implemented using Pyjnius does not auto-restart on Kivy APK close [\#2772](https://github.com/kivy/python-for-android/issues/2772)
+- \[JVM\]: FLAG\_IMMUTABLE or FLAG\_MUTABLE is required when a PendingIntent is created [\#2759](https://github.com/kivy/python-for-android/issues/2759)
+- there is an issue with playing video from URL on the latest p4a releases [\#2744](https://github.com/kivy/python-for-android/issues/2744)
+- App crahes at launch on specific devices \(\[libpython3.9.so\] \_PyEval\_EvalFrameDefault\) \(Adreno 730?\) [\#2723](https://github.com/kivy/python-for-android/issues/2723)
+- Pandas giving error in Buildozer [\#2719](https://github.com/kivy/python-for-android/issues/2719)
+- buildozer -v android debug [\#2711](https://github.com/kivy/python-for-android/issues/2711)
+- \[proposed feature-request\] Lacking psutil recipe [\#2707](https://github.com/kivy/python-for-android/issues/2707)
+- \[ERROR\]: Build failed: Asked to compile for no Archs, so failing. [\#2685](https://github.com/kivy/python-for-android/issues/2685)
+- Feature Request: Give more access to the android project folder inside of the dist folder [\#2614](https://github.com/kivy/python-for-android/issues/2614)
+- `shutil.copy()` fails on external removable storage devices [\#2589](https://github.com/kivy/python-for-android/issues/2589)
+- jnius can't find class "org.kivy.android.PythonActivity" with webview [\#2533](https://github.com/kivy/python-for-android/issues/2533)
+- \[MACOS\] Android app crashes on start when using macos to build [\#2519](https://github.com/kivy/python-for-android/issues/2519)
+- Pillow-SIMD recipe? [\#2420](https://github.com/kivy/python-for-android/issues/2420)
+- --asset & directories [\#2413](https://github.com/kivy/python-for-android/issues/2413)
+- dlopen failed: cannot locate symbol "\_\_register\_atfork" on Android 5.0 [\#2410](https://github.com/kivy/python-for-android/issues/2410)
+- dlib module not found error [\#2395](https://github.com/kivy/python-for-android/issues/2395)
+- lxml build failed for x86 arch [\#2369](https://github.com/kivy/python-for-android/issues/2369)
+- Android 10 storage permission denied [\#2364](https://github.com/kivy/python-for-android/issues/2364)
+- for pytorch [\#2353](https://github.com/kivy/python-for-android/issues/2353)
+- Problem with ffmpeg on Android [\#2345](https://github.com/kivy/python-for-android/issues/2345)
+- NLTK recipe for python for android [\#2320](https://github.com/kivy/python-for-android/issues/2320)
+- build\_tools\_versions comparison code fails for 'Android Rebuilds' SDKs because of different folder naming conventions [\#2318](https://github.com/kivy/python-for-android/issues/2318)
+- verify downloads using sha256? [\#2294](https://github.com/kivy/python-for-android/issues/2294)
+- outdated recipes [\#2277](https://github.com/kivy/python-for-android/issues/2277)
+- Custom recipe for scipy fails with permission issue [\#2267](https://github.com/kivy/python-for-android/issues/2267)
+- Kivy application generated crashes instantly with dlopen failed [\#2266](https://github.com/kivy/python-for-android/issues/2266)
+- EGLlib: validate\_display: 92 error 3008 \(EGL\_BAD\_DISPLAY\) : App crashes immediately \(kivymd\) \(Buildozer\) [\#2258](https://github.com/kivy/python-for-android/issues/2258)
+- libEGL : EGLNativeWindowType disconnect failed [\#2253](https://github.com/kivy/python-for-android/issues/2253)
+- Hao to support multiprocess Queue in Android [\#2249](https://github.com/kivy/python-for-android/issues/2249)
+- autoclass: Class only found when called in specific places? [\#2242](https://github.com/kivy/python-for-android/issues/2242)
+- the app crach in time of import psycopg2 [\#2240](https://github.com/kivy/python-for-android/issues/2240)
+- env must be a dict [\#2170](https://github.com/kivy/python-for-android/issues/2170)
+- Pandas doesn't work [\#2157](https://github.com/kivy/python-for-android/issues/2157)
+- Webview bootstrap can't find 'org.jnius.NativeInvocationHandler'. [\#2140](https://github.com/kivy/python-for-android/issues/2140)
+- clang++: error: linker command failed with exit code 1 [\#2082](https://github.com/kivy/python-for-android/issues/2082)
+- ModuleNotFoundError: No module named 'setuptools' [\#2078](https://github.com/kivy/python-for-android/issues/2078)
+- Scraping web pages with javascript [\#2052](https://github.com/kivy/python-for-android/issues/2052)
+- open webbrowser regsiter\(\) error [\#2047](https://github.com/kivy/python-for-android/issues/2047)
+- Missing javaclass when using able with previously working recipe [\#2041](https://github.com/kivy/python-for-android/issues/2041)
+- :Class not found b'org/kivy/android/PythonActivity$ActivityResultListener' [\#2039](https://github.com/kivy/python-for-android/issues/2039)
+- App\(using socket and opencv\) crash on opening [\#2038](https://github.com/kivy/python-for-android/issues/2038)
+- android apk is crashing after displaying splash screen on phone [\#2030](https://github.com/kivy/python-for-android/issues/2030)
+- Leverage Docker image caching [\#2009](https://github.com/kivy/python-for-android/issues/2009)
+- entrypoint confusion with python3 [\#1999](https://github.com/kivy/python-for-android/issues/1999)
+- Android app crash on opening - Python Initialize [\#1987](https://github.com/kivy/python-for-android/issues/1987)
+- Error building APK: "Missing 'name' key attribute on element activity at AndroidManifest.xml" [\#1979](https://github.com/kivy/python-for-android/issues/1979)
+- Ugent issues on Webview \(Android Back Button to main App\) [\#1961](https://github.com/kivy/python-for-android/issues/1961)
+- JavaException: JVM exception occurred: Fail to connect to camera service [\#1943](https://github.com/kivy/python-for-android/issues/1943)
+- Python version number must have subversion? cannot find Python-3.7.tgz [\#1941](https://github.com/kivy/python-for-android/issues/1941)
+- dlopen failed: jnius.so is for EM\_ARM \(40\) instead of EM\_386 \(3\) [\#1927](https://github.com/kivy/python-for-android/issues/1927)
+- Matplotlib recipe depends on local environment [\#1900](https://github.com/kivy/python-for-android/issues/1900)
+- main window jumps up and down [\#1876](https://github.com/kivy/python-for-android/issues/1876)
+- ctypes.pythonapi issues; getting AttributeError: undefined symbol [\#1866](https://github.com/kivy/python-for-android/issues/1866)
+- \[enhancement\] do not rebuild already built packages [\#1860](https://github.com/kivy/python-for-android/issues/1860)
+- Matplotlib recipe fails sometimes [\#1859](https://github.com/kivy/python-for-android/issues/1859)
+- p4a build with NDK r18b: clang: error: unknown argument: '-mandroid' [\#1853](https://github.com/kivy/python-for-android/issues/1853)
+- Activity lifecycle issues. after onDestroy, application will become unusable [\#1844](https://github.com/kivy/python-for-android/issues/1844)
+- Service AutoRestart did not work [\#1823](https://github.com/kivy/python-for-android/issues/1823)
+- Android debug results in error involving clang++ and linker. [\#1796](https://github.com/kivy/python-for-android/issues/1796)
+- seek\(\) method on a file object doesn't use right arguments [\#1768](https://github.com/kivy/python-for-android/issues/1768)
+- Same issue w/ -lpython2.7 not found, workaround [\#1753](https://github.com/kivy/python-for-android/issues/1753)
+- Several issues when installing packages via pip [\#1745](https://github.com/kivy/python-for-android/issues/1745)
+- Publish a new Kivy Launcher for Python 3 [\#1638](https://github.com/kivy/python-for-android/issues/1638)
+- Travis conditional boostrap build support [\#1588](https://github.com/kivy/python-for-android/issues/1588)
+- Error when execute APK only on device: ImportError: cannot import name \_htmlparser [\#1523](https://github.com/kivy/python-for-android/issues/1523)
+- onSensorChanged continuosly called during app execution [\#1498](https://github.com/kivy/python-for-android/issues/1498)
+- GC deadlock on subprocess [\#1461](https://github.com/kivy/python-for-android/issues/1461)
+- Code runs on old pygame backend but not on SDL2 [\#1411](https://github.com/kivy/python-for-android/issues/1411)
+- build-tools below 25 will not add jars [\#1345](https://github.com/kivy/python-for-android/issues/1345)
+- Flaky continuous integration [\#1306](https://github.com/kivy/python-for-android/issues/1306)
+- Icon/Logo Proposal [\#1264](https://github.com/kivy/python-for-android/issues/1264)
+- Unable to write the config [\#1151](https://github.com/kivy/python-for-android/issues/1151)
+- p4a does not yet work with clang [\#1097](https://github.com/kivy/python-for-android/issues/1097)
+- android module seems to eat up a character from java properties [\#945](https://github.com/kivy/python-for-android/issues/945)
+- TypeError: a bytes-like object is required, not 'str' [\#856](https://github.com/kivy/python-for-android/issues/856)
+- Feature request: access to all permissions [\#843](https://github.com/kivy/python-for-android/issues/843)
+- Extending the launcher [\#565](https://github.com/kivy/python-for-android/issues/565)
+
+**Merged pull requests:**
+
+- Update OpenSSL version to `1.1.1w` [\#2958](https://github.com/kivy/python-for-android/pull/2958) ([prolenodev](https://github.com/prolenodev))
+- Bump Kivy version to `2.3.0` [\#2952](https://github.com/kivy/python-for-android/pull/2952) ([misl6](https://github.com/misl6))
+- `sourceCompatibility` 1.7 and `targetCompatibility` 1.7 are obsolete, use 1.8 by default [\#2942](https://github.com/kivy/python-for-android/pull/2942) ([misl6](https://github.com/misl6))
+- Remove redundant append into WHITELIST\_PATTERNS [\#2935](https://github.com/kivy/python-for-android/pull/2935) ([shyamnathp](https://github.com/shyamnathp))
+- Update sdl2 deps to reflect the same targeted in kivy/kivy [\#2927](https://github.com/kivy/python-for-android/pull/2927) ([misl6](https://github.com/misl6))
+- Update `python-for-android` prerequisites \(`Dockerfile`, `prerequisites.py`, docs\) [\#2923](https://github.com/kivy/python-for-android/pull/2923) ([misl6](https://github.com/misl6))
+- Update Contributing Guidelines and Readme [\#2922](https://github.com/kivy/python-for-android/pull/2922) ([Julian-O](https://github.com/Julian-O))
+- Initial support for PySide6 and Qt [\#2918](https://github.com/kivy/python-for-android/pull/2918) ([shyamnathp](https://github.com/shyamnathp))
+- Introduce FAQ [\#2917](https://github.com/kivy/python-for-android/pull/2917) ([Julian-O](https://github.com/Julian-O))
+- Add \(now mandatory\) `.readthedocs.yaml` file, add docs `requirements.txt` and update sphinx conf [\#2916](https://github.com/kivy/python-for-android/pull/2916) ([misl6](https://github.com/misl6))
+- enable json1 extenstion in sqlite3 [\#2915](https://github.com/kivy/python-for-android/pull/2915) ([HyTurtle](https://github.com/HyTurtle))
+- Bump `pyjnius` version to `1.6.1` [\#2914](https://github.com/kivy/python-for-android/pull/2914) ([misl6](https://github.com/misl6))
+- Remove `distutils` usage, as is not available anymore on Python `3.12` [\#2912](https://github.com/kivy/python-for-android/pull/2912) ([misl6](https://github.com/misl6))
+- Update Lottie player version [\#2900](https://github.com/kivy/python-for-android/pull/2900) ([HugoDaniel](https://github.com/HugoDaniel))
+- Merge master into develop [\#2892](https://github.com/kivy/python-for-android/pull/2892) ([misl6](https://github.com/misl6))
+- Add doc tests, make them pass. [\#2890](https://github.com/kivy/python-for-android/pull/2890) ([Julian-O](https://github.com/Julian-O))
+- Update Android gradle plugin to `8.1.1` and gradle to `8.0.2` [\#2887](https://github.com/kivy/python-for-android/pull/2887) ([misl6](https://github.com/misl6))
+- Add support for Python `3.11` and make it the default while building `hostpython3` and `python3` [\#2850](https://github.com/kivy/python-for-android/pull/2850) ([T-Dynamos](https://github.com/T-Dynamos))
+
+
## [v2023.09.16](https://github.com/kivy/python-for-android/tree/v2023.09.16)
[Full Changelog](https://github.com/kivy/python-for-android/compare/v2023.05.21...v2023.09.16)
diff --git a/pythonforandroid/__init__.py b/pythonforandroid/__init__.py
index bcabd44859..ecdf256efa 100644
--- a/pythonforandroid/__init__.py
+++ b/pythonforandroid/__init__.py
@@ -1 +1 @@
-__version__ = '2023.09.16'
+__version__ = '2024.01.21'
From 9cd52203d1395d9f51f2c5361350251f40a42edd Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Mathias=20Lindstr=C3=B6m?= <kuzeyron@gmail.com>
Date: Tue, 13 Feb 2024 20:13:31 +0200
Subject: [PATCH 022/158] Add support for display cutouts in manifest and add
methods to retrieve data (#2969)
* Fresh changes of the display-cutout functionality
* Fine tuning + added tools
* Updated lib
* Add missing None for default value while grabbing the sizes
---
doc/source/buildoptions.rst | 4 ++
.../bootstraps/common/build/build.py | 3 +
.../build/templates/AndroidManifest.tmpl.xml | 1 +
.../qt/build/templates/strings.tmpl.xml | 10 +++
.../build/templates/AndroidManifest.tmpl.xml | 1 +
.../sdl2/build/templates/strings.tmpl.xml | 10 +++
.../build/templates/AndroidManifest.tmpl.xml | 1 +
.../webview/build/templates/strings.tmpl.xml | 10 +++
.../android/src/android/display_cutout.py | 72 +++++++++++++++++++
9 files changed, 112 insertions(+)
create mode 100644 pythonforandroid/recipes/android/src/android/display_cutout.py
diff --git a/doc/source/buildoptions.rst b/doc/source/buildoptions.rst
index 782bd67295..634f506faf 100644
--- a/doc/source/buildoptions.rst
+++ b/doc/source/buildoptions.rst
@@ -72,6 +72,10 @@ options (this list may not be exhaustive):
- ``--permission``: A permission that needs to be declared into the App ``AndroidManifest.xml``.
For multiple permissions, add multiple ``--permission`` arguments.
``--home-app`` Gives you the option to set your application as a home app (launcher) on your Android device.
+ ``--display-cutout``: A display cutout is an area on some devices that extends into the display surface.
+ It allows for an edge-to-edge experience while providing space for important sensors on the front of the device.
+ (Available options are ``default``, ``shortEdges``, ``never`` and defaults to ``never``)
+ `Android documentation <https://developer.android.com/develop/ui/views/layout/display-cutout>`__.
.. Note ::
``--permission`` accepts the following syntaxes:
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index 29d16ea9f2..51e0a5fa94 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -793,6 +793,9 @@ def create_argument_parser():
'launcher, rather than a single app.'))
ap.add_argument('--home-app', dest='home_app', action='store_true', default=False,
help=('Turn your application into a home app (launcher)'))
+ ap.add_argument('--display-cutout', dest='display_cutout', default='never',
+ help=('Enables display-cutout that renders around the area (notch) on '
+ 'some devices that extends into the display surface'))
ap.add_argument('--permission', dest='permissions', action='append', default=[],
help='The permissions to give this app.', nargs='+')
ap.add_argument('--meta-data', dest='meta_data', action='append', default=[],
diff --git a/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
index 057794e4ed..8ccff2027a 100644
--- a/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/qt/build/templates/AndroidManifest.tmpl.xml
@@ -61,6 +61,7 @@
android:configChanges="mcc|mnc|locale|touchscreen|keyboard|keyboardHidden|navigation|orientation|screenLayout|fontScale|uiMode{% if args.min_sdk_version >= 8 %}|uiMode{% endif %}{% if args.min_sdk_version >= 13 %}|screenSize|smallestScreenSize{% endif %}{% if args.min_sdk_version >= 17 %}|layoutDirection{% endif %}{% if args.min_sdk_version >= 24 %}|density{% endif %}"
android:screenOrientation="{{ args.manifest_orientation }}"
android:exported="true"
+ android:theme="@style/KivySupportCutout"
{% if args.activity_launch_mode %}
android:launchMode="{{ args.activity_launch_mode }}"
{% endif %}
diff --git a/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
index 41c20ac663..fd15c25f47 100644
--- a/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
+++ b/pythonforandroid/bootstraps/qt/build/templates/strings.tmpl.xml
@@ -1,5 +1,15 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
+ <style name="KivySupportCutout">
+ <item name="android:windowNoTitle">true</item>
+ <!-- Display cutout is an area on some devices that extends into the display surface -->
+ {% if args.display_cutout != 'never'%}
+ <item name="android:windowLayoutInDisplayCutoutMode">{{ args.display_cutout }}</item>
+ <item name="android:windowTranslucentStatus">true</item>
+ <item name="android:windowTranslucentNavigation">true</item>
+ <item name="android:windowFullscreen">true</item>
+ {% endif %}
+ </style>
<string name="app_name">{{ args.name }}</string>
<string name="private_version">{{ private_version }}</string>
<string name="presplash_color">{{ args.presplash_color }}</string>
diff --git a/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
index a887a53d54..c31bb3f747 100644
--- a/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
@@ -70,6 +70,7 @@
android:configChanges="mcc|mnc|locale|touchscreen|keyboard|keyboardHidden|navigation|orientation|screenLayout|fontScale|uiMode{% if args.min_sdk_version >= 8 %}|uiMode{% endif %}{% if args.min_sdk_version >= 13 %}|screenSize|smallestScreenSize{% endif %}{% if args.min_sdk_version >= 17 %}|layoutDirection{% endif %}{% if args.min_sdk_version >= 24 %}|density{% endif %}"
android:screenOrientation="{{ args.manifest_orientation }}"
android:exported="true"
+ android:theme="@style/KivySupportCutout"
{% if args.activity_launch_mode %}
android:launchMode="{{ args.activity_launch_mode }}"
{% endif %}
diff --git a/pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml b/pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml
index c8025518be..17e376adbd 100644
--- a/pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml
+++ b/pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml
@@ -1,5 +1,15 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
+ <style name="KivySupportCutout">
+ <item name="android:windowNoTitle">true</item>
+ <!-- Display cutout is an area on some devices that extends into the display surface -->
+ {% if args.display_cutout != 'never'%}
+ <item name="android:windowLayoutInDisplayCutoutMode">{{ args.display_cutout }}</item>
+ <item name="android:windowTranslucentStatus">true</item>
+ <item name="android:windowTranslucentNavigation">true</item>
+ <item name="android:windowFullscreen">true</item>
+ {% endif %}
+ </style>
<string name="app_name">{{ args.name }}</string>
<string name="private_version">{{ private_version }}</string>
<string name="presplash_color">{{ args.presplash_color }}</string>
diff --git a/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
index f9e4fa3c61..9b436b1fa4 100644
--- a/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
+++ b/pythonforandroid/bootstraps/webview/build/templates/AndroidManifest.tmpl.xml
@@ -62,6 +62,7 @@
android:configChanges="mcc|mnc|locale|touchscreen|keyboard|keyboardHidden|navigation|orientation|screenLayout|fontScale|uiMode{% if args.min_sdk_version >= 8 %}|uiMode{% endif %}{% if args.min_sdk_version >= 13 %}|screenSize|smallestScreenSize{% endif %}{% if args.min_sdk_version >= 17 %}|layoutDirection{% endif %}{% if args.min_sdk_version >= 24 %}|density{% endif %}"
android:screenOrientation="{{ args.manifest_orientation }}"
android:exported="true"
+ android:theme="@style/KivySupportCutout"
{% if args.activity_launch_mode %}
android:launchMode="{{ args.activity_launch_mode }}"
{% endif %}
diff --git a/pythonforandroid/bootstraps/webview/build/templates/strings.tmpl.xml b/pythonforandroid/bootstraps/webview/build/templates/strings.tmpl.xml
index 41c20ac663..fd15c25f47 100644
--- a/pythonforandroid/bootstraps/webview/build/templates/strings.tmpl.xml
+++ b/pythonforandroid/bootstraps/webview/build/templates/strings.tmpl.xml
@@ -1,5 +1,15 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
+ <style name="KivySupportCutout">
+ <item name="android:windowNoTitle">true</item>
+ <!-- Display cutout is an area on some devices that extends into the display surface -->
+ {% if args.display_cutout != 'never'%}
+ <item name="android:windowLayoutInDisplayCutoutMode">{{ args.display_cutout }}</item>
+ <item name="android:windowTranslucentStatus">true</item>
+ <item name="android:windowTranslucentNavigation">true</item>
+ <item name="android:windowFullscreen">true</item>
+ {% endif %}
+ </style>
<string name="app_name">{{ args.name }}</string>
<string name="private_version">{{ private_version }}</string>
<string name="presplash_color">{{ args.presplash_color }}</string>
diff --git a/pythonforandroid/recipes/android/src/android/display_cutout.py b/pythonforandroid/recipes/android/src/android/display_cutout.py
new file mode 100644
index 0000000000..dbe5d8a137
--- /dev/null
+++ b/pythonforandroid/recipes/android/src/android/display_cutout.py
@@ -0,0 +1,72 @@
+from jnius import autoclass
+from kivy.core.window import Window
+
+from android import mActivity
+
+__all__ = ('get_cutout_pos', 'get_cutout_size', 'get_width_of_bar',
+ 'get_height_of_bar', 'get_size_of_bar')
+
+
+def _core_cutout():
+ decorview = mActivity.getWindow().getDecorView()
+ cutout = decorview.rootWindowInsets.displayCutout
+
+ return cutout.boundingRects.get(0)
+
+
+def get_cutout_pos():
+ """ Get position of the display-cutout.
+ Returns integer for each positions (xy)
+ """
+ try:
+ cutout = _core_cutout()
+ return int(cutout.left), Window.height - int(cutout.height())
+ except Exception:
+ # Doesn't have a camera builtin with the display
+ return 0, 0
+
+
+def get_cutout_size():
+ """ Get the size (xy) of the front camera.
+ Returns size with float values
+ """
+ try:
+ cutout = _core_cutout()
+ return float(cutout.width()), float(cutout.height())
+ except Exception:
+ # Doesn't have a camera builtin with the display
+ return 0., 0.
+
+
+def get_height_of_bar(bar_target=None):
+ """ Get the height of either statusbar or navigationbar
+ bar_target = status or navigation and defaults to status
+ """
+ bar_target = bar_target or 'status'
+
+ if bar_target not in ('status', 'navigation'):
+ raise Exception("bar_target must be 'status' or 'navigation'")
+
+ try:
+ displayMetrics = autoclass('android.util.DisplayMetrics')
+ mActivity.getWindowManager().getDefaultDisplay().getMetrics(displayMetrics())
+ resources = mActivity.getResources()
+ resourceId = resources.getIdentifier(f'{bar_target}_bar_height', 'dimen',
+ 'android')
+
+ return float(max(resources.getDimensionPixelSize(resourceId), 0))
+ except Exception:
+ # Getting the size is not supported on older Androids
+ return 0.
+
+
+def get_width_of_bar(bar_target=None):
+ " Get the width of the bar "
+ return Window.width
+
+
+def get_size_of_bar(bar_target=None):
+ """ Get the size of either statusbar or navigationbar
+ bar_target = status or navigation and defaults to status
+ """
+ return get_width_of_bar(), get_height_of_bar(bar_target)
From 10175717fafcbc8cbfe21d1e91f8a824a8fae611 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Mon, 19 Feb 2024 18:25:16 +0530
Subject: [PATCH 023/158] (`recipes/materialyoucolor`): update to 2.0.7
---
pythonforandroid/recipes/materialyoucolor/__init__.py | 8 +++-----
1 file changed, 3 insertions(+), 5 deletions(-)
diff --git a/pythonforandroid/recipes/materialyoucolor/__init__.py b/pythonforandroid/recipes/materialyoucolor/__init__.py
index 6ae3ad2caf..39bbdc328a 100644
--- a/pythonforandroid/recipes/materialyoucolor/__init__.py
+++ b/pythonforandroid/recipes/materialyoucolor/__init__.py
@@ -1,13 +1,11 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
-class MaterialyoucolorRecipe(CompiledComponentsPythonRecipe):
- need_stl_shared = True
+class MaterialyoucolorRecipe(CppCompiledComponentsPythonRecipe):
stl_lib_name = "c++_shared"
- version = "2.0.5"
+ version = "2.0.7"
url = "https://github.com/T-Dynamos/materialyoucolor-pyhton/archive/refs/tags/v{version}.tar.gz"
depends = ["setuptools"]
- call_hostpython_via_targetpython = False
recipe = MaterialyoucolorRecipe()
From 4c5e34e201f9e4ede1ae771a9362da353dbe9623 Mon Sep 17 00:00:00 2001
From: Pramod Immaneni <pimmaneni@rivian.com>
Date: Wed, 28 Feb 2024 16:06:16 -0800
Subject: [PATCH 024/158] Passing arch container instead of dereferenced arch
string
---
pythonforandroid/build.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/build.py b/pythonforandroid/build.py
index 4777e2f934..0cc26db264 100644
--- a/pythonforandroid/build.py
+++ b/pythonforandroid/build.py
@@ -764,7 +764,7 @@ def run_pymodules_install(ctx, arch, modules, project_dir=None,
if project_dir is not None and (
project_has_setup_py(project_dir) and not ignore_setup_py
):
- run_setuppy_install(ctx, project_dir, env, arch.arch)
+ run_setuppy_install(ctx, project_dir, env, arch)
elif not ignore_setup_py:
info("No setup.py found in project directory: " + str(project_dir))
From b48204f55eb41e8cc9bdf99d72942faab1da09fb Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Thu, 29 Feb 2024 10:22:35 +0100
Subject: [PATCH 025/158] :white_check_mark: Fix test_get_dep_names_of_package
Recent pythonf-for-android release added setuptools to install_requires.
Pinning to previous release fixes to test.
The error was:
```
def test_get_dep_names_of_package():
# TEST 1 from external ref:
# Check that colorama is returned without the install condition when
# just getting the names (it has a "; ..." conditional originally):
dep_names = get_dep_names_of_package("python-for-android")
assert "colorama" in dep_names
> assert "setuptools" not in dep_names
E AssertionError: assert 'setuptools' not in {'Jinja2', 'appdirs', 'build', 'colorama', 'packaging', 'setuptools', ...}
tests/test_pythonpackage_basic.py:90: AssertionError
```
---
tests/test_pythonpackage_basic.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tests/test_pythonpackage_basic.py b/tests/test_pythonpackage_basic.py
index e98a5f99b0..f1eb68369c 100644
--- a/tests/test_pythonpackage_basic.py
+++ b/tests/test_pythonpackage_basic.py
@@ -85,7 +85,7 @@ def test_get_dep_names_of_package():
# TEST 1 from external ref:
# Check that colorama is returned without the install condition when
# just getting the names (it has a "; ..." conditional originally):
- dep_names = get_dep_names_of_package("python-for-android")
+ dep_names = get_dep_names_of_package("python-for-android==2023.9.16")
assert "colorama" in dep_names
assert "setuptools" not in dep_names
try:
From 88260ae4a724bc977a8b218842734edc54e56f33 Mon Sep 17 00:00:00 2001
From: "wei.cao" <shanghaicw0929@163.com>
Date: Thu, 7 Mar 2024 14:22:20 +0800
Subject: [PATCH 026/158] MANAGE_EXTERNAL_STORAGE
android.permission.MANAGE_EXTERNAL_STORAGE
---
pythonforandroid/recipes/android/src/android/permissions.py | 3 +++
1 file changed, 3 insertions(+)
diff --git a/pythonforandroid/recipes/android/src/android/permissions.py b/pythonforandroid/recipes/android/src/android/permissions.py
index 0ce568fbe4..aa4e310799 100644
--- a/pythonforandroid/recipes/android/src/android/permissions.py
+++ b/pythonforandroid/recipes/android/src/android/permissions.py
@@ -449,6 +449,9 @@ class Permission:
WRITE_VOICEMAIL = (
"com.android.voicemail.permission.WRITE_VOICEMAIL"
)
+ MANAGE_EXTERNAL_STORAGE = ( # Convenient use of paths to manage files
+ "android.permission.MANAGE_EXTERNAL_STORAGE"
+ )
PERMISSION_GRANTED = 0
From a3762dc7c46a7aed2c584d632a72fb263bfee3c0 Mon Sep 17 00:00:00 2001
From: "wei.cao" <shanghaicw0929@163.com>
Date: Thu, 7 Mar 2024 15:14:03 +0800
Subject: [PATCH 027/158] fix Flake8 tests error
fix Flake8 tests error
---
pythonforandroid/recipes/android/src/android/permissions.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/android/src/android/permissions.py b/pythonforandroid/recipes/android/src/android/permissions.py
index aa4e310799..132aab0823 100644
--- a/pythonforandroid/recipes/android/src/android/permissions.py
+++ b/pythonforandroid/recipes/android/src/android/permissions.py
@@ -449,7 +449,7 @@ class Permission:
WRITE_VOICEMAIL = (
"com.android.voicemail.permission.WRITE_VOICEMAIL"
)
- MANAGE_EXTERNAL_STORAGE = ( # Convenient use of paths to manage files
+ MANAGE_EXTERNAL_STORAGE = ( # Convenient use of paths to manage files
"android.permission.MANAGE_EXTERNAL_STORAGE"
)
From 9ef3f9f55caa4cdd4853f33c603f82cd73125777 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Thu, 29 Feb 2024 15:06:51 +0100
Subject: [PATCH 028/158] :white_check_mark: improve sh mocking reliability
Make sure the mocking also works on setup without cmake or make.
Build dependencies should not be required to run uni tests.
The error was:
```
AttributeError: <module 'sh' from 'venv/lib/python3.11/site-packages/sh.py'> does not have the attribute 'cmake'
```
---
tests/recipes/recipe_lib_test.py | 6 +++---
tests/recipes/test_openal.py | 4 ++--
2 files changed, 5 insertions(+), 5 deletions(-)
diff --git a/tests/recipes/recipe_lib_test.py b/tests/recipes/recipe_lib_test.py
index 1f57fe23c1..34aec36dd2 100644
--- a/tests/recipes/recipe_lib_test.py
+++ b/tests/recipes/recipe_lib_test.py
@@ -88,7 +88,7 @@ def test_build_arch(
with mock.patch(
f"pythonforandroid.recipes.{self.recipe_name}.sh.Command"
) as mock_sh_command, mock.patch(
- f"pythonforandroid.recipes.{self.recipe_name}.sh.make"
+ f"pythonforandroid.recipes.{self.recipe_name}.sh.make", create=True
) as mock_make:
self.recipe.build_arch(self.arch)
@@ -130,9 +130,9 @@ def test_build_arch(
# Since the following mocks are dynamic,
# we mock it inside a Context Manager
with mock.patch(
- f"pythonforandroid.recipes.{self.recipe_name}.sh.make"
+ f"pythonforandroid.recipes.{self.recipe_name}.sh.make", create=True
) as mock_make, mock.patch(
- f"pythonforandroid.recipes.{self.recipe_name}.sh.cmake"
+ f"pythonforandroid.recipes.{self.recipe_name}.sh.cmake", create=True
) as mock_cmake:
self.recipe.build_arch(self.arch)
diff --git a/tests/recipes/test_openal.py b/tests/recipes/test_openal.py
index a03d5cd271..fec7eeaa0e 100644
--- a/tests/recipes/test_openal.py
+++ b/tests/recipes/test_openal.py
@@ -9,8 +9,8 @@ class TestOpenalRecipe(BaseTestForCmakeRecipe, unittest.TestCase):
"""
recipe_name = "openal"
- @mock.patch("pythonforandroid.recipes.openal.sh.cmake")
- @mock.patch("pythonforandroid.recipes.openal.sh.make")
+ @mock.patch("pythonforandroid.recipes.openal.sh.cmake", create=True)
+ @mock.patch("pythonforandroid.recipes.openal.sh.make", create=True)
@mock.patch("pythonforandroid.recipes.openal.sh.cp")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
From eca86d2596370cab1132eacd8a1406a85d53c30f Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Thu, 7 Mar 2024 12:21:35 +0100
Subject: [PATCH 029/158] :white_check_mark: Fix broken sdl2 mixer test
Reproduce with:
```
PYTHONPATH=. pytest tests/recipes/test_sdl2_mixer.py
```
The error was:
```
=================================== FAILURES ===================================
__________________ TestSDL2MixerRecipe.test_get_include_dirs ___________________
self = <test_sdl2_mixer.TestSDL2MixerRecipe testMethod=test_get_include_dirs>
def test_get_include_dirs(self):
> list_of_includes = self.recipe.get_include_dirs(self.arch)
tests/recipes/test_sdl2_mixer.py:12:
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
pythonforandroid/recipes/sdl2_mixer/__init__.py:13: in get_include_dirs
os.path.join(self.ctx.bootstrap.build_dir, "jni", "SDL2_mixer", "include")
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
a = None, p = ('jni', 'SDL2_mixer', 'include')
> ???
E TypeError: expected str, bytes or os.PathLike object, not NoneType
<frozen posixpath>:76: TypeError
```
Also do some minor code clean up and absolute import path fixes.
---
tests/recipes/test_sdl2_mixer.py | 8 +++++++-
tests/test_bootstrap.py | 8 ++++----
tests/test_recipe.py | 4 ++--
tox.ini | 1 -
4 files changed, 13 insertions(+), 8 deletions(-)
diff --git a/tests/recipes/test_sdl2_mixer.py b/tests/recipes/test_sdl2_mixer.py
index a583d9aa63..5ac472b7e1 100644
--- a/tests/recipes/test_sdl2_mixer.py
+++ b/tests/recipes/test_sdl2_mixer.py
@@ -1,5 +1,5 @@
import unittest
-from tests.recipes.recipe_lib_test import RecipeCtx
+from tests.recipes.recipe_ctx import RecipeCtx
class TestSDL2MixerRecipe(RecipeCtx, unittest.TestCase):
@@ -8,6 +8,12 @@ class TestSDL2MixerRecipe(RecipeCtx, unittest.TestCase):
"""
recipe_name = "sdl2_mixer"
+ def setUp(self):
+ """Setups bootstrap build_dir."""
+ super().setUp()
+ bootstrap = self.ctx.bootstrap
+ bootstrap.build_dir = bootstrap.get_build_dir()
+
def test_get_include_dirs(self):
list_of_includes = self.recipe.get_include_dirs(self.arch)
self.assertIsInstance(list_of_includes, list)
diff --git a/tests/test_bootstrap.py b/tests/test_bootstrap.py
index 742ea0ba73..eea284b8c9 100644
--- a/tests/test_bootstrap.py
+++ b/tests/test_bootstrap.py
@@ -15,12 +15,12 @@
from pythonforandroid.util import BuildInterruptingException
from pythonforandroid.androidndk import AndroidNDK
-from test_graph import get_fake_recipe
+from tests.test_graph import get_fake_recipe
-class BaseClassSetupBootstrap(object):
+class BaseClassSetupBootstrap:
"""
- An class object which is intended to be used as a base class to configure
+ An class which is intended to be used as a base class to configure
an inherited class of `unittest.TestCase`. This class will override the
`setUp` and `tearDown` methods.
"""
@@ -115,7 +115,7 @@ def test__cmp_bootstraps_by_priority(self):
) < 0)
# Test a random bootstrap is always lower priority than sdl2:
- class _FakeBootstrap(object):
+ class _FakeBootstrap:
def __init__(self, name):
self.name = name
bs1 = _FakeBootstrap("alpha")
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index a50ca444ab..006129f3a2 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -1,16 +1,16 @@
import os
import pytest
+import tempfile
import types
import unittest
import warnings
from unittest import mock
-from backports import tempfile
from pythonforandroid.build import Context
from pythonforandroid.recipe import Recipe, TargetPythonRecipe, import_recipe
from pythonforandroid.archs import ArchAarch_64
from pythonforandroid.bootstrap import Bootstrap
-from test_bootstrap import BaseClassSetupBootstrap
+from tests.test_bootstrap import BaseClassSetupBootstrap
def patch_logger(level):
diff --git a/tox.ini b/tox.ini
index 9b0432c82b..33fb0452ea 100644
--- a/tox.ini
+++ b/tox.ini
@@ -6,7 +6,6 @@ basepython = python3
deps =
pytest
py3: coveralls
- backports.tempfile
# posargs will be replaced by the tox args, so you can override pytest
# args e.g. `tox -- tests/test_graph.py`
commands = pytest {posargs:tests/}
From 69d970561d95ae52d5e847331db178e07a1def6e Mon Sep 17 00:00:00 2001
From: Andre Miras <AndreMiras@users.noreply.github.com>
Date: Sat, 9 Mar 2024 13:54:58 +0100
Subject: [PATCH 030/158] :loud_sound: Improve rebuild recipes logging & bump
`pyzbar` version (#2986)
Add the build command to the logs so it's more clear how to reproduce a
potential rebuild issue.
Bump pyzbar recipe to trigger a demo rebuild.
---
ci/rebuild_updated_recipes.py | 14 +++++++++-----
pythonforandroid/recipes/pyzbar/__init__.py | 2 +-
2 files changed, 10 insertions(+), 6 deletions(-)
diff --git a/ci/rebuild_updated_recipes.py b/ci/rebuild_updated_recipes.py
index 3a693a453d..a1ca96681d 100755
--- a/ci/rebuild_updated_recipes.py
+++ b/ci/rebuild_updated_recipes.py
@@ -59,14 +59,18 @@ def build(target_python, requirements, archs):
requirements.add(target_python.name)
requirements = ','.join(requirements)
logger.info('requirements: {}'.format(requirements))
+ build_command = [
+ 'setup.py', 'apk',
+ '--sdk-dir', android_sdk_home,
+ '--ndk-dir', android_ndk_home,
+ '--requirements', requirements
+ ] + [f"--arch={arch}" for arch in archs]
+ build_command_str = " ".join(build_command)
+ logger.info(f"Build command: {build_command_str}")
with current_directory('testapps/on_device_unit_tests/'):
# iterates to stream the output
- for line in sh.python(
- 'setup.py', 'apk', '--sdk-dir', android_sdk_home,
- '--ndk-dir', android_ndk_home, '--requirements',
- requirements, *[f"--arch={arch}" for arch in archs],
- _err_to_out=True, _iter=True):
+ for line in sh.python(*build_command, _err_to_out=True, _iter=True):
print(line)
diff --git a/pythonforandroid/recipes/pyzbar/__init__.py b/pythonforandroid/recipes/pyzbar/__init__.py
index cf78a558cd..5cd55d5096 100644
--- a/pythonforandroid/recipes/pyzbar/__init__.py
+++ b/pythonforandroid/recipes/pyzbar/__init__.py
@@ -4,7 +4,7 @@
class PyZBarRecipe(PythonRecipe):
- version = '0.1.7'
+ version = '0.1.9'
url = 'https://github.com/NaturalHistoryMuseum/pyzbar/archive/v{version}.tar.gz' # noqa
From ba77db07f1ac07c7b6b2ea32dab6e564dc906201 Mon Sep 17 00:00:00 2001
From: Andre Miras <AndreMiras@users.noreply.github.com>
Date: Sat, 9 Mar 2024 13:57:19 +0100
Subject: [PATCH 031/158] :fire: Remove unused Makefile variables (#2985)
---
Makefile | 8 --------
1 file changed, 8 deletions(-)
diff --git a/Makefile b/Makefile
index b894bed3ea..8ea28c5973 100644
--- a/Makefile
+++ b/Makefile
@@ -3,14 +3,6 @@ PIP=$(VIRTUAL_ENV)/bin/pip
TOX=`which tox`
ACTIVATE=$(VIRTUAL_ENV)/bin/activate
PYTHON=$(VIRTUAL_ENV)/bin/python
-FLAKE8=$(VIRTUAL_ENV)/bin/flake8
-PYTEST=$(VIRTUAL_ENV)/bin/pytest
-SOURCES=src/ tests/
-PYTHON_MAJOR_VERSION=3
-PYTHON_MINOR_VERSION=6
-PYTHON_VERSION=$(PYTHON_MAJOR_VERSION).$(PYTHON_MINOR_VERSION)
-PYTHON_MAJOR_MINOR=$(PYTHON_MAJOR_VERSION)$(PYTHON_MINOR_VERSION)
-PYTHON_WITH_VERSION=python$(PYTHON_VERSION)
DOCKER_IMAGE=kivy/python-for-android
ANDROID_SDK_HOME ?= $(HOME)/.android/android-sdk
ANDROID_NDK_HOME ?= $(HOME)/.android/android-ndk
From bb49cb7a30ce7d730c33eb5f45cf10a6313db371 Mon Sep 17 00:00:00 2001
From: T-Dynamos <anshdadwal298@gmail.com>
Date: Sun, 10 Mar 2024 08:24:06 +0530
Subject: [PATCH 032/158] recipes: new recipe `coincurve`
---
.../recipes/coincurve/__init__.py | 21 ++++++++
.../recipes/coincurve/coincurve.patch | 54 +++++++++++++++++++
.../recipes/libsecp256k1/__init__.py | 4 +-
3 files changed, 77 insertions(+), 2 deletions(-)
create mode 100644 pythonforandroid/recipes/coincurve/__init__.py
create mode 100644 pythonforandroid/recipes/coincurve/coincurve.patch
diff --git a/pythonforandroid/recipes/coincurve/__init__.py b/pythonforandroid/recipes/coincurve/__init__.py
new file mode 100644
index 0000000000..662772cb78
--- /dev/null
+++ b/pythonforandroid/recipes/coincurve/__init__.py
@@ -0,0 +1,21 @@
+import os
+from pythonforandroid.recipe import PythonRecipe
+
+
+class CoincurveRecipe(PythonRecipe):
+ version = "19.0.1"
+ url = "https://github.com/ofek/coincurve/archive/v{version}.tar.gz"
+ call_hostpython_via_targetpython = False
+ depends = ["setuptools", "libffi", "cffi", "libsecp256k1", "asn1crypto"]
+ patches = ["coincurve.patch"]
+
+ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
+ env = super(CoincurveRecipe, self).get_recipe_env(arch, with_flags_in_cc)
+ libsecp256k1 = self.get_recipe("libsecp256k1", self.ctx)
+ libsecp256k1_dir = libsecp256k1.get_build_dir(arch.arch)
+ env["CFLAGS"] += " -I" + os.path.join(libsecp256k1_dir, "include")
+ env["LDFLAGS"] += " -lsecp256k1"
+ return env
+
+
+recipe = CoincurveRecipe()
diff --git a/pythonforandroid/recipes/coincurve/coincurve.patch b/pythonforandroid/recipes/coincurve/coincurve.patch
new file mode 100644
index 0000000000..64bfc2aee5
--- /dev/null
+++ b/pythonforandroid/recipes/coincurve/coincurve.patch
@@ -0,0 +1,54 @@
+diff '--color=auto' -uNr coincurve-19.0.1/setup.py coincurve-19.0.1.patch/setup.py
+--- coincurve-19.0.1/setup.py 2024-03-02 10:40:59.000000000 +0530
++++ coincurve-19.0.1.patch/setup.py 2024-03-10 09:51:58.034737104 +0530
+@@ -47,6 +47,7 @@
+
+
+ def download_library(command):
++ return
+ if command.dry_run:
+ return
+ libdir = absolute('libsecp256k1')
+@@ -189,6 +190,7 @@
+ absolute('libsecp256k1/configure'),
+ '--disable-shared',
+ '--enable-static',
++ '--host=%s' % os.environ['TOOLCHAIN_PREFIX'],
+ '--disable-dependency-tracking',
+ '--with-pic',
+ '--enable-module-extrakeys',
+@@ -269,13 +271,7 @@
+ # ABI?: py_limited_api=True,
+ )
+
+- extension.extra_compile_args = [
+- subprocess.check_output(['pkg-config', '--cflags-only-I', 'libsecp256k1']).strip().decode('utf-8') # noqa S603
+- ]
+- extension.extra_link_args = [
+- subprocess.check_output(['pkg-config', '--libs-only-L', 'libsecp256k1']).strip().decode('utf-8'), # noqa S603
+- subprocess.check_output(['pkg-config', '--libs-only-l', 'libsecp256k1']).strip().decode('utf-8'), # noqa S603
+- ]
++ extension.extra_link_args = ["-lsecp256k1"]
+
+ if os.name == 'nt' or sys.platform == 'win32':
+ # Apparently, the linker on Windows interprets -lxxx as xxx.lib, not libxxx.lib
+@@ -340,7 +336,7 @@
+ license='MIT OR Apache-2.0',
+
+ python_requires='>=3.8',
+- install_requires=['asn1crypto', 'cffi>=1.3.0'],
++ install_requires=[],
+
+ packages=find_packages(exclude=('_cffi_build', '_cffi_build.*', 'libsecp256k1', 'tests')),
+ package_data=package_data,
+diff '--color=auto' -uNr coincurve-19.0.1/setup_support.py coincurve-19.0.1.patch/setup_support.py
+--- coincurve-19.0.1/setup_support.py 2024-03-02 10:40:59.000000000 +0530
++++ coincurve-19.0.1.patch/setup_support.py 2024-03-10 08:53:45.650056659 +0530
+@@ -56,6 +56,7 @@
+
+
+ def _find_lib():
++ return True
+ if 'COINCURVE_IGNORE_SYSTEM_LIB' in os.environ:
+ return False
+
diff --git a/pythonforandroid/recipes/libsecp256k1/__init__.py b/pythonforandroid/recipes/libsecp256k1/__init__.py
index f3a2772cf9..f228ad8cb5 100644
--- a/pythonforandroid/recipes/libsecp256k1/__init__.py
+++ b/pythonforandroid/recipes/libsecp256k1/__init__.py
@@ -9,8 +9,8 @@
class LibSecp256k1Recipe(Recipe):
built_libraries = {'libsecp256k1.so': '.libs'}
-
- url = 'https://github.com/bitcoin-core/secp256k1/archive/master.zip'
+ version = '0.4.1'
+ url = 'https://github.com/bitcoin-core/secp256k1/archive/refs/tags/v{version}.tar.gz'
def build_arch(self, arch):
env = self.get_recipe_env(arch)
From b92522fab879dbfc0028966ca3c59ef46ab7767d Mon Sep 17 00:00:00 2001
From: shyamnathp <shyamnath94.mec@gmail.com>
Date: Wed, 13 Mar 2024 18:05:48 +0100
Subject: [PATCH 033/158] Qt Bootstrap - Adapting to Qt 6.7 release (#2990)
* Qt lib.xml - add placeholder for bundled_libs
* Qt Bootstrap: Set environment variables
- remove QtNative.setEnvironmentVariable in favour of the Java API.
- This removes the dependency on QtNative.
---------
Co-authored-by: Shyamnath Premnadh <Shyamnath.Premnadh@qt.io>
---
.../java/org/kivy/android/PythonActivity.java | 29 +++++++++++++------
.../qt/build/templates/libs.tmpl.xml | 8 +++++
2 files changed, 28 insertions(+), 9 deletions(-)
diff --git a/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java b/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
index 81cad01616..169fd323bf 100644
--- a/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
+++ b/pythonforandroid/bootstraps/qt/build/src/main/java/org/kivy/android/PythonActivity.java
@@ -21,7 +21,6 @@
import android.content.pm.PackageManager;
import org.qtproject.qt.android.bindings.QtActivity;
-import org.qtproject.qt.android.QtNative;
public class PythonActivity extends QtActivity {
@@ -52,6 +51,18 @@ public String getEntryPoint(String search_dir) {
return "main.py";
}
+ public void setEnvironmentVariable(String key, String value) {
+ /**
+ * Sets an environment variable based on key/value.
+ **/
+ try {
+ android.system.Os.setenv(key, value, true);
+ } catch (Exception e) {
+ Log.e("Qt bootstrap", "Unable set environment variable:" + key + "=" + value);
+ e.printStackTrace();
+ }
+ }
+
@Override
public void onCreate(Bundle savedInstanceState) {
this.mActivity = this;
@@ -69,14 +80,14 @@ public void onCreate(Bundle savedInstanceState) {
String entry_point = getEntryPoint(app_root_dir);
Log.v(TAG, "Setting env vars for start.c and Python to use");
- QtNative.setEnvironmentVariable("ANDROID_ENTRYPOINT", entry_point);
- QtNative.setEnvironmentVariable("ANDROID_ARGUMENT", app_root_dir);
- QtNative.setEnvironmentVariable("ANDROID_APP_PATH", app_root_dir);
- QtNative.setEnvironmentVariable("ANDROID_PRIVATE", mFilesDirectory);
- QtNative.setEnvironmentVariable("ANDROID_UNPACK", app_root_dir);
- QtNative.setEnvironmentVariable("PYTHONHOME", app_root_dir);
- QtNative.setEnvironmentVariable("PYTHONPATH", app_root_dir + ":" + app_root_dir + "/lib");
- QtNative.setEnvironmentVariable("PYTHONOPTIMIZE", "2");
+ setEnvironmentVariable("ANDROID_ENTRYPOINT", entry_point);
+ setEnvironmentVariable("ANDROID_ARGUMENT", app_root_dir);
+ setEnvironmentVariable("ANDROID_APP_PATH", app_root_dir);
+ setEnvironmentVariable("ANDROID_PRIVATE", mFilesDirectory);
+ setEnvironmentVariable("ANDROID_UNPACK", app_root_dir);
+ setEnvironmentVariable("PYTHONHOME", app_root_dir);
+ setEnvironmentVariable("PYTHONPATH", app_root_dir + ":" + app_root_dir + "/lib");
+ setEnvironmentVariable("PYTHONOPTIMIZE", "2");
Log.v(TAG, "About to do super onCreate");
super.onCreate(savedInstanceState);
diff --git a/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml b/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
index d423f4152b..9093aa6f84 100644
--- a/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
+++ b/pythonforandroid/bootstraps/qt/build/templates/libs.tmpl.xml
@@ -1,5 +1,13 @@
<?xml version='1.0' encoding='utf-8'?>
<resources>
+
+ <!--
+ The bunlded_libs placeholder is needed for QtLoader.java. Otherwise the application will crash.
+ Adding extra libraries can be done through buildozer directly with android.add_libs_* option
+ -->
+ <array name="bundled_libs">
+ </array>
+
<array name="qt_libs">
<item>{{ arch }};c++_shared</item>
{%- for qt_lib in qt_libs %}
From fe0f790b1f9c3c56ee03fe2faded3d99307b5cfa Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Tue, 26 Mar 2024 16:41:09 +0530
Subject: [PATCH 034/158] (`recipes/materialyoucolor`): update to 2.0.9
---
pythonforandroid/recipes/materialyoucolor/__init__.py | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/pythonforandroid/recipes/materialyoucolor/__init__.py b/pythonforandroid/recipes/materialyoucolor/__init__.py
index 39bbdc328a..32d44a2714 100644
--- a/pythonforandroid/recipes/materialyoucolor/__init__.py
+++ b/pythonforandroid/recipes/materialyoucolor/__init__.py
@@ -3,8 +3,8 @@
class MaterialyoucolorRecipe(CppCompiledComponentsPythonRecipe):
stl_lib_name = "c++_shared"
- version = "2.0.7"
- url = "https://github.com/T-Dynamos/materialyoucolor-pyhton/archive/refs/tags/v{version}.tar.gz"
+ version = "2.0.9"
+ url = "https://github.com/T-Dynamos/materialyoucolor-python/releases/download/v{version}/materialyoucolor-{version}.tar.gz"
depends = ["setuptools"]
From 8110fafabc64192de627807da3f4425779c37969 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 31 Mar 2024 20:05:12 +0530
Subject: [PATCH 035/158] recipes: Introduce `RustCompiledComponentsRecipe`,
add `pydantic-core` and update `cryptography` (#2962)
---
Makefile | 4 +
pythonforandroid/recipe.py | 176 +++++++++++++++++-
.../recipes/cryptography/__init__.py | 25 +--
pythonforandroid/recipes/numpy/__init__.py | 20 +-
.../recipes/pydantic-core/__init__.py | 12 ++
pythonforandroid/recipes/pydantic/__init__.py | 12 --
.../recipes/setuptools/__init__.py | 2 +-
7 files changed, 221 insertions(+), 30 deletions(-)
create mode 100644 pythonforandroid/recipes/pydantic-core/__init__.py
delete mode 100644 pythonforandroid/recipes/pydantic/__init__.py
diff --git a/Makefile b/Makefile
index 8ea28c5973..19a2379ab2 100644
--- a/Makefile
+++ b/Makefile
@@ -23,8 +23,12 @@ virtualenv: $(VIRTUAL_ENV)
test:
$(TOX) -- tests/ --ignore tests/test_pythonpackage.py
+# Also install and configure rust
rebuild_updated_recipes: virtualenv
. $(ACTIVATE) && \
+ curl https://sh.rustup.rs -sSf | sh -s -- -y && \
+ . "$(HOME)/.cargo/env" && \
+ rustup target list && \
ANDROID_SDK_HOME=$(ANDROID_SDK_HOME) ANDROID_NDK_HOME=$(ANDROID_NDK_HOME) \
$(PYTHON) ci/rebuild_updated_recipes.py $(REBUILD_UPDATED_RECIPES_EXTRA_ARGS)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index bbd61e603d..d70571ad08 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1,4 +1,4 @@
-from os.path import basename, dirname, exists, isdir, isfile, join, realpath, split
+from os.path import basename, dirname, exists, isdir, isfile, join, realpath, split, sep
import glob
import hashlib
@@ -7,6 +7,7 @@
import sh
import shutil
import fnmatch
+import zipfile
import urllib.request
from urllib.request import urlretrieve
from os import listdir, unlink, environ, curdir, walk
@@ -20,7 +21,7 @@
import packaging.version
from pythonforandroid.logger import (
- logger, info, warning, debug, shprint, info_main)
+ logger, info, warning, debug, shprint, info_main, error)
from pythonforandroid.util import (
current_directory, ensure_dir, BuildInterruptingException, rmdir, move,
touch)
@@ -175,6 +176,7 @@ def download_file(self, url, target, cwd=None):
"""
if not url:
return
+
info('Downloading {} from {}'.format(self.name, url))
if cwd:
@@ -458,7 +460,6 @@ def unpack(self, arch):
# apparently happens sometimes with
# github zips
pass
- import zipfile
fileh = zipfile.ZipFile(extraction_filename, 'r')
root_directory = fileh.filelist[0].filename.split('/')[0]
if root_directory != basename(directory_name):
@@ -837,6 +838,9 @@ class PythonRecipe(Recipe):
on python2 or python3 which can break the dependency graph
'''
+ hostpython_prerequisites = []
+ '''List of hostpython packages required to build a recipe'''
+
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
@@ -930,6 +934,7 @@ def should_build(self, arch):
def build_arch(self, arch):
'''Install the Python module by calling setup.py install with
the target Python dir.'''
+ self.install_hostpython_prerequisites()
super().build_arch(arch)
self.install_python_package(arch)
@@ -958,9 +963,13 @@ def install_python_package(self, arch, name=None, env=None, is_dir=True):
def get_hostrecipe_env(self, arch):
env = environ.copy()
- env['PYTHONPATH'] = join(dirname(self.real_hostpython_location), 'Lib', 'site-packages')
+ env['PYTHONPATH'] = self.hostpython_site_dir
return env
+ @property
+ def hostpython_site_dir(self):
+ return join(dirname(self.real_hostpython_location), 'Lib', 'site-packages')
+
def install_hostpython_package(self, arch):
env = self.get_hostrecipe_env(arch)
real_hostpython = sh.Command(self.real_hostpython_location)
@@ -969,6 +978,27 @@ def install_hostpython_package(self, arch):
'--install-lib=Lib/site-packages',
_env=env, *self.setup_extra_args)
+ @property
+ def python_version(self):
+ return Recipe.get_recipe("python3", self.ctx).version
+
+ def install_hostpython_prerequisites(self, force_upgrade=True):
+ if len(self.hostpython_prerequisites) == 0:
+ return
+ pip_options = [
+ "install",
+ *self.hostpython_prerequisites,
+ "--target", self.hostpython_site_dir, "--python-version",
+ self.python_version,
+ # Don't use sources, instead wheels
+ "--only-binary=:all:",
+ "--no-deps"
+ ]
+ if force_upgrade:
+ pip_options.append("--upgrade")
+ # Use system's pip
+ shprint(sh.pip, *pip_options)
+
class CompiledComponentsPythonRecipe(PythonRecipe):
pre_build_ext = False
@@ -1127,6 +1157,144 @@ def get_recipe_env(self, arch, with_flags_in_cc=True):
return env
+class RustCompiledComponentsRecipe(PythonRecipe):
+ # Rust toolchain codes
+ # https://doc.rust-lang.org/nightly/rustc/platform-support.html
+ RUST_ARCH_CODES = {
+ "arm64-v8a": "aarch64-linux-android",
+ "armeabi-v7a": "armv7-linux-androideabi",
+ "x86_64": "x86_64-linux-android",
+ "x86": "i686-linux-android",
+ }
+
+ # Build python wheel using `maturin` instead
+ # of default `python -m build [...]`
+ use_maturin = False
+
+ # Directory where to find built wheel
+ # For normal build: "dist/*.whl"
+ # For maturin: "target/wheels/*-linux_*.whl"
+ built_wheel_pattern = None
+
+ call_hostpython_via_targetpython = False
+
+ def __init__(self, *arg, **kwargs):
+ super().__init__(*arg, **kwargs)
+ self.append_deps_if_absent(["python3"])
+ self.set_default_hostpython_deps()
+ if not self.built_wheel_pattern:
+ self.built_wheel_pattern = (
+ "target/wheels/*-linux_*.whl"
+ if self.use_maturin
+ else "dist/*.whl"
+ )
+
+ def set_default_hostpython_deps(self):
+ if not self.use_maturin:
+ self.hostpython_prerequisites += ["build", "setuptools_rust", "wheel", "pyproject_hooks"]
+ else:
+ self.hostpython_prerequisites += ["maturin"]
+
+ def append_deps_if_absent(self, deps):
+ for dep in deps:
+ if dep not in self.depends:
+ self.depends.append(dep)
+
+ def get_recipe_env(self, arch):
+ env = super().get_recipe_env(arch)
+
+ # Set rust build target
+ build_target = self.RUST_ARCH_CODES[arch.arch]
+ cargo_linker_name = "CARGO_TARGET_{}_LINKER".format(
+ build_target.upper().replace("-", "_")
+ )
+ env["CARGO_BUILD_TARGET"] = build_target
+ env[cargo_linker_name] = join(
+ self.ctx.ndk.llvm_prebuilt_dir,
+ "bin",
+ "{}{}-clang".format(
+ # NDK's Clang format
+ build_target.replace("7", "7a")
+ if build_target.startswith("armv7")
+ else build_target,
+ self.ctx.ndk_api,
+ ),
+ )
+ realpython_dir = Recipe.get_recipe("python3", self.ctx).get_build_dir(arch.arch)
+
+ env["RUSTFLAGS"] = "-Clink-args=-L{} -L{}".format(
+ self.ctx.get_libs_dir(arch.arch), join(realpython_dir, "android-build")
+ )
+
+ env["PYO3_CROSS_LIB_DIR"] = realpath(glob.glob(join(
+ realpython_dir, "android-build", "build",
+ "lib.linux-*-{}/".format(self.get_python_formatted_version()),
+ ))[0])
+
+ info_main("Ensuring rust build toolchain")
+ shprint(sh.rustup, "target", "add", build_target)
+
+ # Add host python to PATH
+ env["PATH"] = ("{hostpython_dir}:{old_path}").format(
+ hostpython_dir=Recipe.get_recipe(
+ "hostpython3", self.ctx
+ ).get_path_to_python(),
+ old_path=env["PATH"],
+ )
+ return env
+
+ def get_python_formatted_version(self):
+ parsed_version = packaging.version.parse(self.python_version)
+ return f"{parsed_version.major}.{parsed_version.minor}"
+
+ def check_host_deps(self):
+ if not hasattr(sh, "rustup"):
+ error(
+ "`rustup` was not found on host system."
+ "Please install it using :"
+ "\n`curl https://sh.rustup.rs -sSf | sh`\n"
+ )
+ exit(1)
+
+ def build_arch(self, arch):
+ self.check_host_deps()
+ self.install_hostpython_prerequisites()
+ build_dir = self.get_build_dir(arch.arch)
+ env = self.get_recipe_env(arch)
+ built_wheel = None
+
+ # Copy the exec with version info
+ hostpython_exec = join(
+ sep,
+ *self.hostpython_location.split(sep)[:-1],
+ "python{}".format(self.get_python_formatted_version()),
+ )
+ shprint(sh.cp, self.hostpython_location, hostpython_exec)
+
+ with current_directory(build_dir):
+ if self.use_maturin:
+ shprint(
+ sh.Command(join(self.hostpython_site_dir, "bin", "maturin")),
+ "build", "--interpreter", hostpython_exec, "--skip-auditwheel",
+ _env=env,
+ )
+ else:
+ shprint(
+ sh.Command(hostpython_exec),
+ "-m", "build", "--no-isolation", "--skip-dependency-check", "--wheel",
+ _env=env,
+ )
+ # Find the built wheel
+ built_wheel = realpath(glob.glob(self.built_wheel_pattern)[0])
+
+ info("Unzipping built wheel '{}'".format(basename(built_wheel)))
+
+ # Unzip .whl file into site-packages
+ with zipfile.ZipFile(built_wheel, "r") as zip_ref:
+ zip_ref.extractall(self.ctx.get_python_install_dir(arch.arch))
+ info("Successfully installed '{}'".format(basename(built_wheel)))
+
+
class TargetPythonRecipe(Recipe):
'''Class for target python recipes. Sets ctx.python_recipe to point to
itself, so as to know later what kind of Python was built or used.'''
diff --git a/pythonforandroid/recipes/cryptography/__init__.py b/pythonforandroid/recipes/cryptography/__init__.py
index 182c745996..8e476b3129 100644
--- a/pythonforandroid/recipes/cryptography/__init__.py
+++ b/pythonforandroid/recipes/cryptography/__init__.py
@@ -1,21 +1,24 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
+from pythonforandroid.recipe import RustCompiledComponentsRecipe
+from os.path import join
-class CryptographyRecipe(CompiledComponentsPythonRecipe):
+class CryptographyRecipe(RustCompiledComponentsRecipe):
+
name = 'cryptography'
- version = '2.8'
- url = 'https://github.com/pyca/cryptography/archive/{version}.tar.gz'
+ version = '42.0.1'
+ url = 'https://github.com/pyca/cryptography/archive/refs/tags/{version}.tar.gz'
depends = ['openssl', 'six', 'setuptools', 'cffi']
- call_hostpython_via_targetpython = False
+ # recipe built cffi does not work on apple M1
+ hostpython_prerequisites = ["semantic_version", "cffi"]
def get_recipe_env(self, arch):
env = super().get_recipe_env(arch)
-
- openssl_recipe = Recipe.get_recipe('openssl', self.ctx)
- env['CFLAGS'] += openssl_recipe.include_flags(arch)
- env['LDFLAGS'] += openssl_recipe.link_dirs_flags(arch)
- env['LIBS'] = openssl_recipe.link_libs_flags()
-
+ openssl_build_dir = self.get_recipe('openssl', self.ctx).get_build_dir(arch.arch)
+ build_target = self.RUST_ARCH_CODES[arch.arch].upper().replace("-", "_")
+ openssl_include = "{}_OPENSSL_INCLUDE_DIR".format(build_target)
+ openssl_libs = "{}_OPENSSL_LIB_DIR".format(build_target)
+ env[openssl_include] = join(openssl_build_dir, 'include')
+ env[openssl_libs] = join(openssl_build_dir)
return env
diff --git a/pythonforandroid/recipes/numpy/__init__.py b/pythonforandroid/recipes/numpy/__init__.py
index 55a0279770..7e51eba66b 100644
--- a/pythonforandroid/recipes/numpy/__init__.py
+++ b/pythonforandroid/recipes/numpy/__init__.py
@@ -1,4 +1,4 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
from pythonforandroid.logger import shprint, info
from pythonforandroid.util import current_directory
from multiprocessing import cpu_count
@@ -13,7 +13,11 @@ class NumpyRecipe(CompiledComponentsPythonRecipe):
version = '1.22.3'
url = 'https://pypi.python.org/packages/source/n/numpy/numpy-{version}.zip'
site_packages_name = 'numpy'
- depends = ['setuptools', 'cython']
+ depends = ["cython"]
+
+ # This build specifically requires setuptools version 59.2.0
+ hostpython_prerequisites = ["setuptools==59.2.0"]
+
install_in_hostpython = True
call_hostpython_via_targetpython = False
@@ -36,6 +40,18 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
return env
+ def build_arch(self, arch):
+ self.hostpython_prerequisites = ["setuptools==59.2.0"]
+ self.install_hostpython_prerequisites()
+
+ super().build_arch(arch)
+
+ # Post build step to restore setuptools version
+ self.hostpython_prerequisites = ["setuptools=={}".format(
+ Recipe.get_recipe("setuptools", self.ctx).version)
+ ]
+ self.install_hostpython_prerequisites()
+
def _build_compiled_components(self, arch):
info('Building compiled components in {}'.format(self.name))
diff --git a/pythonforandroid/recipes/pydantic-core/__init__.py b/pythonforandroid/recipes/pydantic-core/__init__.py
new file mode 100644
index 0000000000..8702dfc002
--- /dev/null
+++ b/pythonforandroid/recipes/pydantic-core/__init__.py
@@ -0,0 +1,12 @@
+from pythonforandroid.recipe import RustCompiledComponentsRecipe
+
+
+class PydanticcoreRecipe(RustCompiledComponentsRecipe):
+ version = "2.16.1"
+ url = "https://github.com/pydantic/pydantic-core/archive/refs/tags/v{version}.tar.gz"
+ use_maturin = True
+ hostpython_prerequisites = ["typing_extensions"]
+ site_packages_name = "pydantic_core"
+
+
+recipe = PydanticcoreRecipe()
diff --git a/pythonforandroid/recipes/pydantic/__init__.py b/pythonforandroid/recipes/pydantic/__init__.py
deleted file mode 100644
index 16e61e1b61..0000000000
--- a/pythonforandroid/recipes/pydantic/__init__.py
+++ /dev/null
@@ -1,12 +0,0 @@
-from pythonforandroid.recipe import PythonRecipe
-
-
-class PydanticRecipe(PythonRecipe):
- version = '1.10.4'
- url = 'https://github.com/pydantic/pydantic/archive/refs/tags/v{version}.zip'
- depends = ['setuptools']
- python_depends = ['Cython', 'devtools', 'email-validator', 'typing-extensions', 'python-dotenv']
- call_hostpython_via_targetpython = False
-
-
-recipe = PydanticRecipe()
diff --git a/pythonforandroid/recipes/setuptools/__init__.py b/pythonforandroid/recipes/setuptools/__init__.py
index 8190f8efd1..02b205023d 100644
--- a/pythonforandroid/recipes/setuptools/__init__.py
+++ b/pythonforandroid/recipes/setuptools/__init__.py
@@ -2,7 +2,7 @@
class SetuptoolsRecipe(PythonRecipe):
- version = '51.3.3'
+ version = '69.2.0'
url = 'https://pypi.python.org/packages/source/s/setuptools/setuptools-{version}.tar.gz'
call_hostpython_via_targetpython = False
install_in_hostpython = True
From 5d5ca2b57aa49c508950c927eb42d2ffe14c3cf6 Mon Sep 17 00:00:00 2001
From: Steve Kowalik <steven@wedontsleep.org>
Date: Tue, 9 Apr 2024 15:36:24 +1000
Subject: [PATCH 036/158] Fix mock assertion in ICU recipe
Python 3.12 now does not blindly any methods on mocked objects, which
masks issues. Correct the assertion of has_calls to assert_has_calls in
the ICU recipe.
Fixes #3002
---
tests/recipes/test_icu.py | 5 +++--
1 file changed, 3 insertions(+), 2 deletions(-)
diff --git a/tests/recipes/test_icu.py b/tests/recipes/test_icu.py
index 239b99e4c1..dacf65a0b4 100644
--- a/tests/recipes/test_icu.py
+++ b/tests/recipes/test_icu.py
@@ -52,13 +52,14 @@ def test_build_arch(
# We expect some calls to `sh.Command`
build_root = self.recipe.get_build_dir(self.arch.arch)
- mock_sh_command.has_calls(
+ mock_sh_command.assert_has_calls(
[
mock.call(
os.path.join(build_root, "source", "runConfigureICU")
),
mock.call(os.path.join(build_root, "source", "configure")),
- ]
+ ],
+ any_order=True
)
mock_ensure_dir.assert_called()
mock_chdir.assert_called()
From 33e1b460c11ae3cbf59e8842b6edf7627bc68a47 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sat, 4 May 2024 13:38:09 +0200
Subject: [PATCH 037/158] Sunset `apple-silicon-m1` self-hosted runner, as now
is supported by Github Hosted runners via `macos-latest` tag. Use `macos-13`
for runs on Intel macs (#3011)
---
.github/workflows/push.yml | 32 ++++++++++++++------------------
ci/osx_ci.sh | 13 -------------
2 files changed, 14 insertions(+), 31 deletions(-)
delete mode 100644 ci/osx_ci.sh
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 4eff5d0f61..0cbad15afe 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -110,7 +110,9 @@ jobs:
continue-on-error: true
strategy:
matrix:
- runs_on: [macos-latest, apple-silicon-m1]
+ # macos-latest (ATM macos-14) runs on Apple Silicon,
+ # macos-13 runs on Intel
+ runs_on: ['macos-latest', 'macos-13']
bootstrap:
- name: sdl2
target: testapps-with-numpy
@@ -124,25 +126,21 @@ jobs:
steps:
- name: Checkout python-for-android
uses: actions/checkout@v4
+ - name: Set up Python 3.x
+ uses: actions/setup-python@v5
+ with:
+ python-version: 3.x
- name: Install python-for-android
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
python3 -m pip install -e .
- name: Install prerequisites via pythonforandroid/prerequisites.py (Experimental)
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
python3 pythonforandroid/prerequisites.py
- name: Install dependencies (Legacy)
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
make --file ci/makefiles/osx.mk
- name: Build multi-arch apk Python 3 (armeabi-v7a, arm64-v8a, x86_64, x86)
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
make ${{ matrix.bootstrap.target }}
- name: Copy produced artifacts into dist/ (*.apk, *.aab)
run: |
@@ -199,7 +197,9 @@ jobs:
strategy:
matrix:
android_arch: ["arm64-v8a", "armeabi-v7a", "x86_64", "x86"]
- runs_on: [macos-latest, apple-silicon-m1]
+ # macos-latest (ATM macos-14) runs on Apple Silicon,
+ # macos-13 runs on Intel
+ runs_on: ['macos-latest', 'macos-13']
env:
ANDROID_HOME: ${HOME}/.android
ANDROID_SDK_ROOT: ${HOME}/.android/android-sdk
@@ -211,25 +211,21 @@ jobs:
uses: actions/checkout@v4
with:
fetch-depth: 0
+ - name: Set up Python 3.x
+ uses: actions/setup-python@v5
+ with:
+ python-version: 3.x
- name: Install python-for-android
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
python3 -m pip install -e .
- name: Install prerequisites via pythonforandroid/prerequisites.py (Experimental)
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
python3 pythonforandroid/prerequisites.py
- name: Install dependencies (Legacy)
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
make --file ci/makefiles/osx.mk
- name: Rebuild updated recipes
run: |
- source ci/osx_ci.sh
- arm64_set_path_and_python_version 3.9.7
make rebuild_updated_recipes
coveralls_finish:
diff --git a/ci/osx_ci.sh b/ci/osx_ci.sh
deleted file mode 100644
index 8cdd1ac1af..0000000000
--- a/ci/osx_ci.sh
+++ /dev/null
@@ -1,13 +0,0 @@
-#!/bin/bash
-set -e -x
-
-arm64_set_path_and_python_version(){
- python_version="$1"
- if [[ $(/usr/bin/arch) = arm64 ]]; then
- export PATH=/opt/homebrew/bin:$PATH
- eval "$(pyenv init --path)"
- pyenv install $python_version -s
- pyenv global $python_version
- export PATH=$(pyenv prefix)/bin:$PATH
- fi
-}
\ No newline at end of file
From 87a32be7feedc284d00acb303cad7da2d7a14a03 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Fri, 10 May 2024 00:58:37 +0530
Subject: [PATCH 038/158] recipe: introduce `PyProjectRecipe` and `MesonRecipe`
(to update `pandas`, `numpy` and other recipes) (#3007)
---
Dockerfile | 2 +
Makefile | 2 +-
doc/source/quickstart.rst | 2 +
pythonforandroid/recipe.py | 302 ++++++++++++------
pythonforandroid/recipes/Pillow/__init__.py | 2 +-
pythonforandroid/recipes/cppy/__init__.py | 14 -
.../recipes/cryptography/__init__.py | 8 +-
.../recipes/kiwisolver/__init__.py | 9 +-
.../recipes/matplotlib/__init__.py | 15 +-
.../recipes/matplotlib/skip_macos.patch | 12 +
pythonforandroid/recipes/numpy/__init__.py | 94 ++----
.../add_libm_explicitly_to_build.patch | 20 --
.../recipes/numpy/patches/ranlib.patch | 11 -
.../numpy/patches/remove-default-paths.patch | 28 --
pythonforandroid/recipes/pandas/__init__.py | 28 +-
.../recipes/pandas/fix_numpy_includes.patch | 111 +++++--
.../recipes/pydantic-core/__init__.py | 2 -
pythonforandroid/recipes/scipy/__init__.py | 1 +
pythonforandroid/recommendations.py | 2 +-
setup.py | 2 +-
testapps/on_device_unit_tests/setup.py | 6 +-
21 files changed, 376 insertions(+), 297 deletions(-)
delete mode 100644 pythonforandroid/recipes/cppy/__init__.py
create mode 100644 pythonforandroid/recipes/matplotlib/skip_macos.patch
delete mode 100644 pythonforandroid/recipes/numpy/patches/add_libm_explicitly_to_build.patch
delete mode 100644 pythonforandroid/recipes/numpy/patches/ranlib.patch
delete mode 100644 pythonforandroid/recipes/numpy/patches/remove-default-paths.patch
diff --git a/Dockerfile b/Dockerfile
index b5b2c597ff..408a0802f7 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -57,6 +57,7 @@ RUN ${RETRY} apt -y update -qq > /dev/null \
ant \
autoconf \
automake \
+ autopoint \
ccache \
cmake \
g++ \
@@ -70,6 +71,7 @@ RUN ${RETRY} apt -y update -qq > /dev/null \
make \
openjdk-17-jdk \
patch \
+ patchelf \
pkg-config \
python3 \
python3-dev \
diff --git a/Makefile b/Makefile
index 19a2379ab2..6e30ee5936 100644
--- a/Makefile
+++ b/Makefile
@@ -55,7 +55,7 @@ testapps-with-scipy/%: virtualenv
. $(ACTIVATE) && cd testapps/on_device_unit_tests/ && \
export LEGACY_NDK=$(ANDROID_NDK_HOME_LEGACY) && \
python setup.py $(ARTIFACT) --$(MODE) --sdk-dir $(ANDROID_SDK_HOME) --ndk-dir $(ANDROID_NDK_HOME) \
- --requirements python3,scipy,kivy \
+ --requirements python3,scipy,kivy \
--arch=armeabi-v7a --arch=arm64-v8a
testapps-webview: testapps-webview/debug/apk testapps-webview/release/aab
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 987a00cddb..81c860f888 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -72,6 +72,7 @@ the following command (re-adapted from the `Dockerfile` we use to perform CI bui
ant \
autoconf \
automake \
+ autopoint \
ccache \
cmake \
g++ \
@@ -85,6 +86,7 @@ the following command (re-adapted from the `Dockerfile` we use to perform CI bui
make \
openjdk-17-jdk \
patch \
+ patchelf \
pkg-config \
python3 \
python3-dev \
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index d70571ad08..c4131c53e6 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1,4 +1,4 @@
-from os.path import basename, dirname, exists, isdir, isfile, join, realpath, split, sep
+from os.path import basename, dirname, exists, isdir, isfile, join, realpath, split
import glob
import hashlib
@@ -12,6 +12,8 @@
from urllib.request import urlretrieve
from os import listdir, unlink, environ, curdir, walk
from sys import stdout
+from wheel.wheelfile import WheelFile
+from wheel.cli.tags import tags as wheel_tags
import time
try:
from urlparse import urlparse
@@ -232,7 +234,7 @@ def report_hook(index, blksize, size):
shprint(sh.git, 'clone', '--recursive', url, target)
with current_directory(target):
if self.version:
- shprint(sh.git, 'fetch', '--depth', '1', 'origin', self.version)
+ shprint(sh.git, 'fetch', '--tags', '--depth', '1')
shprint(sh.git, 'checkout', self.version)
branch = sh.git('branch', '--show-current')
if branch:
@@ -478,10 +480,11 @@ def unpack(self, arch):
elif isdir(extraction_filename):
ensure_dir(directory_name)
for entry in listdir(extraction_filename):
- if entry not in ('.git',):
- shprint(sh.cp, '-Rv',
- join(extraction_filename, entry),
- directory_name)
+ # Previously we filtered out the .git folder, but during the build process for some recipes
+ # (e.g. when version is parsed by `setuptools_scm`) that may be needed.
+ shprint(sh.cp, '-Rv',
+ join(extraction_filename, entry),
+ directory_name)
else:
raise Exception(
'Given path is neither a file nor a directory: {}'
@@ -843,7 +846,6 @@ class PythonRecipe(Recipe):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
-
if 'python3' not in self.depends:
# We ensure here that the recipe depends on python even it overrode
# `depends`. We only do this if it doesn't already depend on any
@@ -893,12 +895,12 @@ def folder_name(self):
def get_recipe_env(self, arch=None, with_flags_in_cc=True):
env = super().get_recipe_env(arch, with_flags_in_cc)
-
env['PYTHONNOUSERSITE'] = '1'
-
# Set the LANG, this isn't usually important but is a better default
# as it occasionally matters how Python e.g. reads files
env['LANG'] = "en_GB.UTF-8"
+ # Binaries made by packages installed by pip
+ env["PATH"] = join(self.hostpython_site_dir, "bin") + ":" + env["PATH"]
if not self.call_hostpython_via_targetpython:
env['CFLAGS'] += ' -I{}'.format(
@@ -979,26 +981,37 @@ def install_hostpython_package(self, arch):
_env=env, *self.setup_extra_args)
@property
- def python_version(self):
- return Recipe.get_recipe("python3", self.ctx).version
+ def python_major_minor_version(self):
+ parsed_version = packaging.version.parse(self.ctx.python_recipe.version)
+ return f"{parsed_version.major}.{parsed_version.minor}"
+
+ def install_hostpython_prerequisites(self, packages=None, force_upgrade=True):
+ if not packages:
+ packages = self.hostpython_prerequisites
- def install_hostpython_prerequisites(self, force_upgrade=True):
- if len(self.hostpython_prerequisites) == 0:
+ if len(packages) == 0:
return
+
pip_options = [
"install",
- *self.hostpython_prerequisites,
+ *packages,
"--target", self.hostpython_site_dir, "--python-version",
- self.python_version,
+ self.ctx.python_recipe.version,
# Don't use sources, instead wheels
"--only-binary=:all:",
- "--no-deps"
]
if force_upgrade:
pip_options.append("--upgrade")
# Use system's pip
shprint(sh.pip, *pip_options)
+ def restore_hostpython_prerequisites(self, packages):
+ _packages = []
+ for package in packages:
+ original_version = Recipe.get_recipe(package, self.ctx).version
+ _packages.append(package + "==" + original_version)
+ self.install_hostpython_prerequisites(packages=_packages)
+
class CompiledComponentsPythonRecipe(PythonRecipe):
pre_build_ext = False
@@ -1010,6 +1023,7 @@ def build_arch(self, arch):
calling setup.py install with the target Python dir.
'''
Recipe.build_arch(self, arch)
+ self.install_hostpython_prerequisites()
self.build_compiled_components(arch)
self.install_python_package(arch)
@@ -1157,7 +1171,182 @@ def get_recipe_env(self, arch, with_flags_in_cc=True):
return env
-class RustCompiledComponentsRecipe(PythonRecipe):
+class PyProjectRecipe(PythonRecipe):
+ '''Recipe for projects which containes `pyproject.toml`'''
+
+ # Extra args to pass to `python -m build ...`
+ extra_build_args = []
+ call_hostpython_via_targetpython = False
+
+ def get_recipe_env(self, arch, **kwargs):
+ # Custom hostpython
+ self.ctx.python_recipe.python_exe = join(
+ self.ctx.python_recipe.get_build_dir(arch), "android-build", "python3")
+ env = super().get_recipe_env(arch, **kwargs)
+ build_dir = self.get_build_dir(arch)
+ ensure_dir(build_dir)
+ build_opts = join(build_dir, "build-opts.cfg")
+
+ with open(build_opts, "w") as file:
+ file.write("[bdist_wheel]\nplat-name={}".format(
+ self.get_wheel_platform_tag(arch)
+ ))
+ file.close()
+
+ env["DIST_EXTRA_CONFIG"] = build_opts
+ return env
+
+ def get_wheel_platform_tag(self, arch):
+ return "android_" + {
+ "armeabi-v7a": "arm",
+ "arm64-v8a": "aarch64",
+ "x86_64": "x86_64",
+ "x86": "i686",
+ }[arch.arch]
+
+ def install_wheel(self, arch, built_wheels):
+ _wheel = built_wheels[0]
+ built_wheel_dir = dirname(_wheel)
+ # Fix wheel platform tag
+ wheel_tag = wheel_tags(
+ _wheel,
+ platform_tags=self.get_wheel_platform_tag(arch),
+ remove=True,
+ )
+ selected_wheel = join(built_wheel_dir, wheel_tag)
+
+ _dev_wheel_dir = environ.get("P4A_WHEEL_DIR", False)
+ if _dev_wheel_dir:
+ ensure_dir(_dev_wheel_dir)
+ shprint(sh.cp, selected_wheel, _dev_wheel_dir)
+
+ info(f"Installing built wheel: {wheel_tag}")
+ destination = self.ctx.get_python_install_dir(arch.arch)
+ with WheelFile(selected_wheel) as wf:
+ for zinfo in wf.filelist:
+ wf.extract(zinfo, destination)
+ wf.close()
+
+ def build_arch(self, arch):
+ self.install_hostpython_prerequisites(
+ packages=["build[virtualenv]", "pip"] + self.hostpython_prerequisites
+ )
+ build_dir = self.get_build_dir(arch.arch)
+ env = self.get_recipe_env(arch, with_flags_in_cc=True)
+ # make build dir separatly
+ sub_build_dir = join(build_dir, "p4a_android_build")
+ ensure_dir(sub_build_dir)
+ # copy hostpython to built python to ensure correct selection of libs and includes
+ shprint(sh.cp, self.real_hostpython_location, self.ctx.python_recipe.python_exe)
+
+ build_args = [
+ "-m",
+ "build",
+ "--wheel",
+ "--config-setting",
+ "builddir={}".format(sub_build_dir),
+ ] + self.extra_build_args
+
+ built_wheels = []
+ with current_directory(build_dir):
+ shprint(
+ sh.Command(self.ctx.python_recipe.python_exe), *build_args, _env=env
+ )
+ built_wheels = [realpath(whl) for whl in glob.glob("dist/*.whl")]
+ self.install_wheel(arch, built_wheels)
+
+
+class MesonRecipe(PyProjectRecipe):
+ '''Recipe for projects which uses meson as build system'''
+
+ meson_version = "1.4.0"
+ ninja_version = "1.11.1.1"
+
+ def sanitize_flags(self, *flag_strings):
+ return " ".join(flag_strings).strip().split(" ")
+
+ def get_recipe_meson_options(self, arch):
+ env = self.get_recipe_env(arch, with_flags_in_cc=True)
+ return {
+ "binaries": {
+ "c": arch.get_clang_exe(with_target=True),
+ "cpp": arch.get_clang_exe(with_target=True, plus_plus=True),
+ "ar": self.ctx.ndk.llvm_ar,
+ "strip": self.ctx.ndk.llvm_strip,
+ },
+ "built-in options": {
+ "c_args": self.sanitize_flags(env["CFLAGS"], env["CPPFLAGS"]),
+ "cpp_args": self.sanitize_flags(env["CXXFLAGS"], env["CPPFLAGS"]),
+ "c_link_args": self.sanitize_flags(env["LDFLAGS"]),
+ "cpp_link_args": self.sanitize_flags(env["LDFLAGS"]),
+ },
+ "properties": {
+ "needs_exe_wrapper": True,
+ "sys_root": self.ctx.ndk.sysroot
+ },
+ "host_machine": {
+ "cpu_family": {
+ "arm64-v8a": "aarch64",
+ "armeabi-v7a": "arm",
+ "x86_64": "x86_64",
+ "x86": "x86"
+ }[arch.arch],
+ "cpu": {
+ "arm64-v8a": "aarch64",
+ "armeabi-v7a": "armv7",
+ "x86_64": "x86_64",
+ "x86": "i686"
+ }[arch.arch],
+ "endian": "little",
+ "system": "android",
+ }
+ }
+
+ def write_build_options(self, arch):
+ """Writes python dict to meson config file"""
+ option_data = ""
+ build_options = self.get_recipe_meson_options(arch)
+ for key in build_options.keys():
+ data_chunk = "[{}]".format(key)
+ for subkey in build_options[key].keys():
+ value = build_options[key][subkey]
+ if isinstance(value, int):
+ value = str(value)
+ elif isinstance(value, str):
+ value = "'{}'".format(value)
+ elif isinstance(value, bool):
+ value = "true" if value else "false"
+ elif isinstance(value, list):
+ value = "['" + "', '".join(value) + "']"
+ data_chunk += "\n" + subkey + " = " + value
+ option_data += data_chunk + "\n\n"
+ return option_data
+
+ def ensure_args(self, *args):
+ for arg in args:
+ if arg not in self.extra_build_args:
+ self.extra_build_args.append(arg)
+
+ def build_arch(self, arch):
+ cross_file = join("/tmp", "android.meson.cross")
+ info("Writing cross file at: {}".format(cross_file))
+ # write cross config file
+ with open(cross_file, "w") as file:
+ file.write(self.write_build_options(arch))
+ file.close()
+ # set cross file
+ self.ensure_args('-Csetup-args=--cross-file', '-Csetup-args={}'.format(cross_file))
+ # ensure ninja and meson
+ for dep in [
+ "ninja=={}".format(self.ninja_version),
+ "meson=={}".format(self.meson_version),
+ ]:
+ if dep not in self.hostpython_prerequisites:
+ self.hostpython_prerequisites.append(dep)
+ super().build_arch(arch)
+
+
+class RustCompiledComponentsRecipe(PyProjectRecipe):
# Rust toolchain codes
# https://doc.rust-lang.org/nightly/rustc/platform-support.html
RUST_ARCH_CODES = {
@@ -1167,41 +1356,10 @@ class RustCompiledComponentsRecipe(PythonRecipe):
"x86": "i686-linux-android",
}
- # Build python wheel using `maturin` instead
- # of default `python -m build [...]`
- use_maturin = False
-
- # Directory where to find built wheel
- # For normal build: "dist/*.whl"
- # For maturin: "target/wheels/*-linux_*.whl"
- built_wheel_pattern = None
-
call_hostpython_via_targetpython = False
- def __init__(self, *arg, **kwargs):
- super().__init__(*arg, **kwargs)
- self.append_deps_if_absent(["python3"])
- self.set_default_hostpython_deps()
- if not self.built_wheel_pattern:
- self.built_wheel_pattern = (
- "target/wheels/*-linux_*.whl"
- if self.use_maturin
- else "dist/*.whl"
- )
-
- def set_default_hostpython_deps(self):
- if not self.use_maturin:
- self.hostpython_prerequisites += ["build", "setuptools_rust", "wheel", "pyproject_hooks"]
- else:
- self.hostpython_prerequisites += ["maturin"]
-
- def append_deps_if_absent(self, deps):
- for dep in deps:
- if dep not in self.depends:
- self.depends.append(dep)
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
# Set rust build target
build_target = self.RUST_ARCH_CODES[arch.arch]
@@ -1220,7 +1378,7 @@ def get_recipe_env(self, arch):
self.ctx.ndk_api,
),
)
- realpython_dir = Recipe.get_recipe("python3", self.ctx).get_build_dir(arch.arch)
+ realpython_dir = self.ctx.python_recipe.get_build_dir(arch.arch)
env["RUSTFLAGS"] = "-Clink-args=-L{} -L{}".format(
self.ctx.get_libs_dir(arch.arch), join(realpython_dir, "android-build")
@@ -1228,7 +1386,7 @@ def get_recipe_env(self, arch):
env["PYO3_CROSS_LIB_DIR"] = realpath(glob.glob(join(
realpython_dir, "android-build", "build",
- "lib.linux-*-{}/".format(self.get_python_formatted_version()),
+ "lib.linux-*-{}/".format(self.python_major_minor_version),
))[0])
info_main("Ensuring rust build toolchain")
@@ -1243,10 +1401,6 @@ def get_recipe_env(self, arch):
)
return env
- def get_python_formatted_version(self):
- parsed_version = packaging.version.parse(self.python_version)
- return f"{parsed_version.major}.{parsed_version.minor}"
-
def check_host_deps(self):
if not hasattr(sh, "rustup"):
error(
@@ -1258,41 +1412,7 @@ def check_host_deps(self):
def build_arch(self, arch):
self.check_host_deps()
- self.install_hostpython_prerequisites()
- build_dir = self.get_build_dir(arch.arch)
- env = self.get_recipe_env(arch)
- built_wheel = None
-
- # Copy the exec with version info
- hostpython_exec = join(
- sep,
- *self.hostpython_location.split(sep)[:-1],
- "python{}".format(self.get_python_formatted_version()),
- )
- shprint(sh.cp, self.hostpython_location, hostpython_exec)
-
- with current_directory(build_dir):
- if self.use_maturin:
- shprint(
- sh.Command(join(self.hostpython_site_dir, "bin", "maturin")),
- "build", "--interpreter", hostpython_exec, "--skip-auditwheel",
- _env=env,
- )
- else:
- shprint(
- sh.Command(hostpython_exec),
- "-m", "build", "--no-isolation", "--skip-dependency-check", "--wheel",
- _env=env,
- )
- # Find the built wheel
- built_wheel = realpath(glob.glob(self.built_wheel_pattern)[0])
-
- info("Unzipping built wheel '{}'".format(basename(built_wheel)))
-
- # Unzip .whl file into site-packages
- with zipfile.ZipFile(built_wheel, "r") as zip_ref:
- zip_ref.extractall(self.ctx.get_python_install_dir(arch.arch))
- info("Successfully installed '{}'".format(basename(built_wheel)))
+ super().build_arch(arch)
class TargetPythonRecipe(Recipe):
diff --git a/pythonforandroid/recipes/Pillow/__init__.py b/pythonforandroid/recipes/Pillow/__init__.py
index f8f6929db5..cd770526ff 100644
--- a/pythonforandroid/recipes/Pillow/__init__.py
+++ b/pythonforandroid/recipes/Pillow/__init__.py
@@ -25,7 +25,7 @@ class PillowRecipe(CompiledComponentsPythonRecipe):
version = '8.4.0'
url = 'https://github.com/python-pillow/Pillow/archive/{version}.tar.gz'
- site_packages_name = 'Pillow'
+ site_packages_name = 'PIL'
depends = ['png', 'jpeg', 'freetype', 'setuptools']
opt_depends = ['libwebp']
patches = [join('patches', 'fix-setup.patch')]
diff --git a/pythonforandroid/recipes/cppy/__init__.py b/pythonforandroid/recipes/cppy/__init__.py
deleted file mode 100644
index f61e2c2516..0000000000
--- a/pythonforandroid/recipes/cppy/__init__.py
+++ /dev/null
@@ -1,14 +0,0 @@
-from pythonforandroid.recipe import PythonRecipe
-
-
-class CppyRecipe(PythonRecipe):
- site_packages_name = 'cppy'
- version = '1.1.0'
- url = 'https://github.com/nucleic/cppy/archive/{version}.zip'
- call_hostpython_via_targetpython = False
- # to be detected by the matplotlib install script
- install_in_hostpython = True
- depends = ['setuptools']
-
-
-recipe = CppyRecipe()
diff --git a/pythonforandroid/recipes/cryptography/__init__.py b/pythonforandroid/recipes/cryptography/__init__.py
index 8e476b3129..c6a91a13d7 100644
--- a/pythonforandroid/recipes/cryptography/__init__.py
+++ b/pythonforandroid/recipes/cryptography/__init__.py
@@ -7,12 +7,10 @@ class CryptographyRecipe(RustCompiledComponentsRecipe):
name = 'cryptography'
version = '42.0.1'
url = 'https://github.com/pyca/cryptography/archive/refs/tags/{version}.tar.gz'
- depends = ['openssl', 'six', 'setuptools', 'cffi']
- # recipe built cffi does not work on apple M1
- hostpython_prerequisites = ["semantic_version", "cffi"]
+ depends = ['openssl']
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
openssl_build_dir = self.get_recipe('openssl', self.ctx).get_build_dir(arch.arch)
build_target = self.RUST_ARCH_CODES[arch.arch].upper().replace("-", "_")
openssl_include = "{}_OPENSSL_INCLUDE_DIR".format(build_target)
diff --git a/pythonforandroid/recipes/kiwisolver/__init__.py b/pythonforandroid/recipes/kiwisolver/__init__.py
index 587c2b9a49..c4c19ac257 100644
--- a/pythonforandroid/recipes/kiwisolver/__init__.py
+++ b/pythonforandroid/recipes/kiwisolver/__init__.py
@@ -1,11 +1,12 @@
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class KiwiSolverRecipe(CppCompiledComponentsPythonRecipe):
+class KiwiSolverRecipe(PyProjectRecipe):
site_packages_name = 'kiwisolver'
- version = '1.3.2'
- url = 'https://github.com/nucleic/kiwi/archive/{version}.zip'
+ version = '1.4.5'
+ url = 'git+https://github.com/nucleic/kiwi'
depends = ['cppy']
+ need_stl_shared = True
recipe = KiwiSolverRecipe()
diff --git a/pythonforandroid/recipes/matplotlib/__init__.py b/pythonforandroid/recipes/matplotlib/__init__.py
index f79cde3483..6fffd45596 100644
--- a/pythonforandroid/recipes/matplotlib/__init__.py
+++ b/pythonforandroid/recipes/matplotlib/__init__.py
@@ -1,18 +1,17 @@
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
from pythonforandroid.util import ensure_dir
from os.path import join
import shutil
-class MatplotlibRecipe(CppCompiledComponentsPythonRecipe):
-
- version = '3.5.2'
+class MatplotlibRecipe(PyProjectRecipe):
+ version = '3.8.4'
url = 'https://github.com/matplotlib/matplotlib/archive/v{version}.zip'
-
+ patches = ["skip_macos.patch"]
depends = ['kiwisolver', 'numpy', 'pillow', 'setuptools', 'freetype']
-
python_depends = ['cycler', 'fonttools', 'packaging', 'pyparsing', 'python-dateutil']
+ need_stl_shared = True
def generate_libraries_pc_files(self, arch):
"""
@@ -62,8 +61,8 @@ def prebuild_arch(self, arch):
)
self.generate_libraries_pc_files(arch)
- def get_recipe_env(self, arch=None, with_flags_in_cc=True):
- env = super().get_recipe_env(arch, with_flags_in_cc)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
# we make use of the same directory than `XDG_CACHE_HOME`, for our
# custom library pc files, so we have all the install files that we
diff --git a/pythonforandroid/recipes/matplotlib/skip_macos.patch b/pythonforandroid/recipes/matplotlib/skip_macos.patch
new file mode 100644
index 0000000000..7652750769
--- /dev/null
+++ b/pythonforandroid/recipes/matplotlib/skip_macos.patch
@@ -0,0 +1,12 @@
+diff '--color=auto' -uNr matplotlib-3.8.4/setupext.py matplotlib-3.8.4.mod/setupext.py
+--- matplotlib-3.8.4/setupext.py 2024-04-04 04:06:51.000000000 +0530
++++ matplotlib-3.8.4.mod/setupext.py 2024-04-30 19:31:39.608063438 +0530
+@@ -782,7 +782,7 @@
+ name = 'macosx'
+
+ def check(self):
+- if sys.platform != 'darwin':
++ if True: #sys.platform != 'darwin':
+ raise Skipped("Mac OS-X only")
+ return super().check()
+
diff --git a/pythonforandroid/recipes/numpy/__init__.py b/pythonforandroid/recipes/numpy/__init__.py
index 7e51eba66b..fb34c0c9f7 100644
--- a/pythonforandroid/recipes/numpy/__init__.py
+++ b/pythonforandroid/recipes/numpy/__init__.py
@@ -1,34 +1,29 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
-from pythonforandroid.logger import shprint, info
-from pythonforandroid.util import current_directory
-from multiprocessing import cpu_count
+from pythonforandroid.recipe import Recipe, MesonRecipe
+from pythonforandroid.logger import error
from os.path import join
-import glob
-import sh
import shutil
+NUMPY_NDK_MESSAGE = "In order to build numpy, you must set minimum ndk api (minapi) to `24`.\n"
-class NumpyRecipe(CompiledComponentsPythonRecipe):
- version = '1.22.3'
- url = 'https://pypi.python.org/packages/source/n/numpy/numpy-{version}.zip'
- site_packages_name = 'numpy'
- depends = ["cython"]
+class NumpyRecipe(MesonRecipe):
+ version = 'v1.26.5'
+ url = 'git+https://github.com/numpy/numpy'
+ hostpython_prerequisites = ["Cython>=3.0.6"] # meson does not detects venv's cython
+ extra_build_args = ['-Csetup-args=-Dblas=none', '-Csetup-args=-Dlapack=none']
+ need_stl_shared = True
- # This build specifically requires setuptools version 59.2.0
- hostpython_prerequisites = ["setuptools==59.2.0"]
+ def get_recipe_meson_options(self, arch):
+ options = super().get_recipe_meson_options(arch)
+ # Custom python is required, so that meson
+ # gets libs and config files properly
+ options["binaries"]["python"] = self.ctx.python_recipe.python_exe
+ options["binaries"]["python3"] = self.ctx.python_recipe.python_exe
+ options["properties"]["longdouble_format"] = "IEEE_DOUBLE_LE" if arch.arch in ["armeabi-v7a", "x86"] else "IEEE_QUAD_LE"
+ return options
- install_in_hostpython = True
- call_hostpython_via_targetpython = False
-
- patches = [
- join("patches", "remove-default-paths.patch"),
- join("patches", "add_libm_explicitly_to_build.patch"),
- join("patches", "ranlib.patch"),
- ]
-
- def get_recipe_env(self, arch=None, with_flags_in_cc=True):
- env = super().get_recipe_env(arch, with_flags_in_cc)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
# _PYTHON_HOST_PLATFORM declares that we're cross-compiling
# and avoids issues when building on macOS for Android targets.
@@ -37,54 +32,25 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
# NPY_DISABLE_SVML=1 allows numpy to build for non-AVX512 CPUs
# See: https://github.com/numpy/numpy/issues/21196
env["NPY_DISABLE_SVML"] = "1"
-
+ env["TARGET_PYTHON_EXE"] = join(Recipe.get_recipe(
+ "python3", self.ctx).get_build_dir(arch.arch), "android-build", "python")
return env
- def build_arch(self, arch):
- self.hostpython_prerequisites = ["setuptools==59.2.0"]
- self.install_hostpython_prerequisites()
+ def download_if_necessary(self):
+ # NumPy requires complex math functions which were added in api 24
+ if self.ctx.ndk_api < 24:
+ error(NUMPY_NDK_MESSAGE)
+ exit(1)
+ super().download_if_necessary()
+ def build_arch(self, arch):
super().build_arch(arch)
-
- # Post build step to restore setuptools version
- self.hostpython_prerequisites = ["setuptools=={}".format(
- Recipe.get_recipe("setuptools", self.ctx).version)
- ]
- self.install_hostpython_prerequisites()
-
- def _build_compiled_components(self, arch):
- info('Building compiled components in {}'.format(self.name))
-
- env = self.get_recipe_env(arch)
- with current_directory(self.get_build_dir(arch.arch)):
- hostpython = sh.Command(self.hostpython_location)
- shprint(hostpython, 'setup.py', self.build_cmd, '-v',
- _env=env, *self.setup_extra_args)
- build_dir = glob.glob('build/lib.*')[0]
- shprint(sh.find, build_dir, '-name', '"*.o"', '-exec',
- env['STRIP'], '{}', ';', _env=env)
-
- def _rebuild_compiled_components(self, arch, env):
- info('Rebuilding compiled components in {}'.format(self.name))
-
- hostpython = sh.Command(self.real_hostpython_location)
- shprint(hostpython, 'setup.py', 'clean', '--all', '--force', _env=env)
- shprint(hostpython, 'setup.py', self.build_cmd, '-v', _env=env,
- *self.setup_extra_args)
-
- def build_compiled_components(self, arch):
- self.setup_extra_args = ['-j', str(cpu_count())]
- self._build_compiled_components(arch)
- self.setup_extra_args = []
-
- def rebuild_compiled_components(self, arch, env):
- self.setup_extra_args = ['-j', str(cpu_count())]
- self._rebuild_compiled_components(arch, env)
- self.setup_extra_args = []
+ self.restore_hostpython_prerequisites(["cython"])
def get_hostrecipe_env(self, arch):
env = super().get_hostrecipe_env(arch)
env['RANLIB'] = shutil.which('ranlib')
+ env["LDFLAGS"] += " -lm"
return env
diff --git a/pythonforandroid/recipes/numpy/patches/add_libm_explicitly_to_build.patch b/pythonforandroid/recipes/numpy/patches/add_libm_explicitly_to_build.patch
deleted file mode 100644
index f9ba9e924e..0000000000
--- a/pythonforandroid/recipes/numpy/patches/add_libm_explicitly_to_build.patch
+++ /dev/null
@@ -1,20 +0,0 @@
-diff --git a/numpy/linalg/setup.py b/numpy/linalg/setup.py
-index 66c07c9..d34bd93 100644
---- a/numpy/linalg/setup.py
-+++ b/numpy/linalg/setup.py
-@@ -46,6 +46,7 @@ def configuration(parent_package='', top_path=None):
- sources=['lapack_litemodule.c', get_lapack_lite_sources],
- depends=['lapack_lite/f2c.h'],
- extra_info=lapack_info,
-+ libraries=['m'],
- )
-
- # umath_linalg module
-@@ -54,7 +54,7 @@ def configuration(parent_package='', top_path=None):
- sources=['umath_linalg.c.src', get_lapack_lite_sources],
- depends=['lapack_lite/f2c.h'],
- extra_info=lapack_info,
-- libraries=['npymath'],
-+ libraries=['npymath', 'm'],
- )
- return config
diff --git a/pythonforandroid/recipes/numpy/patches/ranlib.patch b/pythonforandroid/recipes/numpy/patches/ranlib.patch
deleted file mode 100644
index c0b5dad6b4..0000000000
--- a/pythonforandroid/recipes/numpy/patches/ranlib.patch
+++ /dev/null
@@ -1,11 +0,0 @@
-diff -Naur numpy.orig/numpy/distutils/unixccompiler.py numpy/numpy/distutils/unixccompiler.py
---- numpy.orig/numpy/distutils/unixccompiler.py 2022-05-28 10:22:10.000000000 +0200
-+++ numpy/numpy/distutils/unixccompiler.py 2022-05-28 10:22:24.000000000 +0200
-@@ -124,6 +124,7 @@
- # platform intelligence here to skip ranlib if it's not
- # needed -- or maybe Python's configure script took care of
- # it for us, hence the check for leading colon.
-+ self.ranlib = [os.environ.get('RANLIB')]
- if self.ranlib:
- display = '%s:@ %s' % (os.path.basename(self.ranlib[0]),
- output_filename)
diff --git a/pythonforandroid/recipes/numpy/patches/remove-default-paths.patch b/pythonforandroid/recipes/numpy/patches/remove-default-paths.patch
deleted file mode 100644
index 3581f0f9ed..0000000000
--- a/pythonforandroid/recipes/numpy/patches/remove-default-paths.patch
+++ /dev/null
@@ -1,28 +0,0 @@
-diff --git a/numpy/distutils/system_info.py b/numpy/distutils/system_info.py
-index fc7018a..7b514bc 100644
---- a/numpy/distutils/system_info.py
-+++ b/numpy/distutils/system_info.py
-@@ -340,10 +340,10 @@ if os.path.join(sys.prefix, 'lib') not in default_lib_dirs:
- default_include_dirs.append(os.path.join(sys.prefix, 'include'))
- default_src_dirs.append(os.path.join(sys.prefix, 'src'))
-
--default_lib_dirs = [_m for _m in default_lib_dirs if os.path.isdir(_m)]
--default_runtime_dirs = [_m for _m in default_runtime_dirs if os.path.isdir(_m)]
--default_include_dirs = [_m for _m in default_include_dirs if os.path.isdir(_m)]
--default_src_dirs = [_m for _m in default_src_dirs if os.path.isdir(_m)]
-+default_lib_dirs = [] #[_m for _m in default_lib_dirs if os.path.isdir(_m)]
-+default_runtime_dirs =[] # [_m for _m in default_runtime_dirs if os.path.isdir(_m)]
-+default_include_dirs =[] # [_m for _m in default_include_dirs if os.path.isdir(_m)]
-+default_src_dirs =[] # [_m for _m in default_src_dirs if os.path.isdir(_m)]
-
- so_ext = get_shared_lib_extension()
-
-@@ -814,7 +814,7 @@ class system_info(object):
- path = self.get_paths(self.section, key)
- if path == ['']:
- path = []
-- return path
-+ return []
-
- def get_include_dirs(self, key='include_dirs'):
- return self.get_paths(self.section, key)
diff --git a/pythonforandroid/recipes/pandas/__init__.py b/pythonforandroid/recipes/pandas/__init__.py
index a43209a339..3f56adef6c 100644
--- a/pythonforandroid/recipes/pandas/__init__.py
+++ b/pythonforandroid/recipes/pandas/__init__.py
@@ -1,27 +1,25 @@
from os.path import join
+from pythonforandroid.recipe import MesonRecipe
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
-
-class PandasRecipe(CppCompiledComponentsPythonRecipe):
- version = '1.0.3'
- url = 'https://github.com/pandas-dev/pandas/releases/download/v{version}/pandas-{version}.tar.gz' # noqa
-
- depends = ['cython', 'numpy', 'libbz2', 'liblzma']
-
- python_depends = ['python-dateutil', 'pytz']
+class PandasRecipe(MesonRecipe):
+ version = 'v2.2.1'
+ url = 'git+https://github.com/pandas-dev/pandas' # noqa
+ depends = ['numpy', 'libbz2', 'liblzma']
+ hostpython_prerequisites = ["Cython~=3.0.5"] # meson does not detects venv's cython
patches = ['fix_numpy_includes.patch']
-
- call_hostpython_via_targetpython = False
+ python_depends = ['python-dateutil', 'pytz']
need_stl_shared = True
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
# we need the includes from our installed numpy at site packages
# because we need some includes generated at numpy's compile time
+
env['NUMPY_INCLUDES'] = join(
self.ctx.get_python_install_dir(arch.arch), "numpy/core/include",
)
+ env["PYTHON_INCLUDE_DIR"] = self.ctx.python_recipe.include_root(arch)
# this flag below is to fix a runtime error:
# ImportError: dlopen failed: cannot locate symbol
@@ -31,5 +29,9 @@ def get_recipe_env(self, arch):
env['LDFLAGS'] += f' -landroid -l{self.stl_lib_name}'
return env
+ def build_arch(self, arch):
+ super().build_arch(arch)
+ self.restore_hostpython_prerequisites(["cython"])
+
recipe = PandasRecipe()
diff --git a/pythonforandroid/recipes/pandas/fix_numpy_includes.patch b/pythonforandroid/recipes/pandas/fix_numpy_includes.patch
index ef1643b9b1..5c84e6421b 100644
--- a/pythonforandroid/recipes/pandas/fix_numpy_includes.patch
+++ b/pythonforandroid/recipes/pandas/fix_numpy_includes.patch
@@ -1,31 +1,82 @@
---- pandas-1.0.1/setup.py.orig 2020-02-05 17:15:24.000000000 +0100
-+++ pandas-1.0.1/setup.py 2020-03-15 13:47:57.612237225 +0100
-@@ -37,11 +37,12 @@ min_cython_ver = "0.29.13" # note: sync
-
- setuptools_kwargs = {
- "install_requires": [
-- "python-dateutil >= 2.6.1",
-- "pytz >= 2017.2",
-- f"numpy >= {min_numpy_ver}",
-+ # dependencies managed via p4a's recipe
-+ # "python-dateutil >= 2.6.1",
-+ # "pytz >= 2017.2",
-+ # f"numpy >= {min_numpy_ver}",
- ],
-- "setup_requires": [f"numpy >= {min_numpy_ver}"],
-+ "setup_requires": [],
- "zip_safe": False,
- }
-
-@@ -514,7 +515,10 @@ def maybe_cythonize(extensions, *args, *
- )
- raise RuntimeError("Cannot cythonize without Cython installed.")
+diff '--color=auto' -uNr pandas/pandas/_libs/meson.build pandas.mod/pandas/_libs/meson.build
+--- pandas/pandas/_libs/meson.build 2024-04-24 07:24:46.009296003 +0530
++++ pandas.mod/pandas/_libs/meson.build 2024-04-24 07:45:15.221534571 +0530
+@@ -115,7 +115,7 @@
+ ext_name,
+ ext_dict.get('sources'),
+ cython_args: cython_args,
+- include_directories: [inc_np, inc_pd],
++ include_directories: [inc_android, inc_np, inc_pd],
+ dependencies: ext_dict.get('deps', ''),
+ subdir: 'pandas/_libs',
+ install: true
+diff '--color=auto' -uNr pandas/pandas/_libs/tslibs/meson.build pandas.mod/pandas/_libs/tslibs/meson.build
+--- pandas/pandas/_libs/tslibs/meson.build 2024-04-24 07:24:46.019296090 +0530
++++ pandas.mod/pandas/_libs/tslibs/meson.build 2024-04-24 07:45:53.528798309 +0530
+@@ -33,7 +33,7 @@
+ ext_name,
+ ext_dict.get('sources'),
+ cython_args: cython_args,
+- include_directories: [inc_np, inc_pd],
++ include_directories: [inc_android, inc_np, inc_pd],
+ dependencies: ext_dict.get('deps', ''),
+ subdir: 'pandas/_libs/tslibs',
+ install: true
+diff '--color=auto' -uNr pandas/pandas/_libs/window/meson.build pandas.mod/pandas/_libs/window/meson.build
+--- pandas/pandas/_libs/window/meson.build 2024-04-24 07:24:46.029296177 +0530
++++ pandas.mod/pandas/_libs/window/meson.build 2024-04-28 10:47:16.915307381 +0530
+@@ -2,7 +2,7 @@
+ 'aggregations',
+ ['aggregations.pyx'],
+ cython_args: ['-X always_allow_keywords=true'],
+- include_directories: [inc_np, inc_pd],
++ include_directories: [inc_android, inc_np, inc_pd],
+ subdir: 'pandas/_libs/window',
+ override_options : ['cython_language=cpp'],
+ install: true
+@@ -12,7 +12,7 @@
+ 'indexers',
+ ['indexers.pyx'],
+ cython_args: ['-X always_allow_keywords=true'],
+- include_directories: [inc_np, inc_pd],
++ include_directories: [inc_android, inc_np, inc_pd],
+ subdir: 'pandas/_libs/window',
+ install: true
+ )
+diff '--color=auto' -uNr pandas/pandas/meson.build pandas.mod/pandas/meson.build
+--- pandas/pandas/meson.build 2024-04-24 07:24:46.232297943 +0530
++++ pandas.mod/pandas/meson.build 2024-04-24 07:46:12.508929590 +0530
+@@ -3,20 +3,23 @@
+ '-c',
+ '''
+ import os
+-import numpy as np
+-try:
+- # Check if include directory is inside the pandas dir
+- # e.g. a venv created inside the pandas dir
+- # If so, convert it to a relative path
+- incdir = os.path.relpath(np.get_include())
+-except Exception:
+- incdir = np.get_include()
+-print(incdir)
+- '''
++print(os.environ["NUMPY_INCLUDES"])
++ '''
++ ],
++ check: true
++).stdout().strip()
++incdir_android = run_command(py,
++ [
++ '-c',
++ '''
++import os
++print(os.environ["PYTHON_INCLUDE_DIR"])
++ '''
+ ],
+ check: true
+ ).stdout().strip()
+
++inc_android = include_directories(incdir_android)
+ inc_np = include_directories(incdir_numpy)
+ inc_pd = include_directories('_libs/include')
-- numpy_incl = pkg_resources.resource_filename("numpy", "core/include")
-+ if 'NUMPY_INCLUDES' in os.environ:
-+ numpy_incl = os.environ['NUMPY_INCLUDES']
-+ else:
-+ numpy_incl = pkg_resources.resource_filename("numpy", "core/include")
- # TODO: Is this really necessary here?
- for ext in extensions:
- if hasattr(ext, "include_dirs") and numpy_incl not in ext.include_dirs:
diff --git a/pythonforandroid/recipes/pydantic-core/__init__.py b/pythonforandroid/recipes/pydantic-core/__init__.py
index 8702dfc002..bf76a65d0a 100644
--- a/pythonforandroid/recipes/pydantic-core/__init__.py
+++ b/pythonforandroid/recipes/pydantic-core/__init__.py
@@ -4,8 +4,6 @@
class PydanticcoreRecipe(RustCompiledComponentsRecipe):
version = "2.16.1"
url = "https://github.com/pydantic/pydantic-core/archive/refs/tags/v{version}.tar.gz"
- use_maturin = True
- hostpython_prerequisites = ["typing_extensions"]
site_packages_name = "pydantic_core"
diff --git a/pythonforandroid/recipes/scipy/__init__.py b/pythonforandroid/recipes/scipy/__init__.py
index bde9758d8d..242ca04234 100644
--- a/pythonforandroid/recipes/scipy/__init__.py
+++ b/pythonforandroid/recipes/scipy/__init__.py
@@ -19,6 +19,7 @@ class ScipyRecipe(CompiledComponentsPythonRecipe):
url = 'git+https://github.com/scipy/scipy.git'
git_commit = 'b430bf54b5064465983813e2cfef3fcb86c3df07' # version 1.11.3
site_packages_name = 'scipy'
+ hostpython_prerequisites = ['numpy']
depends = ['setuptools', 'cython', 'numpy', 'lapack', 'pybind11']
call_hostpython_via_targetpython = False
need_stl_shared = True
diff --git a/pythonforandroid/recommendations.py b/pythonforandroid/recommendations.py
index cbcfdd2b6e..269a57fcf8 100644
--- a/pythonforandroid/recommendations.py
+++ b/pythonforandroid/recommendations.py
@@ -163,7 +163,7 @@ def check_target_api(api, arch):
MIN_NDK_API = 21
-RECOMMENDED_NDK_API = 21
+RECOMMENDED_NDK_API = 24
OLD_NDK_API_MESSAGE = ('NDK API less than {} is not supported'.format(MIN_NDK_API))
TARGET_NDK_API_GREATER_THAN_TARGET_API_MESSAGE = (
'Target NDK API is {ndk_api}, '
diff --git a/setup.py b/setup.py
index badce08e82..4a628df905 100644
--- a/setup.py
+++ b/setup.py
@@ -22,7 +22,7 @@
install_reqs = [
'appdirs', 'colorama>=0.3.3', 'jinja2',
'sh>=1.10, <2.0; sys_platform!="win32"',
- 'build', 'toml', 'packaging', 'setuptools'
+ 'build', 'toml', 'packaging', 'setuptools', 'wheel~=0.43.0'
]
# (build and toml are used by pythonpackage.py)
diff --git a/testapps/on_device_unit_tests/setup.py b/testapps/on_device_unit_tests/setup.py
index 18fd82af75..a63ca8bcb7 100644
--- a/testapps/on_device_unit_tests/setup.py
+++ b/testapps/on_device_unit_tests/setup.py
@@ -43,7 +43,7 @@
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
'android-api': 27,
- 'ndk-api': 21,
+ 'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
'bootstrap' : 'sdl2',
@@ -57,7 +57,7 @@
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
'android-api': 27,
- 'ndk-api': 21,
+ 'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
'bootstrap' : 'sdl2',
@@ -69,7 +69,7 @@
{
'requirements' : 'python3',
'android-api': 27,
- 'ndk-api': 21,
+ 'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'arm64-v8a',
'bootstrap' : 'service_library',
From 32652a4195d1bbb49eb28f47bf7be43520a37eee Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Wed, 22 May 2024 11:40:55 +0530
Subject: [PATCH 039/158] recipes: add new `pyreqwest_impersonate` recipe
---
.../recipes/pyreqwest_impersonate/__init__.py | 33 +++++++++++++++++++
1 file changed, 33 insertions(+)
create mode 100644 pythonforandroid/recipes/pyreqwest_impersonate/__init__.py
diff --git a/pythonforandroid/recipes/pyreqwest_impersonate/__init__.py b/pythonforandroid/recipes/pyreqwest_impersonate/__init__.py
new file mode 100644
index 0000000000..7e8d5db9ae
--- /dev/null
+++ b/pythonforandroid/recipes/pyreqwest_impersonate/__init__.py
@@ -0,0 +1,33 @@
+from pythonforandroid.logger import info
+from pythonforandroid.recipe import RustCompiledComponentsRecipe
+
+
+class Pyreqwest_impersonateRecipe(RustCompiledComponentsRecipe):
+ version = "v0.4.5"
+ url = "https://github.com/deedy5/pyreqwest_impersonate/archive/refs/tags/{version}.tar.gz"
+
+ def get_recipe_env_post(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env["ANDROID_NDK_HOME"] = self.ctx.ndk.llvm_prebuilt_dir
+ return env
+
+ def get_recipe_env_pre(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env["ANDROID_NDK_HOME"] = self.ctx.ndk_dir
+ return env
+
+ def build_arch(self, arch):
+ # Why need of two env?
+ # Because there are two dependencies which accepts
+ # different ANDROID_NDK_HOME
+ self.get_recipe_env = self.get_recipe_env_pre
+ prebuild_ = super().build_arch
+ try:
+ prebuild_(arch)
+ except Exception:
+ info("pyreqwest_impersonate first build failed, as expected")
+ self.get_recipe_env = self.get_recipe_env_post
+ prebuild_(arch)
+
+
+recipe = Pyreqwest_impersonateRecipe()
From 1461b610dbffd96db42a45906c3c8fd9f0d2a294 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Thu, 28 Mar 2024 16:55:04 +0530
Subject: [PATCH 040/158] recipes: add new `uvloop` recipe
---
.../recipes/libpthread/__init__.py | 51 +++++++++++++++++++
pythonforandroid/recipes/uvloop/__init__.py | 16 ++++++
2 files changed, 67 insertions(+)
create mode 100644 pythonforandroid/recipes/libpthread/__init__.py
create mode 100644 pythonforandroid/recipes/uvloop/__init__.py
diff --git a/pythonforandroid/recipes/libpthread/__init__.py b/pythonforandroid/recipes/libpthread/__init__.py
new file mode 100644
index 0000000000..10feca475a
--- /dev/null
+++ b/pythonforandroid/recipes/libpthread/__init__.py
@@ -0,0 +1,51 @@
+from os import makedirs, remove
+from os.path import exists, join
+import sh
+
+from pythonforandroid.recipe import Recipe
+from pythonforandroid.logger import shprint
+
+
+class LibPthread(Recipe):
+ '''
+ This is a dumb recipe. We may need this because some recipes inserted some
+ flags `-lpthread` without our control, case of:
+
+ - :class:`~pythonforandroid.recipes.uvloop.UvloopRecipe`
+
+ .. note:: the libpthread doesn't exist in android but it is integrated into
+ libc, so we create a symbolic link which we will remove when our build
+ finishes'''
+
+ def build_arch(self, arch):
+ libc_path = join(arch.ndk_lib_dir_versioned, 'libc')
+ # Create a temporary folder to add to link path with a fake libpthread.so:
+ fake_libpthread_temp_folder = join(
+ self.get_build_dir(arch.arch),
+ "p4a-libpthread-recipe-tempdir"
+ )
+ if not exists(fake_libpthread_temp_folder):
+ makedirs(fake_libpthread_temp_folder)
+
+ # Set symlinks, and make sure to update them on every build run:
+ if exists(join(fake_libpthread_temp_folder, "libpthread.so")):
+ remove(join(fake_libpthread_temp_folder, "libpthread.so"))
+ shprint(sh.ln, '-sf',
+ libc_path + '.so',
+ join(fake_libpthread_temp_folder, "libpthread.so"),
+ )
+ if exists(join(fake_libpthread_temp_folder, "libpthread.a")):
+ remove(join(fake_libpthread_temp_folder, "libpthread.a"))
+ shprint(sh.ln, '-sf',
+ libc_path + '.a',
+ join(fake_libpthread_temp_folder, "libpthread.a"),
+ )
+
+ # Add folder as -L link option for all recipes if not done yet:
+ if fake_libpthread_temp_folder not in arch.extra_global_link_paths:
+ arch.extra_global_link_paths.append(
+ fake_libpthread_temp_folder
+ )
+
+
+recipe = LibPthread()
diff --git a/pythonforandroid/recipes/uvloop/__init__.py b/pythonforandroid/recipes/uvloop/__init__.py
new file mode 100644
index 0000000000..e060739026
--- /dev/null
+++ b/pythonforandroid/recipes/uvloop/__init__.py
@@ -0,0 +1,16 @@
+from pythonforandroid.recipe import PyProjectRecipe
+
+
+class UvloopRecipe(PyProjectRecipe):
+ version = 'v0.19.0'
+ url = 'git+https://github.com/MagicStack/uvloop'
+ depends = ['librt', 'libpthread']
+
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env["LIBUV_CONFIGURE_HOST"] = arch.command_prefix
+ env["PLATFORM"] = "android"
+ return env
+
+
+recipe = UvloopRecipe()
From 3d69b695e76f04ca9e9cedda4f9414e201d43983 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Fri, 24 May 2024 22:14:48 +0530
Subject: [PATCH 041/158] pillow: update to `10.3.0`
---
pythonforandroid/recipes/Pillow/__init__.py | 51 ++---
.../recipes/Pillow/patches/fix-setup.patch | 196 ------------------
.../recipes/Pillow/setup.py.patch | 76 +++++++
3 files changed, 89 insertions(+), 234 deletions(-)
delete mode 100644 pythonforandroid/recipes/Pillow/patches/fix-setup.patch
create mode 100644 pythonforandroid/recipes/Pillow/setup.py.patch
diff --git a/pythonforandroid/recipes/Pillow/__init__.py b/pythonforandroid/recipes/Pillow/__init__.py
index cd770526ff..ffac810a2b 100644
--- a/pythonforandroid/recipes/Pillow/__init__.py
+++ b/pythonforandroid/recipes/Pillow/__init__.py
@@ -1,9 +1,9 @@
from os.path import join
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class PillowRecipe(CompiledComponentsPythonRecipe):
+class PillowRecipe(PyProjectRecipe):
"""
A recipe for Pillow (previously known as Pil).
@@ -23,67 +23,42 @@ class PillowRecipe(CompiledComponentsPythonRecipe):
- libwebp: library to encode and decode images in WebP format.
"""
- version = '8.4.0'
+ version = '10.3.0'
url = 'https://github.com/python-pillow/Pillow/archive/{version}.tar.gz'
site_packages_name = 'PIL'
+ patches = ["setup.py.patch"]
depends = ['png', 'jpeg', 'freetype', 'setuptools']
opt_depends = ['libwebp']
- patches = [join('patches', 'fix-setup.patch')]
- call_hostpython_via_targetpython = False
-
- def get_recipe_env(self, arch=None, with_flags_in_cc=True):
- env = super().get_recipe_env(arch, with_flags_in_cc)
-
- png = self.get_recipe('png', self.ctx)
- png_lib_dir = join(png.get_build_dir(arch.arch), '.libs')
- png_inc_dir = png.get_build_dir(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
jpeg = self.get_recipe('jpeg', self.ctx)
jpeg_inc_dir = jpeg_lib_dir = jpeg.get_build_dir(arch.arch)
+ env["JPEG_ROOT"] = "{}:{}".format(jpeg_lib_dir, jpeg_inc_dir)
freetype = self.get_recipe('freetype', self.ctx)
free_lib_dir = join(freetype.get_build_dir(arch.arch), 'objs', '.libs')
free_inc_dir = join(freetype.get_build_dir(arch.arch), 'include')
+ env["FREETYPE_ROOT"] = "{}:{}".format(free_lib_dir, free_inc_dir)
# harfbuzz is a direct dependency of freetype and we need the proper
# flags to successfully build the Pillow recipe, so we add them here.
harfbuzz = self.get_recipe('harfbuzz', self.ctx)
harf_lib_dir = join(harfbuzz.get_build_dir(arch.arch), 'src', '.libs')
harf_inc_dir = harfbuzz.get_build_dir(arch.arch)
+ env["HARFBUZZ_ROOT"] = "{}:{}".format(harf_lib_dir, harf_inc_dir)
+
+ env["ZLIB_ROOT"] = f"{arch.ndk_lib_dir_versioned}:{self.ctx.ndk.sysroot_include_dir}"
# libwebp is an optional dependency, so we add the
# flags if we have it in our `ctx.recipe_build_order`
- build_with_webp_support = 'libwebp' in self.ctx.recipe_build_order
- if build_with_webp_support:
+ if 'libwebp' in self.ctx.recipe_build_order:
webp = self.get_recipe('libwebp', self.ctx)
webp_install = join(
webp.get_build_dir(arch.arch), 'installation'
)
-
- # Add libraries includes to CFLAGS
- cflags = f' -I{png_inc_dir}'
- cflags += f' -I{harf_inc_dir} -I{join(harf_inc_dir, "src")}'
- cflags += f' -I{free_inc_dir}'
- cflags += f' -I{jpeg_inc_dir}'
- if build_with_webp_support:
- cflags += f' -I{join(webp_install, "include")}'
- cflags += f' -I{self.ctx.ndk.sysroot_include_dir}'
-
- # Link the basic Pillow libraries...no need to add webp's libraries
- # since it seems that the linkage is properly made without it :)
- env['LIBS'] = ' -lpng -lfreetype -lharfbuzz -ljpeg -lturbojpeg -lm'
-
- # Add libraries locations to LDFLAGS
- env['LDFLAGS'] += f' -L{png_lib_dir}'
- env['LDFLAGS'] += f' -L{free_lib_dir}'
- env['LDFLAGS'] += f' -L{harf_lib_dir}'
- env['LDFLAGS'] += f' -L{jpeg_lib_dir}'
- if build_with_webp_support:
- env['LDFLAGS'] += f' -L{join(webp_install, "lib")}'
- env['LDFLAGS'] += f' -L{arch.ndk_lib_dir_versioned}'
- if cflags not in env['CFLAGS']:
- env['CFLAGS'] += cflags + " -lm"
+ env["WEBP_ROOT"] = f"{join(webp_install, 'lib')}:{join(webp_install, 'include')}"
return env
diff --git a/pythonforandroid/recipes/Pillow/patches/fix-setup.patch b/pythonforandroid/recipes/Pillow/patches/fix-setup.patch
deleted file mode 100644
index 5c5a3d0536..0000000000
--- a/pythonforandroid/recipes/Pillow/patches/fix-setup.patch
+++ /dev/null
@@ -1,196 +0,0 @@
---- Pillow.orig/setup.py 2021-11-01 14:50:48.000000000 +0100
-+++ Pillow/setup.py 2021-11-01 14:51:31.000000000 +0100
-@@ -125,7 +125,7 @@
- "codec_fd",
- )
-
--DEBUG = False
-+DEBUG = True # So we can easely triage user issues.
-
-
- class DependencyException(Exception):
-@@ -411,46 +411,6 @@
- include_dirs = []
-
- pkg_config = None
-- if _cmd_exists(os.environ.get("PKG_CONFIG", "pkg-config")):
-- pkg_config = _pkg_config
--
-- #
-- # add configured kits
-- for root_name, lib_name in dict(
-- JPEG_ROOT="libjpeg",
-- JPEG2K_ROOT="libopenjp2",
-- TIFF_ROOT=("libtiff-5", "libtiff-4"),
-- ZLIB_ROOT="zlib",
-- FREETYPE_ROOT="freetype2",
-- HARFBUZZ_ROOT="harfbuzz",
-- FRIBIDI_ROOT="fribidi",
-- LCMS_ROOT="lcms2",
-- IMAGEQUANT_ROOT="libimagequant",
-- ).items():
-- root = globals()[root_name]
--
-- if root is None and root_name in os.environ:
-- prefix = os.environ[root_name]
-- root = (os.path.join(prefix, "lib"), os.path.join(prefix, "include"))
--
-- if root is None and pkg_config:
-- if isinstance(lib_name, tuple):
-- for lib_name2 in lib_name:
-- _dbg(f"Looking for `{lib_name2}` using pkg-config.")
-- root = pkg_config(lib_name2)
-- if root:
-- break
-- else:
-- _dbg(f"Looking for `{lib_name}` using pkg-config.")
-- root = pkg_config(lib_name)
--
-- if isinstance(root, tuple):
-- lib_root, include_root = root
-- else:
-- lib_root = include_root = root
--
-- _add_directory(library_dirs, lib_root)
-- _add_directory(include_dirs, include_root)
-
- # respect CFLAGS/CPPFLAGS/LDFLAGS
- for k in ("CFLAGS", "CPPFLAGS", "LDFLAGS"):
-@@ -471,137 +431,6 @@
- for d in os.environ[k].split(os.path.pathsep):
- _add_directory(library_dirs, d)
-
-- _add_directory(library_dirs, os.path.join(sys.prefix, "lib"))
-- _add_directory(include_dirs, os.path.join(sys.prefix, "include"))
--
-- #
-- # add platform directories
--
-- if self.disable_platform_guessing:
-- pass
--
-- elif sys.platform == "cygwin":
-- # pythonX.Y.dll.a is in the /usr/lib/pythonX.Y/config directory
-- _add_directory(
-- library_dirs,
-- os.path.join(
-- "/usr/lib", "python{}.{}".format(*sys.version_info), "config"
-- ),
-- )
--
-- elif sys.platform == "darwin":
-- # attempt to make sure we pick freetype2 over other versions
-- _add_directory(include_dirs, "/sw/include/freetype2")
-- _add_directory(include_dirs, "/sw/lib/freetype2/include")
-- # fink installation directories
-- _add_directory(library_dirs, "/sw/lib")
-- _add_directory(include_dirs, "/sw/include")
-- # darwin ports installation directories
-- _add_directory(library_dirs, "/opt/local/lib")
-- _add_directory(include_dirs, "/opt/local/include")
--
-- # if Homebrew is installed, use its lib and include directories
-- try:
-- prefix = (
-- subprocess.check_output(["brew", "--prefix"])
-- .strip()
-- .decode("latin1")
-- )
-- except Exception:
-- # Homebrew not installed
-- prefix = None
--
-- ft_prefix = None
--
-- if prefix:
-- # add Homebrew's include and lib directories
-- _add_directory(library_dirs, os.path.join(prefix, "lib"))
-- _add_directory(include_dirs, os.path.join(prefix, "include"))
-- _add_directory(
-- include_dirs, os.path.join(prefix, "opt", "zlib", "include")
-- )
-- ft_prefix = os.path.join(prefix, "opt", "freetype")
--
-- if ft_prefix and os.path.isdir(ft_prefix):
-- # freetype might not be linked into Homebrew's prefix
-- _add_directory(library_dirs, os.path.join(ft_prefix, "lib"))
-- _add_directory(include_dirs, os.path.join(ft_prefix, "include"))
-- else:
-- # fall back to freetype from XQuartz if
-- # Homebrew's freetype is missing
-- _add_directory(library_dirs, "/usr/X11/lib")
-- _add_directory(include_dirs, "/usr/X11/include")
--
-- # SDK install path
-- sdk_path = "/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk"
-- if not os.path.exists(sdk_path):
-- try:
-- sdk_path = (
-- subprocess.check_output(["xcrun", "--show-sdk-path"])
-- .strip()
-- .decode("latin1")
-- )
-- except Exception:
-- sdk_path = None
-- if sdk_path:
-- _add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
-- _add_directory(include_dirs, os.path.join(sdk_path, "usr", "include"))
-- elif (
-- sys.platform.startswith("linux")
-- or sys.platform.startswith("gnu")
-- or sys.platform.startswith("freebsd")
-- ):
-- for dirname in _find_library_dirs_ldconfig():
-- _add_directory(library_dirs, dirname)
-- if sys.platform.startswith("linux") and os.environ.get(
-- "ANDROID_ROOT", None
-- ):
-- # termux support for android.
-- # system libraries (zlib) are installed in /system/lib
-- # headers are at $PREFIX/include
-- # user libs are at $PREFIX/lib
-- _add_directory(
-- library_dirs, os.path.join(os.environ["ANDROID_ROOT"], "lib")
-- )
--
-- elif sys.platform.startswith("netbsd"):
-- _add_directory(library_dirs, "/usr/pkg/lib")
-- _add_directory(include_dirs, "/usr/pkg/include")
--
-- elif sys.platform.startswith("sunos5"):
-- _add_directory(library_dirs, "/opt/local/lib")
-- _add_directory(include_dirs, "/opt/local/include")
--
-- # FIXME: check /opt/stuff directories here?
--
-- # standard locations
-- if not self.disable_platform_guessing:
-- _add_directory(library_dirs, "/usr/local/lib")
-- _add_directory(include_dirs, "/usr/local/include")
--
-- _add_directory(library_dirs, "/usr/lib")
-- _add_directory(include_dirs, "/usr/include")
-- # alpine, at least
-- _add_directory(library_dirs, "/lib")
--
-- if sys.platform == "win32":
-- # on Windows, look for the OpenJPEG libraries in the location that
-- # the official installer puts them
-- program_files = os.environ.get("ProgramFiles", "")
-- best_version = (0, 0)
-- best_path = None
-- for name in os.listdir(program_files):
-- if name.startswith("OpenJPEG "):
-- version = tuple(int(x) for x in name[9:].strip().split("."))
-- if version > best_version:
-- best_version = version
-- best_path = os.path.join(program_files, name)
--
-- if best_path:
-- _dbg("Adding %s to search list", best_path)
-- _add_directory(library_dirs, os.path.join(best_path, "lib"))
-- _add_directory(include_dirs, os.path.join(best_path, "include"))
--
- #
- # insert new dirs *before* default libs, to avoid conflicts
- # between Python PYD stub libs and real libraries
\ No newline at end of file
diff --git a/pythonforandroid/recipes/Pillow/setup.py.patch b/pythonforandroid/recipes/Pillow/setup.py.patch
new file mode 100644
index 0000000000..2970e23670
--- /dev/null
+++ b/pythonforandroid/recipes/Pillow/setup.py.patch
@@ -0,0 +1,76 @@
+--- Pillow/setup.py 2024-05-24 19:35:08.270160608 +0530
++++ Pillow.mod/setup.py 2024-05-24 22:07:52.741495666 +0530
+@@ -39,6 +39,7 @@
+ LCMS_ROOT = None
+ TIFF_ROOT = None
+ ZLIB_ROOT = None
++WEBP_ROOT = None
+ FUZZING_BUILD = "LIB_FUZZING_ENGINE" in os.environ
+
+ if sys.platform == "win32" and sys.version_info >= (3, 13):
+@@ -150,6 +151,7 @@
+
+
+ def _find_library_dirs_ldconfig():
++ return []
+ # Based on ctypes.util from Python 2
+
+ ldconfig = "ldconfig" if shutil.which("ldconfig") else "/sbin/ldconfig"
+@@ -460,15 +462,16 @@
+ "HARFBUZZ_ROOT": "harfbuzz",
+ "FRIBIDI_ROOT": "fribidi",
+ "LCMS_ROOT": "lcms2",
++ "WEBP_ROOT": "libwebp",
+ "IMAGEQUANT_ROOT": "libimagequant",
+ }.items():
+ root = globals()[root_name]
+
+ if root is None and root_name in os.environ:
+- prefix = os.environ[root_name]
+- root = (os.path.join(prefix, "lib"), os.path.join(prefix, "include"))
++ root = tuple(os.environ[root_name].split(":"))
+
+ if root is None and pkg_config:
++ continue
+ if isinstance(lib_name, tuple):
+ for lib_name2 in lib_name:
+ _dbg(f"Looking for `{lib_name2}` using pkg-config.")
+@@ -495,14 +498,6 @@
+ for include_dir in include_root:
+ _add_directory(include_dirs, include_dir)
+
+- # respect CFLAGS/CPPFLAGS/LDFLAGS
+- for k in ("CFLAGS", "CPPFLAGS", "LDFLAGS"):
+- if k in os.environ:
+- for match in re.finditer(r"-I([^\s]+)", os.environ[k]):
+- _add_directory(include_dirs, match.group(1))
+- for match in re.finditer(r"-L([^\s]+)", os.environ[k]):
+- _add_directory(library_dirs, match.group(1))
+-
+ # include, rpath, if set as environment variables:
+ for k in ("C_INCLUDE_PATH", "CPATH", "INCLUDE"):
+ if k in os.environ:
+@@ -514,13 +509,10 @@
+ for d in os.environ[k].split(os.path.pathsep):
+ _add_directory(library_dirs, d)
+
+- _add_directory(library_dirs, os.path.join(sys.prefix, "lib"))
+- _add_directory(include_dirs, os.path.join(sys.prefix, "include"))
+-
+ #
+ # add platform directories
+
+- if self.disable_platform_guessing:
++ if True:
+ pass
+
+ elif sys.platform == "cygwin":
+@@ -614,7 +606,7 @@
+ # FIXME: check /opt/stuff directories here?
+
+ # standard locations
+- if not self.disable_platform_guessing:
++ if False: #not self.disable_platform_guessing:
+ _add_directory(library_dirs, "/usr/local/lib")
+ _add_directory(include_dirs, "/usr/local/include")
+
From f0e68762d6dd991786808b81b7855d1946af5d3c Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Tue, 28 May 2024 12:18:39 +0100
Subject: [PATCH 042/158] Add grpcio recipe
---
pythonforandroid/recipes/grpcio/__init__.py | 46 +++++++++++++++++++
.../comment-getserverbyport-r-args.patch | 17 +++++++
.../grpcio/remove-android-log-write.patch | 36 +++++++++++++++
3 files changed, 99 insertions(+)
create mode 100644 pythonforandroid/recipes/grpcio/__init__.py
create mode 100644 pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
create mode 100644 pythonforandroid/recipes/grpcio/remove-android-log-write.patch
diff --git a/pythonforandroid/recipes/grpcio/__init__.py b/pythonforandroid/recipes/grpcio/__init__.py
new file mode 100644
index 0000000000..4276f52e41
--- /dev/null
+++ b/pythonforandroid/recipes/grpcio/__init__.py
@@ -0,0 +1,46 @@
+import glob
+from logging import info
+
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+from pythonforandroid.util import current_directory
+
+
+class GrpcioRecipe(CppCompiledComponentsPythonRecipe):
+ version = '1.64.0'
+ url = 'https://files.pythonhosted.org/packages/source/g/grpcio/grpcio-{version}.tar.gz'
+ depends = ["setuptools", "librt", "libpthread"]
+ patches = ["comment-getserverbyport-r-args.patch", "remove-android-log-write.patch"]
+
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env['NDKPLATFORM'] = "NOTNONE"
+ env['GRPC_PYTHON_BUILD_WITH_CYTHON'] = '1'
+ env["CFLAGS"] += " -U__ANDROID_API__"
+ env["CFLAGS"] += " -D__ANDROID_API__={}".format(self.ctx.ndk_api)
+
+ # turn off c++11 warning error of "invalid suffix on literal"
+ env["CFLAGS"] += " -Wno-reserved-user-defined-literal"
+ env['PLATFORM'] = 'android'
+ env["LDFLAGS"] += " -llog -landroid"
+ return env
+
+ def build_compiled_components(self, arch):
+ info('Building compiled components in {}'.format(self.name))
+
+ env = self.get_recipe_env(arch)
+ hostpython = sh.Command(self.hostpython_location)
+ with current_directory(self.get_build_dir(arch.arch)):
+ if self.install_in_hostpython:
+ shprint(hostpython, 'setup.py', 'clean', '--all', _env=env)
+ shprint(hostpython, 'setup.py', self.build_cmd, '-v',
+ _env=env, *self.setup_extra_args)
+
+ # grpcio creates a build directory and names it `pyb`
+ build_dir = glob.glob('pyb/lib.*')[0]
+ shprint(sh.find, build_dir, '-name', '"*.o"', '-exec',
+ env['STRIP'], '{}', ';', _env=env)
+
+
+recipe = GrpcioRecipe()
diff --git a/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch b/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
new file mode 100644
index 0000000000..6d64105d70
--- /dev/null
+++ b/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
@@ -0,0 +1,17 @@
+Index: ares_config.h
+IDEA additional info:
+Subsystem: com.intellij.openapi.diff.impl.patch.CharsetEP
+<+>UTF-8
+===================================================================
+diff --git a/third_party/cares/config_android/ares_config.h b/third_party/cares/config_android/ares_config.h
+--- a/third_party/cares/config_android/ares_config.h
++++ b/third_party/cares/config_android/ares_config.h (date 1716777985227)
+@@ -43,7 +43,7 @@
+ #define GETNAMEINFO_TYPE_ARG7 int
+
+ /* Specifies the number of arguments to getservbyport_r */
+-#define GETSERVBYPORT_R_ARGS
++//#define GETSERVBYPORT_R_ARGS
+
+ /* Define to 1 if you have AF_INET6. */
+ #define HAVE_AF_INET6
diff --git a/pythonforandroid/recipes/grpcio/remove-android-log-write.patch b/pythonforandroid/recipes/grpcio/remove-android-log-write.patch
new file mode 100644
index 0000000000..b032c65a07
--- /dev/null
+++ b/pythonforandroid/recipes/grpcio/remove-android-log-write.patch
@@ -0,0 +1,36 @@
+Index: log.cc
+IDEA additional info:
+Subsystem: com.intellij.openapi.diff.impl.patch.CharsetEP
+<+>UTF-8
+===================================================================
+diff --git a/src/core/lib/gpr/android/log.cc b/src/core/lib/gpr/android/log.cc
+--- a/src/core/lib/gpr/android/log.cc
++++ b/src/core/lib/gpr/android/log.cc (date 1716778822204)
+@@ -30,18 +30,6 @@
+
+ #include "src/core/lib/gprpp/crash.h"
+
+-static android_LogPriority severity_to_log_priority(gpr_log_severity severity) {
+- switch (severity) {
+- case GPR_LOG_SEVERITY_DEBUG:
+- return ANDROID_LOG_DEBUG;
+- case GPR_LOG_SEVERITY_INFO:
+- return ANDROID_LOG_INFO;
+- case GPR_LOG_SEVERITY_ERROR:
+- return ANDROID_LOG_ERROR;
+- }
+- return ANDROID_LOG_DEFAULT;
+-}
+-
+ void gpr_log(const char* file, int line, gpr_log_severity severity,
+ const char* format, ...) {
+ // Avoid message construction if gpr_log_message won't log
+@@ -70,8 +58,6 @@
+
+ asprintf(&output, "%s:%d] %s", display_file, args->line, args->message);
+
+- __android_log_write(severity_to_log_priority(args->severity), "GRPC", output);
+-
+ // allocated by asprintf => use free, not gpr_free
+ free(output);
+ }
From b9341ef9f171bd11bc2ca62c849290f960293250 Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Fri, 31 May 2024 23:21:14 +0100
Subject: [PATCH 043/158] Migrate to PyProjectRecipe
---
pythonforandroid/recipes/grpcio/__init__.py | 38 +++++--------------
...-ndk-zlib-and-openssl-recipe-include.patch | 16 ++++++++
2 files changed, 26 insertions(+), 28 deletions(-)
create mode 100644 pythonforandroid/recipes/grpcio/use-ndk-zlib-and-openssl-recipe-include.patch
diff --git a/pythonforandroid/recipes/grpcio/__init__.py b/pythonforandroid/recipes/grpcio/__init__.py
index 4276f52e41..3a442f6297 100644
--- a/pythonforandroid/recipes/grpcio/__init__.py
+++ b/pythonforandroid/recipes/grpcio/__init__.py
@@ -1,13 +1,7 @@
-import glob
-from logging import info
+from pythonforandroid.recipe import PyProjectRecipe, Recipe
-import sh
-from pythonforandroid.logger import shprint
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
-from pythonforandroid.util import current_directory
-
-class GrpcioRecipe(CppCompiledComponentsPythonRecipe):
+class GrpcioRecipe(PyProjectRecipe):
version = '1.64.0'
url = 'https://files.pythonhosted.org/packages/source/g/grpcio/grpcio-{version}.tar.gz'
depends = ["setuptools", "librt", "libpthread"]
@@ -15,32 +9,20 @@ class GrpcioRecipe(CppCompiledComponentsPythonRecipe):
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
- env['NDKPLATFORM'] = "NOTNONE"
- env['GRPC_PYTHON_BUILD_WITH_CYTHON'] = '1'
+ env["NDKPLATFORM"] = "NOTNONE"
+ env["GRPC_PYTHON_BUILD_SYSTEM_OPENSSL"] = "1"
+ env["GRPC_PYTHON_BUILD_SYSTEM_ZLIB"] = "1"
+ env["ZLIB_INCLUDE"] = self.ctx.ndk.sysroot_include_dir
+ # replace -I with a space
+ openssl_recipe = Recipe.get_recipe('openssl', self.ctx)
+ env["SSL_INCLUDE"] = openssl_recipe.include_flags(arch).strip().replace("-I", "")
env["CFLAGS"] += " -U__ANDROID_API__"
env["CFLAGS"] += " -D__ANDROID_API__={}".format(self.ctx.ndk_api)
-
# turn off c++11 warning error of "invalid suffix on literal"
env["CFLAGS"] += " -Wno-reserved-user-defined-literal"
- env['PLATFORM'] = 'android'
+ env["PLATFORM"] = "android"
env["LDFLAGS"] += " -llog -landroid"
return env
- def build_compiled_components(self, arch):
- info('Building compiled components in {}'.format(self.name))
-
- env = self.get_recipe_env(arch)
- hostpython = sh.Command(self.hostpython_location)
- with current_directory(self.get_build_dir(arch.arch)):
- if self.install_in_hostpython:
- shprint(hostpython, 'setup.py', 'clean', '--all', _env=env)
- shprint(hostpython, 'setup.py', self.build_cmd, '-v',
- _env=env, *self.setup_extra_args)
-
- # grpcio creates a build directory and names it `pyb`
- build_dir = glob.glob('pyb/lib.*')[0]
- shprint(sh.find, build_dir, '-name', '"*.o"', '-exec',
- env['STRIP'], '{}', ';', _env=env)
-
recipe = GrpcioRecipe()
diff --git a/pythonforandroid/recipes/grpcio/use-ndk-zlib-and-openssl-recipe-include.patch b/pythonforandroid/recipes/grpcio/use-ndk-zlib-and-openssl-recipe-include.patch
new file mode 100644
index 0000000000..7810d50771
--- /dev/null
+++ b/pythonforandroid/recipes/grpcio/use-ndk-zlib-and-openssl-recipe-include.patch
@@ -0,0 +1,16 @@
+--- a/setup.py 2024-05-31 11:20:56.824695569 +0100
++++ b/setup.py 2024-05-31 23:13:40.324392463 +0100
+@@ -299,11 +299,11 @@
+ lambda x: "third_party/boringssl" not in x, CORE_C_FILES
+ )
+ CORE_C_FILES = filter(lambda x: "src/boringssl" not in x, CORE_C_FILES)
+- SSL_INCLUDE = (os.path.join("/usr", "include", "openssl"),)
++ SSL_INCLUDE = tuple(os.environ["SSL_INCLUDE"].split(" "))
+
+ if BUILD_WITH_SYSTEM_ZLIB:
+ CORE_C_FILES = filter(lambda x: "third_party/zlib" not in x, CORE_C_FILES)
+- ZLIB_INCLUDE = (os.path.join("/usr", "include"),)
++ ZLIB_INCLUDE = tuple(os.environ["ZLIB_INCLUDE"].split(" "))
+
+ if BUILD_WITH_SYSTEM_CARES:
+ CORE_C_FILES = filter(lambda x: "third_party/cares" not in x, CORE_C_FILES)
From 4b7500fcee090572b3d94428384083bdae57dce9 Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Sat, 1 Jun 2024 08:04:27 +0100
Subject: [PATCH 044/158] Add patch
---
pythonforandroid/recipes/grpcio/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/grpcio/__init__.py b/pythonforandroid/recipes/grpcio/__init__.py
index 3a442f6297..ceb0f61217 100644
--- a/pythonforandroid/recipes/grpcio/__init__.py
+++ b/pythonforandroid/recipes/grpcio/__init__.py
@@ -5,7 +5,7 @@ class GrpcioRecipe(PyProjectRecipe):
version = '1.64.0'
url = 'https://files.pythonhosted.org/packages/source/g/grpcio/grpcio-{version}.tar.gz'
depends = ["setuptools", "librt", "libpthread"]
- patches = ["comment-getserverbyport-r-args.patch", "remove-android-log-write.patch"]
+ patches = ["comment-getserverbyport-r-args.patch", "remove-android-log-write.patch", "use-ndk-zlib-and-openssl-recipe-include.patch"]
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
From 3a147350f05c23af056021ee8d3c04b72b06beca Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Sat, 1 Jun 2024 08:10:22 +0100
Subject: [PATCH 045/158] Update __init__.py
---
pythonforandroid/recipes/grpcio/__init__.py | 6 +++++-
1 file changed, 5 insertions(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/grpcio/__init__.py b/pythonforandroid/recipes/grpcio/__init__.py
index ceb0f61217..8e3b9cd7a7 100644
--- a/pythonforandroid/recipes/grpcio/__init__.py
+++ b/pythonforandroid/recipes/grpcio/__init__.py
@@ -5,7 +5,11 @@ class GrpcioRecipe(PyProjectRecipe):
version = '1.64.0'
url = 'https://files.pythonhosted.org/packages/source/g/grpcio/grpcio-{version}.tar.gz'
depends = ["setuptools", "librt", "libpthread"]
- patches = ["comment-getserverbyport-r-args.patch", "remove-android-log-write.patch", "use-ndk-zlib-and-openssl-recipe-include.patch"]
+ patches = [
+ "comment-getserverbyport-r-args.patch",
+ "remove-android-log-write.patch",
+ "use-ndk-zlib-and-openssl-recipe-include.patch"
+ ]
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
From af04bee3ec68764380c6190068bb6c7ace89040f Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 2 Jun 2024 10:55:21 +0530
Subject: [PATCH 046/158] `ffpyplayer`: update to `4.5.1`
---
pythonforandroid/recipes/ffpyplayer/__init__.py | 8 ++++----
.../recipes/ffpyplayer/setup.py.patch | 15 +++++++++++++++
2 files changed, 19 insertions(+), 4 deletions(-)
create mode 100644 pythonforandroid/recipes/ffpyplayer/setup.py.patch
diff --git a/pythonforandroid/recipes/ffpyplayer/__init__.py b/pythonforandroid/recipes/ffpyplayer/__init__.py
index 6260037a70..a78f65d8c6 100644
--- a/pythonforandroid/recipes/ffpyplayer/__init__.py
+++ b/pythonforandroid/recipes/ffpyplayer/__init__.py
@@ -1,12 +1,12 @@
-from pythonforandroid.recipe import CythonRecipe
-from pythonforandroid.toolchain import Recipe
+from pythonforandroid.recipe import PyProjectRecipe, Recipe
from os.path import join
-class FFPyPlayerRecipe(CythonRecipe):
- version = 'v4.3.2'
+class FFPyPlayerRecipe(PyProjectRecipe):
+ version = 'v4.5.1'
url = 'https://github.com/matham/ffpyplayer/archive/{version}.zip'
depends = ['python3', 'sdl2', 'ffmpeg']
+ patches = ["setup.py.patch"]
opt_depends = ['openssl', 'ffpyplayer_codecs']
def get_recipe_env(self, arch, with_flags_in_cc=True):
diff --git a/pythonforandroid/recipes/ffpyplayer/setup.py.patch b/pythonforandroid/recipes/ffpyplayer/setup.py.patch
new file mode 100644
index 0000000000..c082358f43
--- /dev/null
+++ b/pythonforandroid/recipes/ffpyplayer/setup.py.patch
@@ -0,0 +1,15 @@
+--- ffpyplayer/setup.py 2024-06-02 11:10:49.691183467 +0530
++++ ffpyplayer.mod/setup.py 2024-06-02 11:20:16.220966873 +0530
+@@ -27,12 +27,6 @@
+ # This sets whether or not Cython gets added to setup_requires.
+ declare_cython = False
+
+-if platform in ('ios', 'android'):
+- # NEVER use or declare cython on these platforms
+- print('Not using cython on %s' % platform)
+- can_use_cython = False
+-else:
+- declare_cython = True
+
+ src_path = build_path = dirname(__file__)
+ print(f'Source/build path: {src_path}')
From 19bebcd67ff0d8cbd55e9cbe814e2fd7a9422007 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 2 Jun 2024 12:51:28 +0530
Subject: [PATCH 047/158] `archs.py`: remove backslash logic
---
pythonforandroid/archs.py | 5 ++---
1 file changed, 2 insertions(+), 3 deletions(-)
diff --git a/pythonforandroid/archs.py b/pythonforandroid/archs.py
index b065174592..4137768096 100644
--- a/pythonforandroid/archs.py
+++ b/pythonforandroid/archs.py
@@ -144,9 +144,8 @@ def get_env(self, with_flags_in_cc=True):
' '
+ " ".join(
[
- "-L'"
- + link_path.replace("'", "'\"'\"'")
- + "'" # no shlex.quote in py2
+ "-L"
+ + link_path
for link_path in self.extra_global_link_paths
]
)
From f691a8b92382df1ca85a35f4c2c74acfd44d15e5 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 2 Jun 2024 20:17:31 +0530
Subject: [PATCH 048/158] `sqlalchemy`: update to `2.0.30`
---
pythonforandroid/recipes/sqlalchemy/__init__.py | 14 +++++++-------
pythonforandroid/recipes/sqlalchemy/zipsafe.patch | 10 ----------
2 files changed, 7 insertions(+), 17 deletions(-)
delete mode 100644 pythonforandroid/recipes/sqlalchemy/zipsafe.patch
diff --git a/pythonforandroid/recipes/sqlalchemy/__init__.py b/pythonforandroid/recipes/sqlalchemy/__init__.py
index 9837a59d09..3b6ab683cb 100644
--- a/pythonforandroid/recipes/sqlalchemy/__init__.py
+++ b/pythonforandroid/recipes/sqlalchemy/__init__.py
@@ -1,15 +1,15 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class SQLAlchemyRecipe(CompiledComponentsPythonRecipe):
+class SQLAlchemyRecipe(PyProjectRecipe):
name = 'sqlalchemy'
- version = '1.3.3'
- url = 'https://pypi.python.org/packages/source/S/SQLAlchemy/SQLAlchemy-{version}.tar.gz'
- call_hostpython_via_targetpython = False
-
+ version = '2.0.30'
+ url = 'https://github.com/sqlalchemy/sqlalchemy/archive/refs/tags/rel_{}.tar.gz'
depends = ['setuptools']
- patches = ['zipsafe.patch']
+ @property
+ def versioned_url(self):
+ return self.url.format(self.version.replace(".", "_"))
recipe = SQLAlchemyRecipe()
diff --git a/pythonforandroid/recipes/sqlalchemy/zipsafe.patch b/pythonforandroid/recipes/sqlalchemy/zipsafe.patch
deleted file mode 100644
index 46bdf60106..0000000000
--- a/pythonforandroid/recipes/sqlalchemy/zipsafe.patch
+++ /dev/null
@@ -1,10 +0,0 @@
---- a/setup.py 2019-04-15 17:45:03.000000000 +0200
-+++ b/setup.py 2019-04-16 20:12:19.056710749 +0200
-@@ -145,6 +145,7 @@
- name="SQLAlchemy",
- version=VERSION,
- description="Database Abstraction Library",
-+ zip_safe=False,
- author="Mike Bayer",
- author_email="mike_mp@zzzcomputing.com",
- url="http://www.sqlalchemy.org",
From 8c931ed4ddbcb20591e1551c193410e3094b9dcd Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Sat, 8 Jun 2024 00:49:44 +0100
Subject: [PATCH 049/158] Fix -lss, lcrypto linking bug
---
pythonforandroid/recipes/grpcio/__init__.py | 1 +
1 file changed, 1 insertion(+)
diff --git a/pythonforandroid/recipes/grpcio/__init__.py b/pythonforandroid/recipes/grpcio/__init__.py
index 8e3b9cd7a7..3cf2437e32 100644
--- a/pythonforandroid/recipes/grpcio/__init__.py
+++ b/pythonforandroid/recipes/grpcio/__init__.py
@@ -26,6 +26,7 @@ def get_recipe_env(self, arch, **kwargs):
env["CFLAGS"] += " -Wno-reserved-user-defined-literal"
env["PLATFORM"] = "android"
env["LDFLAGS"] += " -llog -landroid"
+ env["LDFLAGS"] += openssl_recipe.link_flags(arch)
return env
From 82b1be9fdfe6ed75208d8331fca0aae3bc49e5bd Mon Sep 17 00:00:00 2001
From: Joseph <josephnjoseph43@gmail.com>
Date: Wed, 17 Jul 2024 15:45:40 +0530
Subject: [PATCH 050/158] =?UTF-8?q?=E2=9C=A8Fixed=20wrong=20spelling:=20de?=
=?UTF-8?q?pelopment=20->=20development?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
Fixed wrong spelling of development
---
doc/source/buildoptions.rst | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/doc/source/buildoptions.rst b/doc/source/buildoptions.rst
index 782bd67295..44e9578b1b 100644
--- a/doc/source/buildoptions.rst
+++ b/doc/source/buildoptions.rst
@@ -36,7 +36,7 @@ sdl2
Use this with ``--bootstrap=sdl2``, or just include the
``sdl2`` recipe, e.g. ``--requirements=sdl2,python3``.
-SDL2 is a popular cross-platform depelopment library, particularly for
+SDL2 is a popular cross-platform development library, particularly for
games. It has its own Android project support, which
python-for-android uses as a bootstrap, and to which it adds the
Python build and JNI code to start it.
From 0364903b33b3c9f33d1f9d995b12db84a0ce7421 Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Mon, 29 Jul 2024 08:38:38 +0100
Subject: [PATCH 051/158] Fix cannot locate symbol "getservbyport_y" (#3043)
---
.../comment-getserverbyport-r-args.patch | 36 +++++++++++++------
1 file changed, 26 insertions(+), 10 deletions(-)
diff --git a/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch b/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
index 6d64105d70..92835ac9e6 100644
--- a/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
+++ b/pythonforandroid/recipes/grpcio/comment-getserverbyport-r-args.patch
@@ -1,17 +1,33 @@
-Index: ares_config.h
-IDEA additional info:
-Subsystem: com.intellij.openapi.diff.impl.patch.CharsetEP
-<+>UTF-8
-===================================================================
-diff --git a/third_party/cares/config_android/ares_config.h b/third_party/cares/config_android/ares_config.h
---- a/third_party/cares/config_android/ares_config.h
-+++ b/third_party/cares/config_android/ares_config.h (date 1716777985227)
+diff --git a/third_party/cares/config_darwin/ares_config.h b/third_party/cares/config_darwin/ares_config.h
+--- a/third_party/cares/config_darwin/ares_config.h 2024-07-16 20:46:22.000000000 +0100
++++ b/third_party/cares/config_darwin/ares_config.h 2024-07-29 00:18:30.096755745 +0100
@@ -43,7 +43,7 @@
#define GETNAMEINFO_TYPE_ARG7 int
/* Specifies the number of arguments to getservbyport_r */
--#define GETSERVBYPORT_R_ARGS
-+//#define GETSERVBYPORT_R_ARGS
+-#define GETSERVBYPORT_R_ARGS
++/* #define GETSERVBYPORT_R_ARGS */
/* Define to 1 if you have AF_INET6. */
#define HAVE_AF_INET6
+diff --git a/third_party/cares/config_linux/ares_config.h b/third_party/cares/config_linux/ares_config.h
+--- a/third_party/cares/config_linux/ares_config.h 2024-07-16 20:46:22.000000000 +0100
++++ b/third_party/cares/config_linux/ares_config.h 2024-07-29 00:19:39.479166654 +0100
+@@ -43,7 +43,7 @@
+ #define GETNAMEINFO_TYPE_ARG7 int
+
+ /* Specifies the number of arguments to getservbyport_r */
+-#define GETSERVBYPORT_R_ARGS 6
++/* #define GETSERVBYPORT_R_ARGS 6 */
+
+ /* Define to 1 if you have AF_INET6. */
+ #define HAVE_AF_INET6
+@@ -121,7 +121,7 @@
+ #define HAVE_GETNAMEINFO
+
+ /* Define to 1 if you have the getservbyport_r function. */
+-#define HAVE_GETSERVBYPORT_R
++/* #define HAVE_GETSERVBYPORT_R */
+
+ /* Define to 1 if you have the `gettimeofday' function. */
+ #define HAVE_GETTIMEOFDAY
\ No newline at end of file
From 3ae66d5757fd74045d0dfbbae900e18d12d44d93 Mon Sep 17 00:00:00 2001
From: gon <gonatienza@gmail.com>
Date: Fri, 26 Jul 2024 15:22:18 -0400
Subject: [PATCH 052/158] libsodium-recipe-update to url
libsodium-recipe-update to url
libsodium-recipe-update to url
libsodium-recipe-update to url
libsodium-recipe-update to url
libsodium-recipe-update to url
---
pythonforandroid/recipes/libsodium/__init__.py | 11 ++++++++++-
1 file changed, 10 insertions(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/libsodium/__init__.py b/pythonforandroid/recipes/libsodium/__init__.py
index f66fc18e7f..a8a1909588 100644
--- a/pythonforandroid/recipes/libsodium/__init__.py
+++ b/pythonforandroid/recipes/libsodium/__init__.py
@@ -3,15 +3,24 @@
from pythonforandroid.logger import shprint
from multiprocessing import cpu_count
import sh
+from packaging import version as packaging_version
class LibsodiumRecipe(Recipe):
version = '1.0.16'
- url = 'https://github.com/jedisct1/libsodium/releases/download/{version}/libsodium-{version}.tar.gz'
+ url = 'https://github.com/jedisct1/libsodium/releases/download/{}/libsodium-{}.tar.gz'
depends = []
patches = ['size_max_fix.patch']
built_libraries = {'libsodium.so': 'src/libsodium/.libs'}
+ @property
+ def versioned_url(self):
+ asked_version = packaging_version.parse(self.version)
+ if asked_version > packaging_version.parse('1.0.16'):
+ return self._url.format(self.version + '-RELEASE', self.version)
+ else:
+ return self._url.format(self.version, self.version)
+
def build_arch(self, arch):
env = self.get_recipe_env(arch)
with current_directory(self.get_build_dir(arch.arch)):
From 638f2e2b16dbc14324db59d6daf3251542d9627d Mon Sep 17 00:00:00 2001
From: Emerson MX <emersonmx@gmail.com>
Date: Sat, 3 Aug 2024 20:07:23 -0300
Subject: [PATCH 053/158] Add moderngl recipe
---
pythonforandroid/recipes/moderngl/__init__.py | 17 +++++++++++++++++
1 file changed, 17 insertions(+)
create mode 100644 pythonforandroid/recipes/moderngl/__init__.py
diff --git a/pythonforandroid/recipes/moderngl/__init__.py b/pythonforandroid/recipes/moderngl/__init__.py
new file mode 100644
index 0000000000..38564eb7ce
--- /dev/null
+++ b/pythonforandroid/recipes/moderngl/__init__.py
@@ -0,0 +1,17 @@
+from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+
+
+class ModernGLRecipe(CppCompiledComponentsPythonRecipe):
+ version = '5.10.0'
+ url = 'https://github.com/moderngl/moderngl/archive/refs/tags/{version}.tar.gz'
+
+ site_packages_name = 'moderngl'
+ name = 'moderngl'
+
+ def get_recipe_env(self, arch):
+ env = super().get_recipe_env(arch)
+ env['LDFLAGS'] += ' -lstdc++'
+ return env
+
+
+recipe = ModernGLRecipe()
From 810c0791997d9a7922576bc60581ea634daa2e29 Mon Sep 17 00:00:00 2001
From: Jin Hyung Ahn <zenoengine@gmail.com>
Date: Wed, 7 Aug 2024 10:52:34 +0900
Subject: [PATCH 054/158] Fix broken quickstart link
---
README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/README.md b/README.md
index c7cbb96c9a..96cc898e64 100644
--- a/README.md
+++ b/README.md
@@ -52,7 +52,7 @@ python-for-android is not limited to being used with Buildozer.
More information is available in the
[online documentation](https://python-for-android.readthedocs.io) including a
-[quickstart guide](https://python-for-android.readthedocs.io/en/latest/quickstart/).
+[quickstart guide](https://python-for-android.readthedocs.io/en/latest/quickstart.html).
python-for-android is managed by the [Kivy team](https://kivy.org).
From f526e6b4033bc6b1a56b5b1db7aa8f9246a47dbc Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sun, 2 Jun 2024 19:55:13 +0200
Subject: [PATCH 055/158] :construction_worker: Publish to DockerHub
Our public DockerHub Image was 3 years old as the automatic build became
a pro feature.
With this change we're now publishing to DockerHub from the CI on push
to the develop branch.
Authentication to DockerHub is done using personal access tokens
configured using `andremiras` username, but this can be updated
anytime with another token refs:
- https://app.docker.com/settings/personal-access-tokens
- https://github.com/kivy/python-for-android/settings/secrets/actions
We use a dedicated docker.yml workflow file rather than building on top
of the existing push.yml one to keep things clean and separated. In the
future we may add multi architecture support.
Updating the public image also fixes the issue where the sh logger was
too verbose leading to huge and unreadable CI logs.
---
.github/workflows/docker.yml | 23 +++++++++++++++++++++++
Makefile | 3 +++
README.md | 1 +
3 files changed, 27 insertions(+)
create mode 100644 .github/workflows/docker.yml
diff --git a/.github/workflows/docker.yml b/.github/workflows/docker.yml
new file mode 100644
index 0000000000..63a8813af7
--- /dev/null
+++ b/.github/workflows/docker.yml
@@ -0,0 +1,23 @@
+name: Docker
+
+on:
+ workflow_dispatch:
+ push:
+ branches:
+ - develop
+ pull_request:
+
+jobs:
+ docker:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+ - uses: docker/setup-buildx-action@v3
+ - run: make docker/build
+ - run: make docker/login
+ env:
+ DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }}
+ DOCKERHUB_TOKEN: ${{ secrets.DOCKERHUB_TOKEN }}
+ - name: docker push
+ if: github.ref == 'develop'
+ run: make docker/push
diff --git a/Makefile b/Makefile
index 6e30ee5936..2cf08b7537 100644
--- a/Makefile
+++ b/Makefile
@@ -117,6 +117,9 @@ docker/pull:
docker/build:
docker build --cache-from=$(DOCKER_IMAGE) --tag=$(DOCKER_IMAGE) .
+docker/login:
+ @echo $(DOCKERHUB_TOKEN) | docker login --username $(DOCKERHUB_USERNAME) --password-stdin
+
docker/push:
docker push $(DOCKER_IMAGE)
diff --git a/README.md b/README.md
index 96cc898e64..c7175e6a28 100644
--- a/README.md
+++ b/README.md
@@ -47,6 +47,7 @@ python-for-android is not limited to being used with Buildozer.
[](https://github.com/kivy/python-for-android/actions?query=workflow%3A%22Unit+tests+%26+build+apps%22)
[](https://coveralls.io/github/kivy/python-for-android?branch=develop)
+[](https://github.com/kivy/python-for-android/actions/workflows/docker.yml)
## Documentation
From a85d24ee278360f3314f0874644ccc6944df7fdb Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sun, 11 Aug 2024 22:13:04 +0200
Subject: [PATCH 056/158] :mute: Fix sh logs too verbose
When setuptools is not present and the root logger has no handlers,
Wheels would configure the root logger with DEBUG level, refs:
- https://github.com/pypa/wheel/blob/0.44.0/src/wheel/util.py
- https://github.com/pypa/wheel/blob/0.44.0/src/wheel/_setuptools_logging.py
Both of these conditions are met in our CI, leading to very verbose
and unreadable `sh` logs. Patching it prevents that.
---
pythonforandroid/recipe.py | 7 ++++---
pythonforandroid/util.py | 14 ++++++++++++++
2 files changed, 18 insertions(+), 3 deletions(-)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index c4131c53e6..5165f19030 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -12,8 +12,6 @@
from urllib.request import urlretrieve
from os import listdir, unlink, environ, curdir, walk
from sys import stdout
-from wheel.wheelfile import WheelFile
-from wheel.cli.tags import tags as wheel_tags
import time
try:
from urlparse import urlparse
@@ -26,7 +24,7 @@
logger, info, warning, debug, shprint, info_main, error)
from pythonforandroid.util import (
current_directory, ensure_dir, BuildInterruptingException, rmdir, move,
- touch)
+ touch, patch_wheel_setuptools_logging)
from pythonforandroid.util import load_source as import_recipe
@@ -1205,6 +1203,9 @@ def get_wheel_platform_tag(self, arch):
}[arch.arch]
def install_wheel(self, arch, built_wheels):
+ with patch_wheel_setuptools_logging():
+ from wheel.cli.tags import tags as wheel_tags
+ from wheel.wheelfile import WheelFile
_wheel = built_wheels[0]
built_wheel_dir = dirname(_wheel)
# Fix wheel platform tag
diff --git a/pythonforandroid/util.py b/pythonforandroid/util.py
index 2738d59990..13f38e6c0f 100644
--- a/pythonforandroid/util.py
+++ b/pythonforandroid/util.py
@@ -1,4 +1,5 @@
import contextlib
+from unittest import mock
from fnmatch import fnmatch
import logging
from os.path import exists, join
@@ -163,3 +164,16 @@ def max_build_tool_version(
"""
return max(build_tools_versions, key=build_tools_version_sort_key)
+
+
+def patch_wheel_setuptools_logging():
+ """
+ When setuptools is not present and the root logger has no handlers,
+ Wheels would configure the root logger with DEBUG level, refs:
+ - https://github.com/pypa/wheel/blob/0.44.0/src/wheel/util.py
+ - https://github.com/pypa/wheel/blob/0.44.0/src/wheel/_setuptools_logging.py
+
+ Both of these conditions are met in our CI, leading to very verbose
+ and unreadable `sh` logs. Patching it prevents that.
+ """
+ return mock.patch("wheel._setuptools_logging.configure")
From f8e5c05889c121181f309177ed0bc7b14e896212 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sun, 11 Aug 2024 22:13:04 +0200
Subject: [PATCH 057/158] :arrow_up: Bump sh module to v2, refs #2746
---
pythonforandroid/build.py | 6 +++---
pythonforandroid/recipe.py | 3 +--
pythonforandroid/recipes/python3/__init__.py | 2 +-
setup.py | 2 +-
4 files changed, 6 insertions(+), 7 deletions(-)
diff --git a/pythonforandroid/build.py b/pythonforandroid/build.py
index 0cc26db264..05e4e1db7f 100644
--- a/pythonforandroid/build.py
+++ b/pythonforandroid/build.py
@@ -29,14 +29,14 @@
def get_targets(sdk_dir):
if exists(join(sdk_dir, 'cmdline-tools', 'latest', 'bin', 'avdmanager')):
avdmanager = sh.Command(join(sdk_dir, 'cmdline-tools', 'latest', 'bin', 'avdmanager'))
- targets = avdmanager('list', 'target').stdout.decode('utf-8').split('\n')
+ targets = avdmanager('list', 'target').split('\n')
elif exists(join(sdk_dir, 'tools', 'bin', 'avdmanager')):
avdmanager = sh.Command(join(sdk_dir, 'tools', 'bin', 'avdmanager'))
- targets = avdmanager('list', 'target').stdout.decode('utf-8').split('\n')
+ targets = avdmanager('list', 'target').split('\n')
elif exists(join(sdk_dir, 'tools', 'android')):
android = sh.Command(join(sdk_dir, 'tools', 'android'))
- targets = android('list').stdout.decode('utf-8').split('\n')
+ targets = android('list').split('\n')
else:
raise BuildInterruptingException(
'Could not find `android` or `sdkmanager` binaries in Android SDK',
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 5165f19030..ff34ac29f2 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -467,8 +467,7 @@ def unpack(self, arch):
elif extraction_filename.endswith(
('.tar.gz', '.tgz', '.tar.bz2', '.tbz2', '.tar.xz', '.txz')):
sh.tar('xf', extraction_filename)
- root_directory = sh.tar('tf', extraction_filename).stdout.decode(
- 'utf-8').split('\n')[0].split('/')[0]
+ root_directory = sh.tar('tf', extraction_filename).split('\n')[0].split('/')[0]
if root_directory != basename(directory_name):
move(root_directory, directory_name)
else:
diff --git a/pythonforandroid/recipes/python3/__init__.py b/pythonforandroid/recipes/python3/__init__.py
index 3a9575148a..2334db6add 100644
--- a/pythonforandroid/recipes/python3/__init__.py
+++ b/pythonforandroid/recipes/python3/__init__.py
@@ -320,7 +320,7 @@ def build_arch(self, arch):
android_build = sh.Command(
join(recipe_build_dir,
- 'config.guess'))().stdout.strip().decode('utf-8')
+ 'config.guess'))().strip()
with current_directory(build_dir):
if not exists('config.status'):
diff --git a/setup.py b/setup.py
index 4a628df905..c9cebd2314 100644
--- a/setup.py
+++ b/setup.py
@@ -21,7 +21,7 @@
# https://github.com/kivy/buildozer/issues/722
install_reqs = [
'appdirs', 'colorama>=0.3.3', 'jinja2',
- 'sh>=1.10, <2.0; sys_platform!="win32"',
+ 'sh>=2, <3.0; sys_platform!="win32"',
'build', 'toml', 'packaging', 'setuptools', 'wheel~=0.43.0'
]
# (build and toml are used by pythonpackage.py)
From 88cc5d813378eb2d8418b104fc2d36ae50e88edf Mon Sep 17 00:00:00 2001
From: RobertF <34464649+RobertFlatt@users.noreply.github.com>
Date: Sun, 18 Aug 2024 15:13:10 -1000
Subject: [PATCH 058/158] Add tiktoken recipe
---
pythonforandroid/recipes/tiktoken/__init__.py | 12 ++++++++++++
1 file changed, 12 insertions(+)
create mode 100644 pythonforandroid/recipes/tiktoken/__init__.py
diff --git a/pythonforandroid/recipes/tiktoken/__init__.py b/pythonforandroid/recipes/tiktoken/__init__.py
new file mode 100644
index 0000000000..a6145f9bf5
--- /dev/null
+++ b/pythonforandroid/recipes/tiktoken/__init__.py
@@ -0,0 +1,12 @@
+from pythonforandroid.recipe import RustCompiledComponentsRecipe
+
+
+class TiktokenRecipe(RustCompiledComponentsRecipe):
+ name = 'tiktoken'
+ version = '0.7.0'
+ url = 'https://github.com/openai/tiktoken/archive/refs/tags/{version}.tar.gz'
+ depends = ['regex','requests']
+
+
+recipe = TiktokenRecipe()
+
From 18b1bd675d6eb83a5e0133fea7fa5d1e9ce03216 Mon Sep 17 00:00:00 2001
From: RobertF <34464649+RobertFlatt@users.noreply.github.com>
Date: Sun, 18 Aug 2024 15:18:53 -1000
Subject: [PATCH 059/158] Update __init__.py
---
pythonforandroid/recipes/tiktoken/__init__.py | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/pythonforandroid/recipes/tiktoken/__init__.py b/pythonforandroid/recipes/tiktoken/__init__.py
index a6145f9bf5..1c5407054e 100644
--- a/pythonforandroid/recipes/tiktoken/__init__.py
+++ b/pythonforandroid/recipes/tiktoken/__init__.py
@@ -5,8 +5,7 @@ class TiktokenRecipe(RustCompiledComponentsRecipe):
name = 'tiktoken'
version = '0.7.0'
url = 'https://github.com/openai/tiktoken/archive/refs/tags/{version}.tar.gz'
- depends = ['regex','requests']
+ depends = ['regex', 'requests']
recipe = TiktokenRecipe()
-
From 496f91646da34d0c30d8f435912ce3936628b8f8 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Mon, 19 Aug 2024 11:46:49 +0200
Subject: [PATCH 060/158] :construction_worker: Fix docker login
Run docker login on push to develop rather than during pull requests
---
.github/workflows/docker.yml | 5 +++--
1 file changed, 3 insertions(+), 2 deletions(-)
diff --git a/.github/workflows/docker.yml b/.github/workflows/docker.yml
index 63a8813af7..bd918a9c74 100644
--- a/.github/workflows/docker.yml
+++ b/.github/workflows/docker.yml
@@ -14,10 +14,11 @@ jobs:
- uses: actions/checkout@v4
- uses: docker/setup-buildx-action@v3
- run: make docker/build
- - run: make docker/login
env:
DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }}
DOCKERHUB_TOKEN: ${{ secrets.DOCKERHUB_TOKEN }}
- name: docker push
if: github.ref == 'develop'
- run: make docker/push
+ run: |
+ make docker/login
+ make docker/push
From 80dc7a44dba1d3f5610a9b925de3145d7e93d6c0 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Tue, 20 Aug 2024 20:16:34 +0200
Subject: [PATCH 061/158] :bug: Fix rebuild updated recipes
The bug probably got introduced with #3049
Update the tiktoken to demonstrate the fix.
---
ci/rebuild_updated_recipes.py | 2 +-
pythonforandroid/recipes/tiktoken/__init__.py | 1 +
2 files changed, 2 insertions(+), 1 deletion(-)
diff --git a/ci/rebuild_updated_recipes.py b/ci/rebuild_updated_recipes.py
index a1ca96681d..26cba3bc7e 100755
--- a/ci/rebuild_updated_recipes.py
+++ b/ci/rebuild_updated_recipes.py
@@ -39,7 +39,7 @@ def modified_recipes(branch='origin/develop'):
# using the contrib version on purpose rather than sh.git, since it comes
# with a bunch of fixes, e.g. disabled TTY, see:
# https://stackoverflow.com/a/20128598/185510
- git_diff = sh.contrib.git.diff('--name-only', branch)
+ git_diff = sh.contrib.git.diff('--name-only', branch).split("\n")
recipes = set()
for file_path in git_diff:
if 'pythonforandroid/recipes/' in file_path:
diff --git a/pythonforandroid/recipes/tiktoken/__init__.py b/pythonforandroid/recipes/tiktoken/__init__.py
index 1c5407054e..5e2a9a7243 100644
--- a/pythonforandroid/recipes/tiktoken/__init__.py
+++ b/pythonforandroid/recipes/tiktoken/__init__.py
@@ -5,6 +5,7 @@ class TiktokenRecipe(RustCompiledComponentsRecipe):
name = 'tiktoken'
version = '0.7.0'
url = 'https://github.com/openai/tiktoken/archive/refs/tags/{version}.tar.gz'
+ sha512sum = "bb2d8fd5acd660d60e690769e46cf29b06361343ea30e35613d27d55f44acf9834e51aef28f4ff316ef66f2130042079718cea04b2353301aef334cd7bd6d221"
depends = ['regex', 'requests']
From 74b891e4b93f0999170a0191184a766cf00e7e19 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Thu, 22 Aug 2024 10:09:33 +0200
Subject: [PATCH 062/158] :recycle: Minor code cleaning
Reuse `project_has_setup_py()` when appropriate and make some
constructions more concise.
Also fix minor docstring typo.
---
pythonforandroid/build.py | 8 ++------
pythonforandroid/pythonpackage.py | 7 +++----
pythonforandroid/recipe.py | 2 +-
pythonforandroid/toolchain.py | 29 +++++++++--------------------
4 files changed, 15 insertions(+), 31 deletions(-)
diff --git a/pythonforandroid/build.py b/pythonforandroid/build.py
index 05e4e1db7f..98e2d70b2b 100644
--- a/pythonforandroid/build.py
+++ b/pythonforandroid/build.py
@@ -672,7 +672,6 @@ def run_pymodules_install(ctx, arch, modules, project_dir=None,
# Bail out if no python deps and no setup.py to process:
if not modules and (
ignore_setup_py or
- project_dir is None or
not project_has_setup_py(project_dir)
):
info('No Python modules and no setup.py to process, skipping')
@@ -688,8 +687,7 @@ def run_pymodules_install(ctx, arch, modules, project_dir=None,
"If this fails, it may mean that the module has compiled "
"components and needs a recipe."
)
- if project_dir is not None and \
- project_has_setup_py(project_dir) and not ignore_setup_py:
+ if project_has_setup_py(project_dir) and not ignore_setup_py:
info(
"Will process project install, if it fails then the "
"project may not be compatible for Android install."
@@ -761,9 +759,7 @@ def run_pymodules_install(ctx, arch, modules, project_dir=None,
_env=copy.copy(env))
# Afterwards, run setup.py if present:
- if project_dir is not None and (
- project_has_setup_py(project_dir) and not ignore_setup_py
- ):
+ if project_has_setup_py(project_dir) and not ignore_setup_py:
run_setuppy_install(ctx, project_dir, env, arch)
elif not ignore_setup_py:
info("No setup.py found in project directory: " + str(project_dir))
diff --git a/pythonforandroid/pythonpackage.py b/pythonforandroid/pythonpackage.py
index 9e4c29bd81..602235cf23 100644
--- a/pythonforandroid/pythonpackage.py
+++ b/pythonforandroid/pythonpackage.py
@@ -556,11 +556,10 @@ def _extract_info_from_package(dependency,
# Get build requirements from pyproject.toml if requested:
requirements = []
- if os.path.exists(os.path.join(output_folder,
- 'pyproject.toml')
- ) and include_build_requirements:
+ pyproject_toml_path = os.path.join(output_folder, 'pyproject.toml')
+ if os.path.exists(pyproject_toml_path) and include_build_requirements:
# Read build system from pyproject.toml file: (PEP518)
- with open(os.path.join(output_folder, 'pyproject.toml')) as f:
+ with open(pyproject_toml_path) as f:
build_sys = toml.load(f)['build-system']
if "requires" in build_sys:
requirements += build_sys["requires"]
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index ff34ac29f2..a52abeb029 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1169,7 +1169,7 @@ def get_recipe_env(self, arch, with_flags_in_cc=True):
class PyProjectRecipe(PythonRecipe):
- '''Recipe for projects which containes `pyproject.toml`'''
+ """Recipe for projects which contain `pyproject.toml`"""
# Extra args to pass to `python -m build ...`
extra_build_args = []
diff --git a/pythonforandroid/toolchain.py b/pythonforandroid/toolchain.py
index 1347038b8b..92d13da867 100644
--- a/pythonforandroid/toolchain.py
+++ b/pythonforandroid/toolchain.py
@@ -29,7 +29,7 @@
from pythonforandroid import __version__
from pythonforandroid.bootstrap import Bootstrap
-from pythonforandroid.build import Context, build_recipes
+from pythonforandroid.build import Context, build_recipes, project_has_setup_py
from pythonforandroid.distribution import Distribution, pretty_log_dists
from pythonforandroid.entrypoints import main
from pythonforandroid.graph import get_recipe_order_and_bootstrap
@@ -569,18 +569,18 @@ def add_parser(subparsers, *args, **kwargs):
args, unknown = parser.parse_known_args(sys.argv[1:])
args.unknown_args = unknown
- if hasattr(args, "private") and args.private is not None:
+ if getattr(args, "private", None) is not None:
# Pass this value on to the internal bootstrap build.py:
args.unknown_args += ["--private", args.private]
- if hasattr(args, "build_mode") and args.build_mode == "release":
+ if getattr(args, "build_mode", None) == "release":
args.unknown_args += ["--release"]
- if hasattr(args, "with_debug_symbols") and args.with_debug_symbols:
+ if getattr(args, "with_debug_symbols", False):
args.unknown_args += ["--with-debug-symbols"]
- if hasattr(args, "ignore_setup_py") and args.ignore_setup_py:
+ if getattr(args, "ignore_setup_py", False):
args.use_setup_py = False
- if hasattr(args, "activity_class_name") and args.activity_class_name != 'org.kivy.android.PythonActivity':
+ if getattr(args, "activity_class_name", "org.kivy.android.PythonActivity") != 'org.kivy.android.PythonActivity':
args.unknown_args += ["--activity-class-name", args.activity_class_name]
- if hasattr(args, "service_class_name") and args.service_class_name != 'org.kivy.android.PythonService':
+ if getattr(args, "service_class_name", "org.kivy.android.PythonService") != 'org.kivy.android.PythonService':
args.unknown_args += ["--service-class-name", args.service_class_name]
self.args = args
@@ -603,21 +603,13 @@ def add_parser(subparsers, *args, **kwargs):
args, "with_debug_symbols", False
)
- have_setup_py_or_similar = False
- if getattr(args, "private", None) is not None:
- project_dir = getattr(args, "private")
- if (os.path.exists(os.path.join(project_dir, "setup.py")) or
- os.path.exists(os.path.join(project_dir,
- "pyproject.toml"))):
- have_setup_py_or_similar = True
-
# Process requirements and put version in environ
if hasattr(args, 'requirements'):
requirements = []
# Add dependencies from setup.py, but only if they are recipes
# (because otherwise, setup.py itself will install them later)
- if (have_setup_py_or_similar and
+ if (project_has_setup_py(getattr(args, "private", None)) and
getattr(args, "use_setup_py", False)):
try:
info("Analyzing package dependencies. MAY TAKE A WHILE.")
@@ -698,10 +690,7 @@ def warn_on_deprecated_args(self, args):
# Output warning if setup.py is present and neither --ignore-setup-py
# nor --use-setup-py was specified.
- if getattr(args, "private", None) is not None and \
- (os.path.exists(os.path.join(args.private, "setup.py")) or
- os.path.exists(os.path.join(args.private, "pyproject.toml"))
- ):
+ if project_has_setup_py(getattr(args, "private", None)):
if not getattr(args, "use_setup_py", False) and \
not getattr(args, "ignore_setup_py", False):
warning(" **** FUTURE BEHAVIOR CHANGE WARNING ****")
From 851891d05a6bbabda2293e2cc370c948e5b9e1df Mon Sep 17 00:00:00 2001
From: Jithesh Kuyyalil <jithesh82@gmail.com>
Date: Sun, 25 Aug 2024 05:56:39 -0400
Subject: [PATCH 063/158] aubio recipe created (#3044)
Aubio recipe.
Note that this hasn't been ported to cross compile from macOS yet, the error on 0.4.9 was:
```
src/aubio_priv.h:95:10: fatal error: 'Accelerate/Accelerate.h' file not found
#include <Accelerate/Accelerate.h>
```
---
pythonforandroid/recipes/aubio/__init__.py | 18 ++++++++++++++++++
1 file changed, 18 insertions(+)
create mode 100644 pythonforandroid/recipes/aubio/__init__.py
diff --git a/pythonforandroid/recipes/aubio/__init__.py b/pythonforandroid/recipes/aubio/__init__.py
new file mode 100644
index 0000000000..241a92a23b
--- /dev/null
+++ b/pythonforandroid/recipes/aubio/__init__.py
@@ -0,0 +1,18 @@
+"""
+Aubio recipe.
+Note that this hasn't been ported to cross compile from macOS yet,
+the error on 0.4.9 was: src/aubio_priv.h:95:10:
+fatal error: 'Accelerate/Accelerate.h' file not found
+#include <Accelerate/Accelerate.h>
+"""
+
+from pythonforandroid.recipe import PyProjectRecipe
+
+
+class AubioRecipe(PyProjectRecipe):
+ version = "0.4.9"
+ url = "https://aubio.org/pub/aubio-{version}.tar.bz2"
+ depends = ["numpy", "setuptools"]
+
+
+recipe = AubioRecipe()
From 541fe992ea86b3902f0e0f776167555da6dbca01 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Tue, 24 Sep 2024 14:18:57 -0300
Subject: [PATCH 064/158] bump FFMpegRecipe and PyAVRecipe versions
---
pythonforandroid/recipes/av/__init__.py | 6 ++++-
.../patches/compilation_syntax_errors.patch | 27 +++++++++++++++++++
pythonforandroid/recipes/ffmpeg/__init__.py | 6 +----
3 files changed, 33 insertions(+), 6 deletions(-)
create mode 100644 pythonforandroid/recipes/av/patches/compilation_syntax_errors.patch
diff --git a/pythonforandroid/recipes/av/__init__.py b/pythonforandroid/recipes/av/__init__.py
index 816f27e35f..d5c1acae54 100644
--- a/pythonforandroid/recipes/av/__init__.py
+++ b/pythonforandroid/recipes/av/__init__.py
@@ -5,11 +5,15 @@
class PyAVRecipe(CythonRecipe):
name = "av"
- version = "10.0.0"
+ version = "13.0.0"
url = "https://github.com/PyAV-Org/PyAV/archive/v{version}.zip"
depends = ["python3", "cython", "ffmpeg", "av_codecs"]
opt_depends = ["openssl"]
+ patches = ['patches/compilation_syntax_errors.patch']
+
+ def prebuild_arch(self, arch):
+ self.apply_patches(arch)
def get_recipe_env(self, arch, with_flags_in_cc=True):
env = super().get_recipe_env(arch)
diff --git a/pythonforandroid/recipes/av/patches/compilation_syntax_errors.patch b/pythonforandroid/recipes/av/patches/compilation_syntax_errors.patch
new file mode 100644
index 0000000000..c9a7e9adb3
--- /dev/null
+++ b/pythonforandroid/recipes/av/patches/compilation_syntax_errors.patch
@@ -0,0 +1,27 @@
+diff --git a/av/container/streams.pyx b/av/container/streams.pyx
+index 17e4992..502ac5a 100644
+--- a/av/container/streams.pyx
++++ b/av/container/streams.pyx
+@@ -144,7 +144,7 @@ cdef class StreamContainer:
+
+ return stream_index
+
+- def best(self, str type, /, Stream related = None):
++ def best(self, str type, Stream related=None):
+ """best(type: Literal["video", "audio", "subtitle", "attachment", "data"], /, related: Stream | None)
+ Finds the "best" stream in the file. Wraps :ffmpeg:`av_find_best_stream`. Example::
+
+diff --git a/av/filter/context.pyx b/av/filter/context.pyx
+index b820d3d..8908b56 100644
+--- a/av/filter/context.pyx
++++ b/av/filter/context.pyx
+@@ -77,7 +77,8 @@ cdef class FilterContext:
+
+ @property
+ def graph(self):
+- if (graph := self._graph()):
++ graph = self._graph()
++ if graph:
+ return graph
+ else:
+ raise RuntimeError("graph is unallocated")
diff --git a/pythonforandroid/recipes/ffmpeg/__init__.py b/pythonforandroid/recipes/ffmpeg/__init__.py
index 9414552f0b..c6068bd999 100644
--- a/pythonforandroid/recipes/ffmpeg/__init__.py
+++ b/pythonforandroid/recipes/ffmpeg/__init__.py
@@ -4,20 +4,16 @@
class FFMpegRecipe(Recipe):
- version = 'n4.3.1'
+ version = 'n6.1.2'
# Moved to github.com instead of ffmpeg.org to improve download speed
url = 'https://github.com/FFmpeg/FFmpeg/archive/{version}.zip'
depends = ['sdl2'] # Need this to build correct recipe order
opts_depends = ['openssl', 'ffpyplayer_codecs']
- patches = ['patches/configure.patch']
def should_build(self, arch):
build_dir = self.get_build_dir(arch.arch)
return not exists(join(build_dir, 'lib', 'libavcodec.so'))
- def prebuild_arch(self, arch):
- self.apply_patches(arch)
-
def get_recipe_env(self, arch):
env = super().get_recipe_env(arch)
env['NDK'] = self.ctx.ndk_dir
From c8f7a5cd39d7798d0514762b3f90f1b42b48337a Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Tue, 24 Sep 2024 14:25:55 -0300
Subject: [PATCH 065/158] remove patch no longer needed
---
.../recipes/ffmpeg/patches/configure.patch | 11 -----------
1 file changed, 11 deletions(-)
delete mode 100644 pythonforandroid/recipes/ffmpeg/patches/configure.patch
diff --git a/pythonforandroid/recipes/ffmpeg/patches/configure.patch b/pythonforandroid/recipes/ffmpeg/patches/configure.patch
deleted file mode 100644
index cacf0294e2..0000000000
--- a/pythonforandroid/recipes/ffmpeg/patches/configure.patch
+++ /dev/null
@@ -1,11 +0,0 @@
---- ./configure 2020-10-11 19:12:16.759760904 +0200
-+++ ./configure.patch 2020-10-11 19:15:49.059533563 +0200
-@@ -6361,7 +6361,7 @@
- enabled librsvg && require_pkg_config librsvg librsvg-2.0 librsvg-2.0/librsvg/rsvg.h rsvg_handle_render_cairo
- enabled librtmp && require_pkg_config librtmp librtmp librtmp/rtmp.h RTMP_Socket
- enabled librubberband && require_pkg_config librubberband "rubberband >= 1.8.1" rubberband/rubberband-c.h rubberband_new -lstdc++ && append librubberband_extralibs "-lstdc++"
--enabled libshine && require_pkg_config libshine shine shine/layer3.h shine_encode_buffer
-+enabled libshine && require "shine" shine/layer3.h shine_encode_buffer -lshine -lm
- enabled libsmbclient && { check_pkg_config libsmbclient smbclient libsmbclient.h smbc_init ||
- require libsmbclient libsmbclient.h smbc_init -lsmbclient; }
- enabled libsnappy && require libsnappy snappy-c.h snappy_compress -lsnappy -lstdc++
\ No newline at end of file
From 63fc70324d3afddc58b1d842497b9e8dca3539de Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Thu, 26 Sep 2024 11:29:00 -0300
Subject: [PATCH 066/158] re-add patch; fix linking with static libx264
---
pythonforandroid/recipes/ffmpeg/__init__.py | 9 +++++---
.../recipes/ffmpeg/patches/configure.patch | 22 +++++++++++++++++++
2 files changed, 28 insertions(+), 3 deletions(-)
create mode 100644 pythonforandroid/recipes/ffmpeg/patches/configure.patch
diff --git a/pythonforandroid/recipes/ffmpeg/__init__.py b/pythonforandroid/recipes/ffmpeg/__init__.py
index c6068bd999..90d64bfb5e 100644
--- a/pythonforandroid/recipes/ffmpeg/__init__.py
+++ b/pythonforandroid/recipes/ffmpeg/__init__.py
@@ -8,7 +8,8 @@ class FFMpegRecipe(Recipe):
# Moved to github.com instead of ffmpeg.org to improve download speed
url = 'https://github.com/FFmpeg/FFmpeg/archive/{version}.zip'
depends = ['sdl2'] # Need this to build correct recipe order
- opts_depends = ['openssl', 'ffpyplayer_codecs']
+ opts_depends = ['openssl', 'ffpyplayer_codecs', 'av_codecs']
+ patches = ['patches/configure.patch']
def should_build(self, arch):
build_dir = self.get_build_dir(arch.arch)
@@ -39,7 +40,9 @@ def build_arch(self, arch):
'-DOPENSSL_API_COMPAT=0x10002000L']
ldflags += ['-L' + build_dir]
- if 'ffpyplayer_codecs' in self.ctx.recipe_build_order:
+ codecs_opts = {"ffpyplayer_codecs", "av_codecs"}
+ if codecs_opts.intersection(self.ctx.recipe_build_order):
+
# Enable GPL
flags += ['--enable-gpl']
@@ -48,7 +51,7 @@ def build_arch(self, arch):
build_dir = Recipe.get_recipe(
'libx264', self.ctx).get_build_dir(arch.arch)
cflags += ['-I' + build_dir + '/include/']
- ldflags += ['-lx264', '-L' + build_dir + '/lib/']
+ ldflags += [build_dir + '/lib/' + 'libx264.a']
# libshine
flags += ['--enable-libshine']
diff --git a/pythonforandroid/recipes/ffmpeg/patches/configure.patch b/pythonforandroid/recipes/ffmpeg/patches/configure.patch
new file mode 100644
index 0000000000..e274359cb7
--- /dev/null
+++ b/pythonforandroid/recipes/ffmpeg/patches/configure.patch
@@ -0,0 +1,22 @@
+diff --git a/configure b/configure
+index 5af693c954..d1d0a4f0a2 100755
+--- a/configure
++++ b/configure
+@@ -6800,7 +6800,7 @@ enabled librsvg && require_pkg_config librsvg librsvg-2.0 librsvg-2.0/
+ enabled librtmp && require_pkg_config librtmp librtmp librtmp/rtmp.h RTMP_Socket
+ enabled librubberband && require_pkg_config librubberband "rubberband >= 1.8.1" rubberband/rubberband-c.h rubberband_new -lstdc++ && append librubberband_extralibs "-lstdc++"
+ enabled libshaderc && require_pkg_config spirv_compiler "shaderc >= 2019.1" shaderc/shaderc.h shaderc_compiler_initialize
+-enabled libshine && require_pkg_config libshine shine shine/layer3.h shine_encode_buffer
++enabled libshine && require "shine" shine/layer3.h shine_encode_buffer -lshine -lm
+ enabled libsmbclient && { check_pkg_config libsmbclient smbclient libsmbclient.h smbc_init ||
+ require libsmbclient libsmbclient.h smbc_init -lsmbclient; }
+ enabled libsnappy && require libsnappy snappy-c.h snappy_compress -lsnappy -lstdc++
+@@ -6850,7 +6850,7 @@ enabled libvpx && {
+ enabled libwebp && {
+ enabled libwebp_encoder && require_pkg_config libwebp "libwebp >= 0.2.0" webp/encode.h WebPGetEncoderVersion
+ enabled libwebp_anim_encoder && check_pkg_config libwebp_anim_encoder "libwebpmux >= 0.4.0" webp/mux.h WebPAnimEncoderOptionsInit; }
+-enabled libx264 && require_pkg_config libx264 x264 "stdint.h x264.h" x264_encoder_encode &&
++enabled libx264 && require "x264" "stdint.h x264.h" x264_encoder_encode &&
+ require_cpp_condition libx264 x264.h "X264_BUILD >= 122" && {
+ [ "$toolchain" != "msvc" ] ||
+ require_cpp_condition libx264 x264.h "X264_BUILD >= 158"; } &&
From 41077f3c3c3faca739bfcc7db777f9a537a84801 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Thu, 26 Sep 2024 11:30:30 -0300
Subject: [PATCH 067/158] remove unneeded override
---
pythonforandroid/recipes/av/__init__.py | 3 ---
1 file changed, 3 deletions(-)
diff --git a/pythonforandroid/recipes/av/__init__.py b/pythonforandroid/recipes/av/__init__.py
index d5c1acae54..70205e37a0 100644
--- a/pythonforandroid/recipes/av/__init__.py
+++ b/pythonforandroid/recipes/av/__init__.py
@@ -12,9 +12,6 @@ class PyAVRecipe(CythonRecipe):
opt_depends = ["openssl"]
patches = ['patches/compilation_syntax_errors.patch']
- def prebuild_arch(self, arch):
- self.apply_patches(arch)
-
def get_recipe_env(self, arch, with_flags_in_cc=True):
env = super().get_recipe_env(arch)
From 18b4e4650e6f1305e02e8c71e06f2c5f6d39c504 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Thu, 26 Sep 2024 23:18:16 -0300
Subject: [PATCH 068/158] add comment
---
pythonforandroid/recipes/ffmpeg/__init__.py | 2 ++
1 file changed, 2 insertions(+)
diff --git a/pythonforandroid/recipes/ffmpeg/__init__.py b/pythonforandroid/recipes/ffmpeg/__init__.py
index 90d64bfb5e..3bc824834f 100644
--- a/pythonforandroid/recipes/ffmpeg/__init__.py
+++ b/pythonforandroid/recipes/ffmpeg/__init__.py
@@ -51,6 +51,8 @@ def build_arch(self, arch):
build_dir = Recipe.get_recipe(
'libx264', self.ctx).get_build_dir(arch.arch)
cflags += ['-I' + build_dir + '/include/']
+ # Newer versions of FFmpeg prioritize the dynamic library and ignore
+ # the static one, unless the static library path is explicitly set.
ldflags += [build_dir + '/lib/' + 'libx264.a']
# libshine
From 369ff57b50f9ca4f1451d4131aad1bfe707b69d1 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Fri, 27 Sep 2024 23:10:18 +0200
Subject: [PATCH 069/158] :construction_worker: Docker workflow fixes
- fix `github.ref` for DockerHub publish on push to develop
- fix docker login to handle special characters in the token
- publish docker tag to DockerHub on git tagging
---
.github/workflows/docker.yml | 13 +++++++++++--
Makefile | 8 ++++++--
2 files changed, 17 insertions(+), 4 deletions(-)
diff --git a/.github/workflows/docker.yml b/.github/workflows/docker.yml
index bd918a9c74..4912e6205c 100644
--- a/.github/workflows/docker.yml
+++ b/.github/workflows/docker.yml
@@ -5,6 +5,8 @@ on:
push:
branches:
- develop
+ tags:
+ - "*"
pull_request:
jobs:
@@ -14,11 +16,18 @@ jobs:
- uses: actions/checkout@v4
- uses: docker/setup-buildx-action@v3
- run: make docker/build
+ - name: docker login
+ if: github.ref == 'refs/heads/develop' || startsWith(github.ref, 'refs/tags/')
env:
DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }}
DOCKERHUB_TOKEN: ${{ secrets.DOCKERHUB_TOKEN }}
+ run: make docker/login
- name: docker push
- if: github.ref == 'develop'
+ if: github.ref == 'refs/heads/develop'
+ run: make docker/push
+ run: echo make docker/push
+ - name: docker push (tag)
+ if: startsWith(github.ref, 'refs/tags/')
run: |
- make docker/login
+ make docker/tag DOCKER_TAG=${GITHUB_REF#refs/tags/}
make docker/push
diff --git a/Makefile b/Makefile
index 2cf08b7537..67d71e7bcb 100644
--- a/Makefile
+++ b/Makefile
@@ -4,6 +4,7 @@ TOX=`which tox`
ACTIVATE=$(VIRTUAL_ENV)/bin/activate
PYTHON=$(VIRTUAL_ENV)/bin/python
DOCKER_IMAGE=kivy/python-for-android
+DOCKER_TAG=latest
ANDROID_SDK_HOME ?= $(HOME)/.android/android-sdk
ANDROID_NDK_HOME ?= $(HOME)/.android/android-ndk
ANDROID_NDK_HOME_LEGACY ?= $(HOME)/.android/android-ndk-legacy
@@ -118,10 +119,13 @@ docker/build:
docker build --cache-from=$(DOCKER_IMAGE) --tag=$(DOCKER_IMAGE) .
docker/login:
- @echo $(DOCKERHUB_TOKEN) | docker login --username $(DOCKERHUB_USERNAME) --password-stdin
+ @echo $$DOCKERHUB_TOKEN | docker login --username $(DOCKERHUB_USERNAME) --password-stdin
+
+docker/tag:
+ docker tag $(DOCKER_IMAGE):latest $(DOCKER_IMAGE):$(DOCKER_TAG)
docker/push:
- docker push $(DOCKER_IMAGE)
+ docker push $(DOCKER_IMAGE):$(DOCKER_TAG)
docker/run/test: docker/build
docker run --rm --env-file=.env $(DOCKER_IMAGE) 'make test'
From 2a4dd96a135c542181bc8c0b9bafc3c8e3835434 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Tue, 1 Oct 2024 14:25:17 +0200
Subject: [PATCH 070/158] :bug: Docker workflow fix
Follow up from #3067
---
.github/workflows/docker.yml | 1 -
1 file changed, 1 deletion(-)
diff --git a/.github/workflows/docker.yml b/.github/workflows/docker.yml
index 4912e6205c..903737eff9 100644
--- a/.github/workflows/docker.yml
+++ b/.github/workflows/docker.yml
@@ -25,7 +25,6 @@ jobs:
- name: docker push
if: github.ref == 'refs/heads/develop'
run: make docker/push
- run: echo make docker/push
- name: docker push (tag)
if: startsWith(github.ref, 'refs/tags/')
run: |
From c4b3314768f2604af3aeed6ee1ac75a3e1f34372 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Tue, 8 Oct 2024 09:24:05 -0300
Subject: [PATCH 071/158] bump pyav to 13.1.0
---
pythonforandroid/recipes/av/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/av/__init__.py b/pythonforandroid/recipes/av/__init__.py
index 70205e37a0..f189467add 100644
--- a/pythonforandroid/recipes/av/__init__.py
+++ b/pythonforandroid/recipes/av/__init__.py
@@ -5,7 +5,7 @@
class PyAVRecipe(CythonRecipe):
name = "av"
- version = "13.0.0"
+ version = "13.1.0"
url = "https://github.com/PyAV-Org/PyAV/archive/v{version}.zip"
depends = ["python3", "cython", "ffmpeg", "av_codecs"]
From 1068a8141aefbf0c62e0db556caf90556b89930a Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Thu, 17 Oct 2024 18:01:45 -0700
Subject: [PATCH 072/158] add support for private github repos
---
pythonforandroid/recipe.py | 13 +++++++++++++
1 file changed, 13 insertions(+)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index a52abeb029..699db55d0c 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -59,6 +59,11 @@ class Recipe(metaclass=RecipeMeta):
if you want.
'''
+ _github_access_token = None
+ '''Used to access a private git repository. Specify the github-supplied
+ access token in order to download the private repository files.
+ '''
+
_version = None
'''A string giving the version of the software the recipe describes,
e.g. ``2.0.3`` or ``master``.'''
@@ -170,6 +175,11 @@ def versioned_url(self):
return None
return self.url.format(version=self.version)
+ @property
+ def github_access_token(self):
+ key = "GITHUB_ACCESS_TOKEN_" + self.name
+ return environ.get(key, self._github_access_token)
+
def download_file(self, url, target, cwd=None):
"""
(internal) Download an ``url`` to a ``target``.
@@ -205,6 +215,9 @@ def report_hook(index, blksize, size):
# jqueryui.com returns a 403 w/ the default user agent
# Mozilla/5.0 doesnt handle redirection for liblzma
url_opener.addheaders = [('User-agent', 'Wget/1.0')]
+ if self.github_access_token:
+ url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'),
+ ('Accept', 'application/vnd.github.v3.raw')]
urlretrieve(url, target, report_hook)
except OSError as e:
attempts += 1
From 9f26b7a1e4fe61817bbba0dfbfe5fa5a9935835d Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Thu, 17 Oct 2024 18:18:54 -0700
Subject: [PATCH 073/158] formatting
---
pythonforandroid/recipe.py | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 699db55d0c..1d966d36d3 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -216,8 +216,7 @@ def report_hook(index, blksize, size):
# Mozilla/5.0 doesnt handle redirection for liblzma
url_opener.addheaders = [('User-agent', 'Wget/1.0')]
if self.github_access_token:
- url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'),
- ('Accept', 'application/vnd.github.v3.raw')]
+ url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'), ('Accept', 'application/vnd.github.v3.raw')]
urlretrieve(url, target, report_hook)
except OSError as e:
attempts += 1
From 8f1d40f50b413afa363677d1f0250d6f81079947 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Thu, 17 Oct 2024 20:18:57 -0700
Subject: [PATCH 074/158] use application/vnd.github+json
---
pythonforandroid/recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 1d966d36d3..fe2cd6b416 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -216,7 +216,7 @@ def report_hook(index, blksize, size):
# Mozilla/5.0 doesnt handle redirection for liblzma
url_opener.addheaders = [('User-agent', 'Wget/1.0')]
if self.github_access_token:
- url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'), ('Accept', 'application/vnd.github.v3.raw')]
+ url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'), ('Accept', 'application/vnd.github+json')]
urlretrieve(url, target, report_hook)
except OSError as e:
attempts += 1
From b3d239eadd5613fbf950acdf104318d75f865d18 Mon Sep 17 00:00:00 2001
From: Christian Clauss <cclauss@me.com>
Date: Sun, 20 Oct 2024 12:40:53 +0200
Subject: [PATCH 075/158] GitHub Actions: Disable the failing sdl2_scipy test
This test was added in #2617 but is now [failing our pull requests](https://github.com/kivy/python-for-android/pulls).
* #2617
@mzakharo, @misl6, @AndreMiras
---
.github/workflows/push.yml | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 0cbad15afe..89cbb56c51 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -64,8 +64,8 @@ jobs:
bootstrap:
- name: sdl2
target: testapps-with-numpy
- - name: sdl2_scipy
- target: testapps-with-scipy
+ # - name: sdl2_scipy # TODO: Re-enable this failing test.
+ # target: testapps-with-scipy
- name: webview
target: testapps-webview
- name: service_library
From d009d0f7a62775f09167cd3b9091ea4a5f2d3c7c Mon Sep 17 00:00:00 2001
From: Christian Clauss <cclauss@me.com>
Date: Sun, 20 Oct 2024 17:43:14 +0200
Subject: [PATCH 076/158] Fix typos discovered by codespell
---
CONTRIBUTING.md | 2 +-
README.md | 4 ++--
doc/source/buildoptions.rst | 2 +-
doc/source/index.rst | 4 ++--
doc/source/services.rst | 2 +-
pythonforandroid/bootstraps/common/build/build.py | 6 +++---
.../bootstraps/common/build/jni/application/src/start.c | 2 +-
.../build/src/main/java/org/kamranzafar/jtar/TarHeader.java | 2 +-
.../build/src/main/java/org/kivy/android/PythonService.java | 2 +-
.../build/src/main/java/org/renpy/android/Hardware.java | 2 +-
.../bootstraps/service_only/build/blacklist.txt | 2 +-
pythonforandroid/bootstraps/webview/build/blacklist.txt | 2 +-
pythonforandroid/graph.py | 2 +-
pythonforandroid/pythonpackage.py | 2 +-
pythonforandroid/recipe.py | 6 +++---
pythonforandroid/recipes/android/src/android/broadcast.py | 2 +-
pythonforandroid/recipes/android/src/android/permissions.py | 2 +-
pythonforandroid/recipes/ffpyplayer/__init__.py | 2 +-
pythonforandroid/recipes/libxml2/glob.c | 2 +-
pythonforandroid/recipes/libxml2/glob.h | 2 +-
pythonforandroid/recipes/opencv/__init__.py | 2 +-
tests/test_build.py | 2 +-
22 files changed, 28 insertions(+), 28 deletions(-)
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index be864e06f0..2e118304bd 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -178,7 +178,7 @@ packages:
of the project will be run. This happens with cross compilation set up
(`CC`/`CFLAGS`/... set to use the
proper toolchain) and a custom site-packages location.
- The actual comand is a simple `pip install .` in the project folder
+ The actual command is a simple `pip install .` in the project folder
with some extra options: e.g. all dependencies that were already
installed by recipes will be pinned with a `-c` constraints file
to make sure pip won't install them, and build isolation will be
diff --git a/README.md b/README.md
index c7175e6a28..0c4cb57a6b 100644
--- a/README.md
+++ b/README.md
@@ -12,7 +12,7 @@ It can generate:
* [Android App Bundle](https://developer.android.com/guide/app-bundle/faq)
(AAB) files which can be shared on [Google Play Store](https://play.google.com/store/).
* [Android Archive](https://developer.android.com/studio/projects/android-library)
- (AAR) files which can be used as a re-usable bundle of resources for other
+ (AAR) files which can be used as a reusable bundle of resources for other
projects.
It supports multiple CPU architectures.
@@ -26,7 +26,7 @@ a Python web server.
It automatically supports dependencies on most pure Python packages. For other
packages, including those that depend on C code, a special "recipe" must be
written to support cross-compiling. python-for-android comes with recipes for
-many of the mosty popular libraries (e.g. numpy and sqlalchemy) built in.
+many of the mostly popular libraries (e.g. numpy and sqlalchemy) built in.
python-for-android works by cross-compiling the Python interpreter and its
dependencies for Android devices, and bundling it with the app's python code
diff --git a/doc/source/buildoptions.rst b/doc/source/buildoptions.rst
index 977516ba25..167f4fd718 100644
--- a/doc/source/buildoptions.rst
+++ b/doc/source/buildoptions.rst
@@ -250,7 +250,7 @@ across all the other bootstraps, the Qt bootstrap introduces the following 3 new
These build options are automatically populated by the ``pyside6-android-deploy`` tool, but can be
modified by updating the ``buildozer.spec`` file. Apart from the above 3 build options, the tool
also automatically identifies the values to be fed into the cli options ``--permission``, ``--add-jar``
-depending on the PySide6 modules used by the applicaiton.
+depending on the PySide6 modules used by the application.
Requirements blacklist (APK size optimization)
----------------------------------------------
diff --git a/doc/source/index.rst b/doc/source/index.rst
index 929fa384c7..24de1bc80c 100644
--- a/doc/source/index.rst
+++ b/doc/source/index.rst
@@ -13,7 +13,7 @@ It can generate:
* `Android App Bundle <https://developer.android.com/guide/app-bundle/faq>`_
(AAB) files which can be shared on `Google Play Store <https://play.google.com/store/>`_.
* `Android Archive <https://developer.android.com/studio/projects/android-library>`_
- (AAR) files which can be used as a re-usable bundle of resources for other
+ (AAR) files which can be used as a reusable bundle of resources for other
projects.
It supports multiple CPU architectures.
@@ -27,7 +27,7 @@ a Python web server.
It automatically supports dependencies on most pure Python packages. For other
packages, including those that depend on C code, a special "recipe" must be
written to support cross-compiling. python-for-android comes with recipes for
-many of the mosty popular libraries (e.g. numpy and sqlalchemy) built in.
+many of the mostly popular libraries (e.g. numpy and sqlalchemy) built in.
python-for-android works by cross-compiling the Python interpreter and its
dependencies for Android devices, and bundling it with the app's python code
diff --git a/doc/source/services.rst b/doc/source/services.rst
index c71b035a76..99abdba561 100644
--- a/doc/source/services.rst
+++ b/doc/source/services.rst
@@ -89,7 +89,7 @@ of your APK.
If you are using buildozer, the identifier is set by the ``package.name``
and ``package.domain`` values in your buildozer.spec file.
The name of the service is ``ServiceMyservice``, where ``Myservice``
-is the name specied by one of the ``services`` values, but with the first
+is the name specified by one of the ``services`` values, but with the first
letter upper case.
If you are using python-for-android directly, the identifier is set by the ``--package``
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index 51e0a5fa94..94382049bf 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -55,7 +55,7 @@ def get_bootstrap_name():
curdir = dirname(__file__)
BLACKLIST_PATTERNS = [
- # code versionning
+ # code versioning
'^*.hg/*',
'^*.git/*',
'^*.bzr/*',
@@ -170,7 +170,7 @@ def clean(tinfo):
files.append((fn, relpath(realpath(fn), sd)))
files.sort() # deterministic
- # create tar.gz of thoses files
+ # create tar.gz of those files
gf = GzipFile(tfn, 'wb', mtime=0) # deterministic
tf = tarfile.open(None, 'w', gf, format=tarfile.USTAR_FORMAT)
dirs = []
@@ -339,7 +339,7 @@ def make_package(args):
else:
shutil.copytree(res_dir, res_dir_initial)
- # Add user resouces
+ # Add user resources
for resource in args.resources:
resource_src, resource_dest = resource.split(":")
if isfile(realpath(resource_src)):
diff --git a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
index ce93ca27fd..88999faf98 100644
--- a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
+++ b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
@@ -417,7 +417,7 @@ void Java_org_kivy_android_PythonActivity_nativeInit(JNIEnv* env, jclass cls, jo
{
/* This nativeInit follows SDL2 */
- /* This interface could expand with ABI negotiation, calbacks, etc. */
+ /* This interface could expand with ABI negotiation, callbacks, etc. */
/* SDL_Android_Init(env, cls); */
/* SDL_SetMainReady(); */
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kamranzafar/jtar/TarHeader.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kamranzafar/jtar/TarHeader.java
index b9d3a86bef..0e0be2cf3a 100755
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kamranzafar/jtar/TarHeader.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kamranzafar/jtar/TarHeader.java
@@ -48,7 +48,7 @@
* '4' Block special
* '5' Directory
* '6' FIFO
- * '7' Contigous
+ * '7' Contiguous
* </pre>
*
*
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonService.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonService.java
index 76d3b2e77b..f28946d501 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonService.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonService.java
@@ -180,7 +180,7 @@ public void onDestroy() {
@Override
public void onTaskRemoved(Intent rootIntent) {
super.onTaskRemoved(rootIntent);
- //sticky servcie runtime/restart is managed by the OS. leave it running when app is closed
+ //sticky service runtime/restart is managed by the OS. leave it running when app is closed
if (startType() != START_STICKY) {
stopSelf();
}
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/renpy/android/Hardware.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/renpy/android/Hardware.java
index 847576282e..8ed165233d 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/renpy/android/Hardware.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/renpy/android/Hardware.java
@@ -260,7 +260,7 @@ public static boolean checkNetwork()
}
/**
- * To recieve network state changes
+ * To receive network state changes
*/
public static void registerNetworkCheck()
{
diff --git a/pythonforandroid/bootstraps/service_only/build/blacklist.txt b/pythonforandroid/bootstraps/service_only/build/blacklist.txt
index 53cc634b7d..64e2598722 100644
--- a/pythonforandroid/bootstraps/service_only/build/blacklist.txt
+++ b/pythonforandroid/bootstraps/service_only/build/blacklist.txt
@@ -84,7 +84,7 @@ lib-dynload/_testcapi.so
plat-linux3/regen
#>sqlite3
-# conditionnal include depending if some recipes are included or not.
+# conditional include depending if some recipes are included or not.
sqlite3/*
lib-dynload/_sqlite3.so
#<sqlite3
diff --git a/pythonforandroid/bootstraps/webview/build/blacklist.txt b/pythonforandroid/bootstraps/webview/build/blacklist.txt
index 53cc634b7d..64e2598722 100644
--- a/pythonforandroid/bootstraps/webview/build/blacklist.txt
+++ b/pythonforandroid/bootstraps/webview/build/blacklist.txt
@@ -84,7 +84,7 @@ lib-dynload/_testcapi.so
plat-linux3/regen
#>sqlite3
-# conditionnal include depending if some recipes are included or not.
+# conditional include depending if some recipes are included or not.
sqlite3/*
lib-dynload/_sqlite3.so
#<sqlite3
diff --git a/pythonforandroid/graph.py b/pythonforandroid/graph.py
index 4edb8f4c90..4f8866a805 100644
--- a/pythonforandroid/graph.py
+++ b/pythonforandroid/graph.py
@@ -264,7 +264,7 @@ def get_recipe_order_and_bootstrap(ctx, names, bs=None, blacklist=None):
names.append(cleaned_up_tuple)
# Do check for obvious conflicts (that would trigger in any order, and
- # without comitting to any specific choice in a multi-choice tuple of
+ # without committing to any specific choice in a multi-choice tuple of
# dependencies):
obvious_conflict_checker(ctx, names, blacklist=blacklist)
# If we get here, no obvious conflicts!
diff --git a/pythonforandroid/pythonpackage.py b/pythonforandroid/pythonpackage.py
index 602235cf23..0649d8848a 100644
--- a/pythonforandroid/pythonpackage.py
+++ b/pythonforandroid/pythonpackage.py
@@ -79,7 +79,7 @@ def extract_metainfo_files_from_package(
output_folder,
debug=False
):
- """ Extracts metdata files from the given package to the given folder,
+ """ Extracts metadata files from the given package to the given folder,
which may be referenced in any way that is permitted in
a requirements.txt file or install_requires=[] listing.
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index a52abeb029..7d99131413 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -203,7 +203,7 @@ def report_hook(index, blksize, size):
while True:
try:
# jqueryui.com returns a 403 w/ the default user agent
- # Mozilla/5.0 doesnt handle redirection for liblzma
+ # Mozilla/5.0 does not handle redirection for liblzma
url_opener.addheaders = [('User-agent', 'Wget/1.0')]
urlretrieve(url, target, report_hook)
except OSError as e:
@@ -729,7 +729,7 @@ def prepare_build_dir(self, arch):
class BootstrapNDKRecipe(Recipe):
'''A recipe class for recipes built in an Android project jni dir with
- an Android.mk. These are not cached separatly, but built in the
+ an Android.mk. These are not cached separately, but built in the
bootstrap's own building directory.
To build an NDK project which is not part of the bootstrap, see
@@ -1233,7 +1233,7 @@ def build_arch(self, arch):
)
build_dir = self.get_build_dir(arch.arch)
env = self.get_recipe_env(arch, with_flags_in_cc=True)
- # make build dir separatly
+ # make build dir separately
sub_build_dir = join(build_dir, "p4a_android_build")
ensure_dir(sub_build_dir)
# copy hostpython to built python to ensure correct selection of libs and includes
diff --git a/pythonforandroid/recipes/android/src/android/broadcast.py b/pythonforandroid/recipes/android/src/android/broadcast.py
index 3232d83bbf..750e9c87ef 100644
--- a/pythonforandroid/recipes/android/src/android/broadcast.py
+++ b/pythonforandroid/recipes/android/src/android/broadcast.py
@@ -32,7 +32,7 @@ def _expand_partial_name(partial_name):
else:
name = 'ACTION_{}'.format(partial_name.upper())
if not hasattr(Intent, name):
- raise Exception('The intent {} doesnt exist'.format(name))
+ raise Exception('The intent {} does not exist'.format(name))
return getattr(Intent, name)
# resolve actions/categories first
diff --git a/pythonforandroid/recipes/android/src/android/permissions.py b/pythonforandroid/recipes/android/src/android/permissions.py
index 132aab0823..7baa15af0e 100644
--- a/pythonforandroid/recipes/android/src/android/permissions.py
+++ b/pythonforandroid/recipes/android/src/android/permissions.py
@@ -590,7 +590,7 @@ def request_permissions(permissions, callback=None):
See Android documentation for onPermissionsCallbackResult for
further information.
- Note that if the request is interupted the callback may contain an empty
+ Note that if the request is interrupted the callback may contain an empty
list of permissions, without permissions being granted; the App should
check that each permission requested has been granted.
diff --git a/pythonforandroid/recipes/ffpyplayer/__init__.py b/pythonforandroid/recipes/ffpyplayer/__init__.py
index a78f65d8c6..1198ff3e9c 100644
--- a/pythonforandroid/recipes/ffpyplayer/__init__.py
+++ b/pythonforandroid/recipes/ffpyplayer/__init__.py
@@ -32,7 +32,7 @@ def get_recipe_env(self, arch, with_flags_in_cc=True):
env['LIBLINK'] = 'NOTNONE'
# ffmpeg recipe enables GPL components only if ffpyplayer_codecs recipe used.
- # Therefor we need to disable libpostproc if skipped.
+ # Therefore we need to disable libpostproc if skipped.
if 'ffpyplayer_codecs' not in self.ctx.recipe_build_order:
env["CONFIG_POSTPROC"] = '0'
diff --git a/pythonforandroid/recipes/libxml2/glob.c b/pythonforandroid/recipes/libxml2/glob.c
index cec80ed7ca..b0ff6d0525 100644
--- a/pythonforandroid/recipes/libxml2/glob.c
+++ b/pythonforandroid/recipes/libxml2/glob.c
@@ -683,7 +683,7 @@ glob3(Char *pathbuf, Char *pathend, Char *pathend_last,
/*
- * Extend the gl_pathv member of a glob_t structure to accomodate a new item,
+ * Extend the gl_pathv member of a glob_t structure to accommodate a new item,
* add the new item, and update gl_pathc.
*
* This assumes the BSD realloc, which only copies the block when its size
diff --git a/pythonforandroid/recipes/libxml2/glob.h b/pythonforandroid/recipes/libxml2/glob.h
index 351b6c46bb..08cdf8d58a 100644
--- a/pythonforandroid/recipes/libxml2/glob.h
+++ b/pythonforandroid/recipes/libxml2/glob.h
@@ -80,7 +80,7 @@ typedef struct {
#define GLOB_NOSPACE (-1) /* Malloc call failed. */
#define GLOB_ABORTED (-2) /* Unignored error. */
#define GLOB_NOMATCH (-3) /* No match and GLOB_NOCHECK was not set. */
-#define GLOB_NOSYS (-4) /* Obsolete: source comptability only. */
+#define GLOB_NOSYS (-4) /* Obsolete: source compatibility only. */
#endif /* __POSIX_VISIBLE >= 199209 */
#if __BSD_VISIBLE
diff --git a/pythonforandroid/recipes/opencv/__init__.py b/pythonforandroid/recipes/opencv/__init__.py
index 650c77e508..8f7e2b8c07 100644
--- a/pythonforandroid/recipes/opencv/__init__.py
+++ b/pythonforandroid/recipes/opencv/__init__.py
@@ -90,7 +90,7 @@ def build_arch(self, arch):
version=python_link_version),
'-DBUILD_WITH_STANDALONE_TOOLCHAIN=ON',
- # Force to build as shared libraries the cv2's dependant
+ # Force to build as shared libraries the cv2's dependent
# libs or we will not be able to link with our python
'-DBUILD_SHARED_LIBS=ON',
'-DBUILD_STATIC_LIBS=OFF',
diff --git a/tests/test_build.py b/tests/test_build.py
index 7556d68bbc..49f6311621 100644
--- a/tests/test_build.py
+++ b/tests/test_build.py
@@ -51,7 +51,7 @@ def test_strip_if_with_debug_symbols(self):
assert m_CythonRecipe().strip_object_files.called is False
# Make sure strip object files IS called when
- # `with_debug_symbols` is fasle:
+ # `with_debug_symbols` is false:
ctx.with_debug_symbols = False
assert run_pymodules_install(ctx, ctx.archs[0], modules, project_dir) is None
assert m_CythonRecipe().strip_object_files.called is True
From a8fb08042f6f9792c14d437350b95a35b3decee8 Mon Sep 17 00:00:00 2001
From: Christian Clauss <cclauss@me.com>
Date: Sun, 20 Oct 2024 17:48:18 +0200
Subject: [PATCH 077/158] mostly --> most x 2
---
README.md | 2 +-
doc/source/index.rst | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 0c4cb57a6b..21cf3bc3ee 100644
--- a/README.md
+++ b/README.md
@@ -26,7 +26,7 @@ a Python web server.
It automatically supports dependencies on most pure Python packages. For other
packages, including those that depend on C code, a special "recipe" must be
written to support cross-compiling. python-for-android comes with recipes for
-many of the mostly popular libraries (e.g. numpy and sqlalchemy) built in.
+many of the most popular libraries (e.g. numpy and sqlalchemy) built in.
python-for-android works by cross-compiling the Python interpreter and its
dependencies for Android devices, and bundling it with the app's python code
diff --git a/doc/source/index.rst b/doc/source/index.rst
index 24de1bc80c..383e185d95 100644
--- a/doc/source/index.rst
+++ b/doc/source/index.rst
@@ -27,7 +27,7 @@ a Python web server.
It automatically supports dependencies on most pure Python packages. For other
packages, including those that depend on C code, a special "recipe" must be
written to support cross-compiling. python-for-android comes with recipes for
-many of the mostly popular libraries (e.g. numpy and sqlalchemy) built in.
+many of the most popular libraries (e.g. numpy and sqlalchemy) built in.
python-for-android works by cross-compiling the Python interpreter and its
dependencies for Android devices, and bundling it with the app's python code
From 254440db5c293d9d5cf8ec0da04a1e1e9bc00b2a Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Sun, 20 Oct 2024 10:26:15 -0700
Subject: [PATCH 078/158] remove .decode() on string. Perhaps code that was
never migrated from python 2.7?
---
pythonforandroid/toolchain.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/toolchain.py b/pythonforandroid/toolchain.py
index 92d13da867..e05bb3a8fe 100644
--- a/pythonforandroid/toolchain.py
+++ b/pythonforandroid/toolchain.py
@@ -1023,7 +1023,7 @@ def _build_package(self, args, package_type):
# .../build/bootstrap_builds/sdl2-python3/gradlew
# if docker on windows, gradle contains CRLF
output = shprint(
- sh.Command('dos2unix'), gradlew._path.decode('utf8'),
+ sh.Command('dos2unix'), gradlew._path,
_tail=20, _critical=True, _env=env
)
if args.build_mode == "debug":
From e4ade43f942f1fe7c6bd227c5d6e9179eb1800e1 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Sun, 20 Oct 2024 11:21:18 -0700
Subject: [PATCH 079/158] switch to generically specifying additional headers.
Improved documentation
---
doc/source/recipes.rst | 12 ++++++++++++
pythonforandroid/recipe.py | 23 +++++++++++++++--------
2 files changed, 27 insertions(+), 8 deletions(-)
diff --git a/doc/source/recipes.rst b/doc/source/recipes.rst
index 0a2a736592..ea0ff6c174 100644
--- a/doc/source/recipes.rst
+++ b/doc/source/recipes.rst
@@ -54,10 +54,22 @@ omitted if the source is somehow loaded from elsewhere.
You must include ``recipe = YourRecipe()``. This variable is accessed
when the recipe is imported.
+Specifying the URL
+------------------
+
.. note:: The url includes the ``{version}`` tag. You should only
access the url with the ``versioned_url`` property, which
replaces this with the version attribute.
+.. note:: you may need to specify additional headers to allow python-for-android
+ to download the archive. Specify your additional headers by setting the
+ download_headers property.
+
+For example, when downloading from a private github repository, you can specify the following:
+```
+[('Authorization', 'token <your personal access token>'), ('Accept', 'application/vnd.github+json')]
+```
+
The actual build process takes place via three core methods::
def prebuild_arch(self, arch):
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index fe2cd6b416..ad7dbab1a5 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -59,9 +59,16 @@ class Recipe(metaclass=RecipeMeta):
if you want.
'''
- _github_access_token = None
- '''Used to access a private git repository. Specify the github-supplied
- access token in order to download the private repository files.
+ _download_headers = None
+ '''Add additional headers used when downloading the package, typically
+ for authorization purposes.
+
+ Specified as an array of tuples:
+ [("header name", "header value")]
+
+ For example, when downloading from a private
+ github repository, you can specify the following:
+ [('Authorization', 'token <your personal access token>'), ('Accept', 'application/vnd.github+json')]
'''
_version = None
@@ -176,9 +183,9 @@ def versioned_url(self):
return self.url.format(version=self.version)
@property
- def github_access_token(self):
- key = "GITHUB_ACCESS_TOKEN_" + self.name
- return environ.get(key, self._github_access_token)
+ def download_headers(self):
+ key = "DOWNLOAD_HEADERS_" + self.name
+ return environ.get(key, self._download_headers)
def download_file(self, url, target, cwd=None):
"""
@@ -215,8 +222,8 @@ def report_hook(index, blksize, size):
# jqueryui.com returns a 403 w/ the default user agent
# Mozilla/5.0 doesnt handle redirection for liblzma
url_opener.addheaders = [('User-agent', 'Wget/1.0')]
- if self.github_access_token:
- url_opener.addheaders += [('Authorization', f'token {self.github_access_token}'), ('Accept', 'application/vnd.github+json')]
+ if self.download_headers:
+ url_opener.addheaders += self.download_headers
urlretrieve(url, target, report_hook)
except OSError as e:
attempts += 1
From e58b75ef6c63f586c0b2375e931acf3ef10cdab1 Mon Sep 17 00:00:00 2001
From: Christian Clauss <cclauss@me.com>
Date: Sun, 20 Oct 2024 22:54:21 +0200
Subject: [PATCH 080/158] CHANGELOG.md: Fix typos discovered by codespell
---
CHANGELOG.md | 88 ++++++++++++++++++++++++++--------------------------
1 file changed, 44 insertions(+), 44 deletions(-)
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 5686bcbb88..60f3137ae3 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -14,7 +14,7 @@
- unknown argument "fp-model" and strict is not a directory or a file [\#2359](https://github.com/kivy/python-for-android/issues/2359)
- Copy past is not working on kivy mobile app [\#2270](https://github.com/kivy/python-for-android/issues/2270)
- Flaky test failure in blacklist\(?\) - investigation needed [\#1781](https://github.com/kivy/python-for-android/issues/1781)
-- Problem with loding gevent: BadZipfile: File is not a zip file [\#1739](https://github.com/kivy/python-for-android/issues/1739)
+- Problem with loading gevent: BadZipfile: File is not a zip file [\#1739](https://github.com/kivy/python-for-android/issues/1739)
- ImportError when importing files containing \N{name} escape sequence [\#1060](https://github.com/kivy/python-for-android/issues/1060)
- Error with permission specification via setup.cfg [\#985](https://github.com/kivy/python-for-android/issues/985)
@@ -30,28 +30,28 @@
- use build aar in kotlin app ,can't load /lib/arm64/libpybundle.so file [\#2940](https://github.com/kivy/python-for-android/issues/2940)
- Feature Request: Pymssql [\#2936](https://github.com/kivy/python-for-android/issues/2936)
- LXML v4.8.0 fails to build. [\#2928](https://github.com/kivy/python-for-android/issues/2928)
-- Tryin to apply a plugin fails [\#2926](https://github.com/kivy/python-for-android/issues/2926)
+- Trying to apply a plugin fails [\#2926](https://github.com/kivy/python-for-android/issues/2926)
- ModuleNotFoundError: No module named '\_sysconfigdata\_\_darwin\_darwin' [\#2925](https://github.com/kivy/python-for-android/issues/2925)
- ReadTheDocs version is unclear. [\#2920](https://github.com/kivy/python-for-android/issues/2920)
- How to get real file path from uri [\#2911](https://github.com/kivy/python-for-android/issues/2911)
- And [\#2910](https://github.com/kivy/python-for-android/issues/2910)
- ModuleNotFoundError: No module named 'backports'
[\#2909](https://github.com/kivy/python-for-android/issues/2909)
-- not able to acess files unless connected to adb once [\#2907](https://github.com/kivy/python-for-android/issues/2907)
+- not able to access files unless connected to adb once [\#2907](https://github.com/kivy/python-for-android/issues/2907)
- opening files in other apps [\#2906](https://github.com/kivy/python-for-android/issues/2906)
- ImportError: dlopen failed: cannot locate symbol "\_ZTVSt9bad\_alloc" [\#2903](https://github.com/kivy/python-for-android/issues/2903)
- Fails to build pyjnius [\#2902](https://github.com/kivy/python-for-android/issues/2902)
- Kivy app crashes on startup [\#2895](https://github.com/kivy/python-for-android/issues/2895)
- aar file does not import properly in version v2023.09.16 [\#2894](https://github.com/kivy/python-for-android/issues/2894)
- App is crashing with Pyrebase4 [\#2893](https://github.com/kivy/python-for-android/issues/2893)
-- shared libs builds with 32 bit arch instaead of 64 bit [\#2888](https://github.com/kivy/python-for-android/issues/2888)
+- shared libs builds with 32 bit arch instead of 64 bit [\#2888](https://github.com/kivy/python-for-android/issues/2888)
- liblzma download error [\#2885](https://github.com/kivy/python-for-android/issues/2885)
- Misconfiguration causing failure in compilation. [\#2879](https://github.com/kivy/python-for-android/issues/2879)
- cygrpc.so is for EM\_X86\_64 \(62\) instead of EM\_AARCH64 \(183\) [\#2853](https://github.com/kivy/python-for-android/issues/2853)
- Are you able to build cffi==1.15.1? [\#2847](https://github.com/kivy/python-for-android/issues/2847)
- java.lang.IllegalStateException [\#2844](https://github.com/kivy/python-for-android/issues/2844)
- \[BUG\]: ctypes: AttributeError: undefined symbol: PyCapsule\_New [\#2840](https://github.com/kivy/python-for-android/issues/2840)
-- kivy cant load image in requesturl android [\#2832](https://github.com/kivy/python-for-android/issues/2832)
+- kivy can't load image in requesturl android [\#2832](https://github.com/kivy/python-for-android/issues/2832)
- Feature Request: Add Python `3.11` support [\#2798](https://github.com/kivy/python-for-android/issues/2798)
- Error Build APK FIle using Flask [\#2783](https://github.com/kivy/python-for-android/issues/2783)
- macOS: gwadlew fails at build tools stage \(newest build tools is 34.0.0-rc3, brew/openjdk@20\). [\#2781](https://github.com/kivy/python-for-android/issues/2781)
@@ -61,7 +61,7 @@
- Background service implemented using Pyjnius does not auto-restart on Kivy APK close [\#2772](https://github.com/kivy/python-for-android/issues/2772)
- \[JVM\]: FLAG\_IMMUTABLE or FLAG\_MUTABLE is required when a PendingIntent is created [\#2759](https://github.com/kivy/python-for-android/issues/2759)
- there is an issue with playing video from URL on the latest p4a releases [\#2744](https://github.com/kivy/python-for-android/issues/2744)
-- App crahes at launch on specific devices \(\[libpython3.9.so\] \_PyEval\_EvalFrameDefault\) \(Adreno 730?\) [\#2723](https://github.com/kivy/python-for-android/issues/2723)
+- App crashes at launch on specific devices \(\[libpython3.9.so\] \_PyEval\_EvalFrameDefault\) \(Adreno 730?\) [\#2723](https://github.com/kivy/python-for-android/issues/2723)
- Pandas giving error in Buildozer [\#2719](https://github.com/kivy/python-for-android/issues/2719)
- buildozer -v android debug [\#2711](https://github.com/kivy/python-for-android/issues/2711)
- \[proposed feature-request\] Lacking psutil recipe [\#2707](https://github.com/kivy/python-for-android/issues/2707)
@@ -88,14 +88,14 @@
- libEGL : EGLNativeWindowType disconnect failed [\#2253](https://github.com/kivy/python-for-android/issues/2253)
- Hao to support multiprocess Queue in Android [\#2249](https://github.com/kivy/python-for-android/issues/2249)
- autoclass: Class only found when called in specific places? [\#2242](https://github.com/kivy/python-for-android/issues/2242)
-- the app crach in time of import psycopg2 [\#2240](https://github.com/kivy/python-for-android/issues/2240)
+- the app crash in time of import psycopg2 [\#2240](https://github.com/kivy/python-for-android/issues/2240)
- env must be a dict [\#2170](https://github.com/kivy/python-for-android/issues/2170)
- Pandas doesn't work [\#2157](https://github.com/kivy/python-for-android/issues/2157)
- Webview bootstrap can't find 'org.jnius.NativeInvocationHandler'. [\#2140](https://github.com/kivy/python-for-android/issues/2140)
- clang++: error: linker command failed with exit code 1 [\#2082](https://github.com/kivy/python-for-android/issues/2082)
- ModuleNotFoundError: No module named 'setuptools' [\#2078](https://github.com/kivy/python-for-android/issues/2078)
- Scraping web pages with javascript [\#2052](https://github.com/kivy/python-for-android/issues/2052)
-- open webbrowser regsiter\(\) error [\#2047](https://github.com/kivy/python-for-android/issues/2047)
+- open webbrowser register\(\) error [\#2047](https://github.com/kivy/python-for-android/issues/2047)
- Missing javaclass when using able with previously working recipe [\#2041](https://github.com/kivy/python-for-android/issues/2041)
- :Class not found b'org/kivy/android/PythonActivity$ActivityResultListener' [\#2039](https://github.com/kivy/python-for-android/issues/2039)
- App\(using socket and opencv\) crash on opening [\#2038](https://github.com/kivy/python-for-android/issues/2038)
@@ -121,9 +121,9 @@
- Same issue w/ -lpython2.7 not found, workaround [\#1753](https://github.com/kivy/python-for-android/issues/1753)
- Several issues when installing packages via pip [\#1745](https://github.com/kivy/python-for-android/issues/1745)
- Publish a new Kivy Launcher for Python 3 [\#1638](https://github.com/kivy/python-for-android/issues/1638)
-- Travis conditional boostrap build support [\#1588](https://github.com/kivy/python-for-android/issues/1588)
+- Travis conditional bootstrap build support [\#1588](https://github.com/kivy/python-for-android/issues/1588)
- Error when execute APK only on device: ImportError: cannot import name \_htmlparser [\#1523](https://github.com/kivy/python-for-android/issues/1523)
-- onSensorChanged continuosly called during app execution [\#1498](https://github.com/kivy/python-for-android/issues/1498)
+- onSensorChanged continuously called during app execution [\#1498](https://github.com/kivy/python-for-android/issues/1498)
- GC deadlock on subprocess [\#1461](https://github.com/kivy/python-for-android/issues/1461)
- Code runs on old pygame backend but not on SDL2 [\#1411](https://github.com/kivy/python-for-android/issues/1411)
- build-tools below 25 will not add jars [\#1345](https://github.com/kivy/python-for-android/issues/1345)
@@ -148,7 +148,7 @@
- Initial support for PySide6 and Qt [\#2918](https://github.com/kivy/python-for-android/pull/2918) ([shyamnathp](https://github.com/shyamnathp))
- Introduce FAQ [\#2917](https://github.com/kivy/python-for-android/pull/2917) ([Julian-O](https://github.com/Julian-O))
- Add \(now mandatory\) `.readthedocs.yaml` file, add docs `requirements.txt` and update sphinx conf [\#2916](https://github.com/kivy/python-for-android/pull/2916) ([misl6](https://github.com/misl6))
-- enable json1 extenstion in sqlite3 [\#2915](https://github.com/kivy/python-for-android/pull/2915) ([HyTurtle](https://github.com/HyTurtle))
+- enable json1 extension in sqlite3 [\#2915](https://github.com/kivy/python-for-android/pull/2915) ([HyTurtle](https://github.com/HyTurtle))
- Bump `pyjnius` version to `1.6.1` [\#2914](https://github.com/kivy/python-for-android/pull/2914) ([misl6](https://github.com/misl6))
- Remove `distutils` usage, as is not available anymore on Python `3.12` [\#2912](https://github.com/kivy/python-for-android/pull/2912) ([misl6](https://github.com/misl6))
- Update Lottie player version [\#2900](https://github.com/kivy/python-for-android/pull/2900) ([HugoDaniel](https://github.com/HugoDaniel))
@@ -166,12 +166,12 @@
- Microphone And Audio permissions doesn't work [\#2889](https://github.com/kivy/python-for-android/issues/2889)
- Error with Scipy [\#2883](https://github.com/kivy/python-for-android/issues/2883)
-- Download failed \( Downloading sqlite3 from https://www.sqlite.org/2021/sqlite-amalgamation-3350500.zip \) during buildozer -v andriod debug [\#2881](https://github.com/kivy/python-for-android/issues/2881)
+- Download failed \( Downloading sqlite3 from https://www.sqlite.org/2021/sqlite-amalgamation-3350500.zip \) during buildozer -v android debug [\#2881](https://github.com/kivy/python-for-android/issues/2881)
- ONNXruntime lib open failed due to 64-bit [\#2880](https://github.com/kivy/python-for-android/issues/2880)
- Question: how to write a recipe to download and install\(coz build fail\) a wheel package directly? [\#2878](https://github.com/kivy/python-for-android/issues/2878)
- Impossible to install python for android [\#2867](https://github.com/kivy/python-for-android/issues/2867)
- scipy with kivy [\#2861](https://github.com/kivy/python-for-android/issues/2861)
-- False positve parser warning. [\#2856](https://github.com/kivy/python-for-android/issues/2856)
+- False positive parser warning. [\#2856](https://github.com/kivy/python-for-android/issues/2856)
- After successfully building app, its crashes with this error, using firebase-admin [\#2854](https://github.com/kivy/python-for-android/issues/2854)
- Kivy [\#2837](https://github.com/kivy/python-for-android/issues/2837)
- not installing on windows 10 [\#2836](https://github.com/kivy/python-for-android/issues/2836)
@@ -240,7 +240,7 @@
- c/\_cffi\_backend.c:407:23: error: expression is not assignable [\#2753](https://github.com/kivy/python-for-android/issues/2753)
- not install [\#2749](https://github.com/kivy/python-for-android/issues/2749)
- APK crashes upon launch. logcat error: null pointer dereference \(occurs with imported modules\) [\#2358](https://github.com/kivy/python-for-android/issues/2358)
-- Error occured while building the aplication using buildozer [\#2104](https://github.com/kivy/python-for-android/issues/2104)
+- Error occurred while building the application using buildozer [\#2104](https://github.com/kivy/python-for-android/issues/2104)
- "Could Not Extract Public Data" Needs very explicit instructions or feedback to the user [\#260](https://github.com/kivy/python-for-android/issues/260)
**Merged pull requests:**
@@ -317,12 +317,12 @@
- ffpyplayer recipe broken after SDL2 upgrade [\#2698](https://github.com/kivy/python-for-android/issues/2698)
- ModuleNotFoundError: No module named 'android' [\#2697](https://github.com/kivy/python-for-android/issues/2697)
- Error When I set android.api = 31 [\#2696](https://github.com/kivy/python-for-android/issues/2696)
-- Is threre any way to protect .py code [\#2695](https://github.com/kivy/python-for-android/issues/2695)
+- Is there any way to protect .py code [\#2695](https://github.com/kivy/python-for-android/issues/2695)
- args.service\_class\_name results in link error [\#2679](https://github.com/kivy/python-for-android/issues/2679)
- Remove `x86_64` suffix from ndk download link [\#2676](https://github.com/kivy/python-for-android/issues/2676)
- Pillow 9.2.0 recipe? [\#2671](https://github.com/kivy/python-for-android/issues/2671)
- `ffmpeg`: unable to find library -lvpx [\#2665](https://github.com/kivy/python-for-android/issues/2665)
-- Buildozer fails while build numpy recipie 'UnixCCompiler' object has no attribute 'cxx\_compiler' [\#2664](https://github.com/kivy/python-for-android/issues/2664)
+- Buildozer fails while build numpy recipe 'UnixCCompiler' object has no attribute 'cxx\_compiler' [\#2664](https://github.com/kivy/python-for-android/issues/2664)
- \[PEP 517\] Relax installation-time "pep517\<0.7.0" requirement [\#2573](https://github.com/kivy/python-for-android/issues/2573)
- Add support for custom resources [\#2298](https://github.com/kivy/python-for-android/issues/2298)
- Auto-correct / word suggestion does not work with Swiftkey/Samsung keyboard [\#2010](https://github.com/kivy/python-for-android/issues/2010)
@@ -427,7 +427,7 @@
- "Unit test apk" + "Unit test aab" + "Test updated recipes" test jobs should be run also on macOS \(both Intel and Apple Silicon\) [\#2569](https://github.com/kivy/python-for-android/issues/2569)
- unpackPyBundle\(\) on startup crashes already running service [\#2564](https://github.com/kivy/python-for-android/issues/2564)
- Webview app fail to startup. [\#2559](https://github.com/kivy/python-for-android/issues/2559)
-- genericndkbuild receipe Not compiling with android api \> 28 [\#2555](https://github.com/kivy/python-for-android/issues/2555)
+- GenericNDKBuildRecipe Not compiling with android api \> 28 [\#2555](https://github.com/kivy/python-for-android/issues/2555)
- Is there a way to build smaller apks? [\#2553](https://github.com/kivy/python-for-android/issues/2553)
- Webview, icon [\#2552](https://github.com/kivy/python-for-android/issues/2552)
- SONAME header not present in libpython3.8.so [\#2548](https://github.com/kivy/python-for-android/issues/2548)
@@ -436,7 +436,7 @@
- \[Temporary Resolved\] Python 4 android in mac os with Apple Silicon via Roseta [\#2528](https://github.com/kivy/python-for-android/issues/2528)
- Scipy is not installed due to "Error: 'numpy' must be installed before running the build." [\#2509](https://github.com/kivy/python-for-android/issues/2509)
- Lapack depends on arm-linux-androideabi-gfortran [\#2508](https://github.com/kivy/python-for-android/issues/2508)
-- Apk file built by buildozer is large in comparision to other apks [\#2473](https://github.com/kivy/python-for-android/issues/2473)
+- Apk file built by buildozer is large in comparison to other apks [\#2473](https://github.com/kivy/python-for-android/issues/2473)
- p4a is not compatible with ndk \>= 22 [\#2391](https://github.com/kivy/python-for-android/issues/2391)
- Sympy module. Error in buildozer: no module named sympy.testing [\#2381](https://github.com/kivy/python-for-android/issues/2381)
- build.gradle 'compile' depreciated [\#2362](https://github.com/kivy/python-for-android/issues/2362)
@@ -562,7 +562,7 @@
**Closed issues:**
-- Global varible/objetct isn't being shared between multiple files on Android, while the same code works normally under Windows. [\#2485](https://github.com/kivy/python-for-android/issues/2485)
+- Global variable/objetct isn't being shared between multiple files on Android, while the same code works normally under Windows. [\#2485](https://github.com/kivy/python-for-android/issues/2485)
- audiostream recipe does not copy .java files, leading to a crash when trying to open mic on android [\#2483](https://github.com/kivy/python-for-android/issues/2483)
- Applying patches for hostpython3 fails [\#2476](https://github.com/kivy/python-for-android/issues/2476)
- p4a failing to build webapp [\#2475](https://github.com/kivy/python-for-android/issues/2475)
@@ -610,7 +610,7 @@
- How to add webp support to pillow? [\#2254](https://github.com/kivy/python-for-android/issues/2254)
- Unable to run executable on Android [\#2251](https://github.com/kivy/python-for-android/issues/2251)
- Provide a native service option [\#2243](https://github.com/kivy/python-for-android/issues/2243)
-- the kivy app crach in time of import psycopg2 [\#2241](https://github.com/kivy/python-for-android/issues/2241)
+- the kivy app crash in time of import psycopg2 [\#2241](https://github.com/kivy/python-for-android/issues/2241)
- How can we make a camera App with zoom by Kivy [\#2238](https://github.com/kivy/python-for-android/issues/2238)
- Building pycrypto for arm64-v8a fails [\#2230](https://github.com/kivy/python-for-android/issues/2230)
- buildozer error Building pycrypto for arm64-v8a and x86\_64 [\#2216](https://github.com/kivy/python-for-android/issues/2216)
@@ -673,7 +673,7 @@
- :arrow\_up: Updates to Kivy2 [\#2384](https://github.com/kivy/python-for-android/pull/2384) ([kuzeyron](https://github.com/kuzeyron))
- fix travis [\#2383](https://github.com/kivy/python-for-android/pull/2383) ([obfusk](https://github.com/obfusk))
- :bug: Fixes pip command on Travis and bumps actions/setup-python [\#2382](https://github.com/kivy/python-for-android/pull/2382) ([AndreMiras](https://github.com/AndreMiras))
-- docs: fix simple typo, pacakged -\> packaged [\#2377](https://github.com/kivy/python-for-android/pull/2377) ([timgates42](https://github.com/timgates42))
+- docs: fix simple typo, packaged -\> packaged [\#2377](https://github.com/kivy/python-for-android/pull/2377) ([timgates42](https://github.com/timgates42))
- Fix master only merges [\#2376](https://github.com/kivy/python-for-android/pull/2376) ([inclement](https://github.com/inclement))
- Audiostream Fix [\#2375](https://github.com/kivy/python-for-android/pull/2375) ([xloem](https://github.com/xloem))
- Add service information for buildozer.spec [\#2372](https://github.com/kivy/python-for-android/pull/2372) ([xloem](https://github.com/xloem))
@@ -873,7 +873,7 @@
- Hadi [\#2048](https://github.com/kivy/python-for-android/issues/2048)
- p4a \(2019.10.6\) project build file management [\#2045](https://github.com/kivy/python-for-android/issues/2045)
- listdir of primary\_external\_storage\_path\(\) fails [\#2032](https://github.com/kivy/python-for-android/issues/2032)
-- Can't use AsyncImage with HTTPS URL \(or any HTTPS url wit any request\): fix is to manually load certifi [\#1827](https://github.com/kivy/python-for-android/issues/1827)
+- Can't use AsyncImage with HTTPS URL \(or any HTTPS url with any request\): fix is to manually load certifi [\#1827](https://github.com/kivy/python-for-android/issues/1827)
**Merged pull requests:**
@@ -1255,7 +1255,7 @@
- python3 + openssl compilation fail [\#1590](https://github.com/kivy/python-for-android/issues/1590)
- Update conditional build to python3 [\#1485](https://github.com/kivy/python-for-android/issues/1485)
-- building apk failes \( python 3 \) [\#746](https://github.com/kivy/python-for-android/issues/746)
+- building apk fails \( python 3 \) [\#746](https://github.com/kivy/python-for-android/issues/746)
**Closed issues:**
@@ -1349,7 +1349,7 @@
- Auto-close awaiting-reply labeled issues [\#1331](https://github.com/kivy/python-for-android/issues/1331)
- App crashes when sending POST request http [\#1329](https://github.com/kivy/python-for-android/issues/1329)
- kivy build error with python3crystax [\#1328](https://github.com/kivy/python-for-android/issues/1328)
-- No "pil" or "pillow" avaliable for Python 3 [\#1326](https://github.com/kivy/python-for-android/issues/1326)
+- No "pil" or "pillow" available for Python 3 [\#1326](https://github.com/kivy/python-for-android/issues/1326)
- pillow fails on import [\#1325](https://github.com/kivy/python-for-android/issues/1325)
- Unavoidable PySDL2 crash on app resume [\#1323](https://github.com/kivy/python-for-android/issues/1323)
- Tons of different PySDL2 crashes when tabbing in/out of application during loading or right after it finished [\#1321](https://github.com/kivy/python-for-android/issues/1321)
@@ -1414,20 +1414,20 @@
- The python3crystax recipe can only be built when using the CrystaX NDK. [\#1225](https://github.com/kivy/python-for-android/issues/1225)
- Didn't find any valid dependency graphs. [\#1222](https://github.com/kivy/python-for-android/issues/1222)
- Google requiring API Target 26 in Aug/Nov 2018 [\#1219](https://github.com/kivy/python-for-android/issues/1219)
-- p4a cant find android sdk [\#1218](https://github.com/kivy/python-for-android/issues/1218)
+- p4a can't find android sdk [\#1218](https://github.com/kivy/python-for-android/issues/1218)
- IOError: \[Errno 2\] No such file or directory: u'/Users/gauravgupta/kivy/.buildozer/android/platform/build/dists/myellipse/build/outputs/apk/myellipse-debug.apk' [\#1216](https://github.com/kivy/python-for-android/issues/1216)
-- Buildozer android application crashs on android [\#1215](https://github.com/kivy/python-for-android/issues/1215)
+- Buildozer android application crashes on android [\#1215](https://github.com/kivy/python-for-android/issues/1215)
- Multiple issues with latest Android SDK [\#1212](https://github.com/kivy/python-for-android/issues/1212)
-- sdkmanager doesnt exist [\#1210](https://github.com/kivy/python-for-android/issues/1210)
+- sdkmanager doesn't exist [\#1210](https://github.com/kivy/python-for-android/issues/1210)
- add-jar doesn't work [\#1208](https://github.com/kivy/python-for-android/issues/1208)
- Kivy not working on Sony devices [\#1206](https://github.com/kivy/python-for-android/issues/1206)
- sqlite pre-populated database not being found [\#1203](https://github.com/kivy/python-for-android/issues/1203)
- Try python3.7 with Google NDK [\#1202](https://github.com/kivy/python-for-android/issues/1202)
- commit 3534a761 [\#1200](https://github.com/kivy/python-for-android/issues/1200)
-- Kivy basic button program doesnt work [\#1199](https://github.com/kivy/python-for-android/issues/1199)
+- Kivy basic button program doesn't work [\#1199](https://github.com/kivy/python-for-android/issues/1199)
- Error Pythonforandroid.toolchain -m [\#1196](https://github.com/kivy/python-for-android/issues/1196)
- assert keyword do not work [\#1193](https://github.com/kivy/python-for-android/issues/1193)
-- macOS Hight Siera installation issue [\#1192](https://github.com/kivy/python-for-android/issues/1192)
+- macOS High Siera installation issue [\#1192](https://github.com/kivy/python-for-android/issues/1192)
- failed to setup p4a on ubuntu \(multiple issues\) [\#1191](https://github.com/kivy/python-for-android/issues/1191)
- Cryptography woes: Can we freshen up our Python version... [\#1190](https://github.com/kivy/python-for-android/issues/1190)
- Kivy crashes immediately on start, on Sony devices [\#1188](https://github.com/kivy/python-for-android/issues/1188)
@@ -1468,7 +1468,7 @@
- ImportError: No module named audioop [\#1067](https://github.com/kivy/python-for-android/issues/1067)
- sqlite3 not working with android\_new [\#1053](https://github.com/kivy/python-for-android/issues/1053)
- dlopen failed: python2.7/site-packages/grpc/\_cython/cygrpc.so not 32-bit: 2 [\#1052](https://github.com/kivy/python-for-android/issues/1052)
-- Cant start service app [\#1049](https://github.com/kivy/python-for-android/issues/1049)
+- Can't start service app [\#1049](https://github.com/kivy/python-for-android/issues/1049)
- Cannot build APK with python3crystax and flask - conflicting dependencies [\#1041](https://github.com/kivy/python-for-android/issues/1041)
- Slow build process since sh 1.12.5 [\#1038](https://github.com/kivy/python-for-android/issues/1038)
- Python3 + PyYaml conflict [\#1031](https://github.com/kivy/python-for-android/issues/1031)
@@ -1496,7 +1496,7 @@
- ImportError android [\#943](https://github.com/kivy/python-for-android/issues/943)
- Older android version can't load libraries properly [\#925](https://github.com/kivy/python-for-android/issues/925)
- sed: 1: "Modules/Setup.local": invalid command code M [\#924](https://github.com/kivy/python-for-android/issues/924)
-- Python3: armeabi used to copy, but armeabi-v7a choosen [\#913](https://github.com/kivy/python-for-android/issues/913)
+- Python3: armeabi used to copy, but armeabi-v7a chosen [\#913](https://github.com/kivy/python-for-android/issues/913)
- ImportError for sqlite3 [\#910](https://github.com/kivy/python-for-android/issues/910)
- PyGame backend: error while using android.copy\_libs = 1 [\#888](https://github.com/kivy/python-for-android/issues/888)
- pytz installation works, but requires user to make build folder manually [\#884](https://github.com/kivy/python-for-android/issues/884)
@@ -1648,7 +1648,7 @@
- Unify configChanges manifest entry and add missing values [\#1522](https://github.com/kivy/python-for-android/pull/1522) ([etc0de](https://github.com/etc0de))
- Add google repository at allprojects [\#1521](https://github.com/kivy/python-for-android/pull/1521) ([wo01](https://github.com/wo01))
- Fix bytes/unicode issues in android recipe [\#1516](https://github.com/kivy/python-for-android/pull/1516) ([KeyWeeUsr](https://github.com/KeyWeeUsr))
-- Uses target python3 on conditional buids, fixes \#1485 [\#1515](https://github.com/kivy/python-for-android/pull/1515) ([AndreMiras](https://github.com/AndreMiras))
+- Uses target python3 on conditional builds, fixes \#1485 [\#1515](https://github.com/kivy/python-for-android/pull/1515) ([AndreMiras](https://github.com/AndreMiras))
- Updates websocket-client recipe, fixes \#1253 [\#1513](https://github.com/kivy/python-for-android/pull/1513) ([AndreMiras](https://github.com/AndreMiras))
- No need to decode into unicode when running in python 3 [\#1512](https://github.com/kivy/python-for-android/pull/1512) ([jtoledo1974](https://github.com/jtoledo1974))
- Update gradle version [\#1507](https://github.com/kivy/python-for-android/pull/1507) ([opacam](https://github.com/opacam))
@@ -1785,14 +1785,14 @@
- --presplash and --icon aren't mentioned in revamp docs [\#975](https://github.com/kivy/python-for-android/issues/975)
- NDK automatic lookup tries to pick a tarball [\#972](https://github.com/kivy/python-for-android/issues/972)
- Kivy is broken on recent master [\#970](https://github.com/kivy/python-for-android/issues/970)
-- device doesnt go on sleep mode [\#969](https://github.com/kivy/python-for-android/issues/969)
+- device doesn't go on sleep mode [\#969](https://github.com/kivy/python-for-android/issues/969)
- The python2 build imports cython from the system python in /usr/lib/... [\#964](https://github.com/kivy/python-for-android/issues/964)
- "Could not remove android presplash" if 'android' is not in requirements [\#963](https://github.com/kivy/python-for-android/issues/963)
- "AndroidJoystick is not supported by your version of linux" confusing message in log [\#962](https://github.com/kivy/python-for-android/issues/962)
- Could not ping localhost:5000 [\#960](https://github.com/kivy/python-for-android/issues/960)
- Using pyjnius leads to crash \(sometimes?\) if app built by new toolchain [\#959](https://github.com/kivy/python-for-android/issues/959)
- Android screen rotation is probably broken [\#955](https://github.com/kivy/python-for-android/issues/955)
-- Compling pyo with sdl is breaking ply / enaml [\#947](https://github.com/kivy/python-for-android/issues/947)
+- Compiling pyo with sdl is breaking ply / enaml [\#947](https://github.com/kivy/python-for-android/issues/947)
- Access WiFi information? [\#940](https://github.com/kivy/python-for-android/issues/940)
- p4a erroring on SSL connection [\#939](https://github.com/kivy/python-for-android/issues/939)
- Compiling PIL seems to use pyconfig.h from the wrong directory [\#937](https://github.com/kivy/python-for-android/issues/937)
@@ -1866,7 +1866,7 @@
- raise exc\_info\[0\], exc\_info\[1\], exc\_info\[2\] - Syntax Error [\#721](https://github.com/kivy/python-for-android/issues/721)
- Some input files use or override a deprecated API. [\#719](https://github.com/kivy/python-for-android/issues/719)
- Unexpected "malformed start tag" error with HTMLParser [\#715](https://github.com/kivy/python-for-android/issues/715)
-- open\(\) build-in function don't work as expected [\#706](https://github.com/kivy/python-for-android/issues/706)
+- open\(\) built-in function don't work as expected [\#706](https://github.com/kivy/python-for-android/issues/706)
- Webview bootstrap. Where to start? \[$100\] [\#700](https://github.com/kivy/python-for-android/issues/700)
- Back button doesn't work [\#699](https://github.com/kivy/python-for-android/issues/699)
- FileNotFoundError '/bin/sh' with subprocess.py python3crystax [\#691](https://github.com/kivy/python-for-android/issues/691)
@@ -1936,9 +1936,9 @@
- \[revamp\] Recipes can only depend on other recipes [\#449](https://github.com/kivy/python-for-android/issues/449)
- \[revamp\] p4a silently fails if --private is not absolute [\#448](https://github.com/kivy/python-for-android/issues/448)
- \[revamp\] Invalid syntax for python3 [\#444](https://github.com/kivy/python-for-android/issues/444)
-- \[revamp\] toolchain.py ignores recipes with errrors [\#440](https://github.com/kivy/python-for-android/issues/440)
+- \[revamp\] toolchain.py ignores recipes with errors [\#440](https://github.com/kivy/python-for-android/issues/440)
- Error when trying to create an apk package with buildozer or with distribute.sh [\#435](https://github.com/kivy/python-for-android/issues/435)
-- pylibpd failes to compile [\#434](https://github.com/kivy/python-for-android/issues/434)
+- pylibpd fails to compile [\#434](https://github.com/kivy/python-for-android/issues/434)
- \[revamp\] --android\_api is ignored on SDL2 bootstrap [\#425](https://github.com/kivy/python-for-android/issues/425)
- OSError: \[Errno 2\] No such file or directory: '/home/username/code/kivy/examples/demo/touchtracer' [\#424](https://github.com/kivy/python-for-android/issues/424)
- \[revamp\] - Darwin patches applied on Linux [\#423](https://github.com/kivy/python-for-android/issues/423)
@@ -1975,7 +1975,7 @@
- pygame.display.set\_mode runs out of memory [\#331](https://github.com/kivy/python-for-android/issues/331)
- Python Service multiple instances [\#329](https://github.com/kivy/python-for-android/issues/329)
- Kivy Launcher should include Plyer [\#328](https://github.com/kivy/python-for-android/issues/328)
-- Issue in Cython file compilation and building kivy.graphics.vertex\_instruction extentions [\#326](https://github.com/kivy/python-for-android/issues/326)
+- Issue in Cython file compilation and building kivy.graphics.vertex\_instruction extensions [\#326](https://github.com/kivy/python-for-android/issues/326)
- Failed Cython Compilation [\#325](https://github.com/kivy/python-for-android/issues/325)
- Python3 Branch: jinja2 traceback on buildozer --verbose android debug [\#322](https://github.com/kivy/python-for-android/issues/322)
- About ctypes [\#319](https://github.com/kivy/python-for-android/issues/319)
@@ -1989,7 +1989,7 @@
- My `./distribute.sh` suddenly stop building today. [\#294](https://github.com/kivy/python-for-android/issues/294)
- Create recipes for storm and psycopg2 [\#293](https://github.com/kivy/python-for-android/issues/293)
- error in your VM Image after ./distribute.sh -m kivy [\#291](https://github.com/kivy/python-for-android/issues/291)
-- Eror in Apk process building [\#290](https://github.com/kivy/python-for-android/issues/290)
+- Error in Apk process building [\#290](https://github.com/kivy/python-for-android/issues/290)
- IOError: \[Errno 2\] No usable temporary directory [\#289](https://github.com/kivy/python-for-android/issues/289)
- Path commands on readme and official-website-toolchain don't seem to work [\#281](https://github.com/kivy/python-for-android/issues/281)
- Twisted/recipe.sh can't find zope.interface [\#280](https://github.com/kivy/python-for-android/issues/280)
@@ -2041,11 +2041,11 @@
- Kivy TextInput doesn't work with SwiftKey keyboard [\#184](https://github.com/kivy/python-for-android/issues/184)
- Error in using debug flag, calling ANT debugger? [\#179](https://github.com/kivy/python-for-android/issues/179)
- Update setuptools version [\#176](https://github.com/kivy/python-for-android/issues/176)
-- Problems with distibute.sh [\#175](https://github.com/kivy/python-for-android/issues/175)
+- Problems with distribute.sh [\#175](https://github.com/kivy/python-for-android/issues/175)
- Rst editor crashing on android [\#174](https://github.com/kivy/python-for-android/issues/174)
- Doubt about distribute.sh [\#173](https://github.com/kivy/python-for-android/issues/173)
- Own Activities in AndroidManifest.xml [\#172](https://github.com/kivy/python-for-android/issues/172)
-- Error to execute comand ./distribute.sh -m "openssl pil kivy" [\#167](https://github.com/kivy/python-for-android/issues/167)
+- Error to execute command ./distribute.sh -m "openssl pil kivy" [\#167](https://github.com/kivy/python-for-android/issues/167)
- keyboard stays on screen in android, after app exit [\#166](https://github.com/kivy/python-for-android/issues/166)
- ffmpeg ndk r9 incompatibility [\#165](https://github.com/kivy/python-for-android/issues/165)
- kivy touchtracer apk not running on emmulator [\#162](https://github.com/kivy/python-for-android/issues/162)
@@ -2059,7 +2059,7 @@
- configure: error: C compiler cannot create executables [\#145](https://github.com/kivy/python-for-android/issues/145)
- GCC 4.4.3 depreciated in android NDK [\#143](https://github.com/kivy/python-for-android/issues/143)
- dlopen fail on android 4.3 [\#141](https://github.com/kivy/python-for-android/issues/141)
-- new depencency ordering broke some usecases with buildozer [\#140](https://github.com/kivy/python-for-android/issues/140)
+- new dependency ordering broke some usecases with buildozer [\#140](https://github.com/kivy/python-for-android/issues/140)
- Android Keyboard information [\#139](https://github.com/kivy/python-for-android/issues/139)
- Android keyboards do not recognize Password fields as secure passwords [\#138](https://github.com/kivy/python-for-android/issues/138)
- kivy.network.urlrequest: No callback parameter: on\_failure [\#137](https://github.com/kivy/python-for-android/issues/137)
@@ -2131,7 +2131,7 @@
- Error: Target id 'android-8' is not valid. [\#25](https://github.com/kivy/python-for-android/issues/25)
- Build Error: "/usr/lib/libpython2.7.so: file not recognized: File format not recognized" [\#16](https://github.com/kivy/python-for-android/issues/16)
- FFMpeg doesn't build [\#13](https://github.com/kivy/python-for-android/issues/13)
-- Problem init enviroment [\#12](https://github.com/kivy/python-for-android/issues/12)
+- Problem init environment [\#12](https://github.com/kivy/python-for-android/issues/12)
- error when build the APK [\#10](https://github.com/kivy/python-for-android/issues/10)
- arm-linux-androideabi-gcc: Internal error: Killed \(program cc1\) [\#9](https://github.com/kivy/python-for-android/issues/9)
- Build: Pyrex error [\#7](https://github.com/kivy/python-for-android/issues/7)
@@ -2200,7 +2200,7 @@
- Allow installation on Windows [\#902](https://github.com/kivy/python-for-android/pull/902) ([ethanhs](https://github.com/ethanhs))
- Made clean-recipe-build delete dists [\#901](https://github.com/kivy/python-for-android/pull/901) ([inclement](https://github.com/inclement))
- Improved log for missing python modules [\#900](https://github.com/kivy/python-for-android/pull/900) ([inclement](https://github.com/inclement))
-- Addded textinput scatter testapp [\#899](https://github.com/kivy/python-for-android/pull/899) ([inclement](https://github.com/inclement))
+- Added textinput scatter testapp [\#899](https://github.com/kivy/python-for-android/pull/899) ([inclement](https://github.com/inclement))
- Allow list of permissions for bdist [\#898](https://github.com/kivy/python-for-android/pull/898) ([KeyWeeUsr](https://github.com/KeyWeeUsr))
- Fix bdistapk for launcher [\#896](https://github.com/kivy/python-for-android/pull/896) ([KeyWeeUsr](https://github.com/KeyWeeUsr))
- Added symlink\_java\_src dev option [\#894](https://github.com/kivy/python-for-android/pull/894) ([inclement](https://github.com/inclement))
From 5e14116bf09ca1d48719f7fa7e8c438f1d896ab8 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Sun, 20 Oct 2024 16:00:15 -0700
Subject: [PATCH 081/158] specify environment variable for download headers as
a JSON formatted set of values
---
doc/source/recipes.rst | 6 ++++++
pythonforandroid/recipe.py | 14 ++++++++++++--
2 files changed, 18 insertions(+), 2 deletions(-)
diff --git a/doc/source/recipes.rst b/doc/source/recipes.rst
index ea0ff6c174..bfe49ca717 100644
--- a/doc/source/recipes.rst
+++ b/doc/source/recipes.rst
@@ -66,10 +66,16 @@ Specifying the URL
download_headers property.
For example, when downloading from a private github repository, you can specify the following:
+
+(For the download_headers property in your recipe)
```
[('Authorization', 'token <your personal access token>'), ('Accept', 'application/vnd.github+json')]
```
+(For the DOWNLOAD_HEADERS_my-package-name environment variable - specify as a JSON formatted set of values)
+```
+ [["Authorization","token <your personal access token>"],["Accept", "application/vnd.github+json"]]
+```
The actual build process takes place via three core methods::
def prebuild_arch(self, arch):
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index ad7dbab1a5..0204e405ed 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1,7 +1,7 @@
from os.path import basename, dirname, exists, isdir, isfile, join, realpath, split
import glob
-
import hashlib
+import json
from re import match
import sh
@@ -64,7 +64,10 @@ class Recipe(metaclass=RecipeMeta):
for authorization purposes.
Specified as an array of tuples:
- [("header name", "header value")]
+ [("header1", "foo"), ("header2", "bar")]
+
+ When specifying as an environment variable (DOWNLOAD_HEADER_my-package-name), use a JSON formatted fragement:
+ [["header1","foo"],["header2", "bar"]]
For example, when downloading from a private
github repository, you can specify the following:
@@ -185,6 +188,13 @@ def versioned_url(self):
@property
def download_headers(self):
key = "DOWNLOAD_HEADERS_" + self.name
+ env_headers = environ.get(key)
+ if env_headers:
+ try:
+ return [tuple(h) for h in json.loads(env_headers)]
+ except Exception as ex:
+ raise ValueError(f'Invalid Download headers for {key} - must be JSON formatted as [["header1","foo"],["header2","bar"]]: {ex}')
+
return environ.get(key, self._download_headers)
def download_file(self, url, target, cwd=None):
From 3df4180c2713113bf3dafe871aba8532f76c9dda Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Fri, 27 Sep 2024 23:10:18 +0200
Subject: [PATCH 082/158] :construction_worker: On-demand CI custom build
Make it possible to trigger a build from the CI on demand.
The following parameters are configurable on demand:
- artifact type (e.g. aab, aar, apk)
- arch (comma separated list)
- bootstrap (e.g. q2, sdl2, service_library, service_only, webview)
- build mode (e.g. debug, release)
- operating system (e.g. ubuntu-latest, macos-latest)
- requirements (comma separated list)
Also bump a few action versions.
---
.github/workflows/custom-build.yml | 95 ++++++++++++++++++++++++++++++
.github/workflows/push.yml | 4 +-
.github/workflows/pypi-release.yml | 6 +-
Makefile | 14 +++++
4 files changed, 114 insertions(+), 5 deletions(-)
create mode 100644 .github/workflows/custom-build.yml
diff --git a/.github/workflows/custom-build.yml b/.github/workflows/custom-build.yml
new file mode 100644
index 0000000000..8cc381a29e
--- /dev/null
+++ b/.github/workflows/custom-build.yml
@@ -0,0 +1,95 @@
+name: Custom build
+
+on:
+ workflow_dispatch:
+ inputs:
+ arch:
+ description: "Comma separated architectures (e.g., armeabi-v7a, arm64-v8a, x86_64, x86)"
+ required: true
+ default: "armeabi-v7a,arm64-v8a,x86_64,x86"
+ artifact:
+ description: "Artifact type"
+ required: true
+ default: "apk"
+ type: choice
+ options:
+ - "aab"
+ - "aar"
+ - "apk"
+ bootstrap:
+ description: "Bootstrap to use"
+ required: true
+ default: "sdl2"
+ type: choice
+ options:
+ - "qt"
+ - "sdl2"
+ - "service_library"
+ - "service_only"
+ - "webview"
+ mode:
+ description: "Build mode"
+ required: true
+ default: "debug"
+ type: choice
+ options:
+ - "debug"
+ - "release"
+ os:
+ description: "Operating system to run on"
+ required: true
+ default: "ubuntu-latest"
+ type: choice
+ options:
+ - "ubuntu-latest"
+ - "macos-latest"
+ requirements:
+ description: "Comma separated requirements"
+ required: true
+ default: "python3,kivy"
+
+env:
+ APK_ARTIFACT_FILENAME: bdist_unit_tests_app-debug-1.1.apk
+ AAB_ARTIFACT_FILENAME: bdist_unit_tests_app-release-1.1.aab
+ AAR_ARTIFACT_FILENAME: bdist_unit_tests_app-release-1.1.aar
+ PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE: 0
+
+jobs:
+ build:
+ name: Build test APP [ ${{ github.event.inputs.arch }} | ${{ github.event.inputs.artifact }} | ${{ github.event.inputs.bootstrap }} | ${{ github.event.inputs.mode }} | ${{ github.event.inputs.os }} | ${{ github.event.inputs.requirements }}]
+ runs-on: ${{ github.event.inputs.os }}
+ steps:
+ - name: Checkout python-for-android
+ uses: actions/checkout@v4
+ - name: Pull the python-for-android docker image
+ run: make docker/pull
+ - name: Build python-for-android docker image
+ run: make docker/build
+ - name: Build multi-arch artifact with docker
+ run: |
+ docker run --name p4a-latest kivy/python-for-android make ARCH=${{ github.event.inputs.arch }} ARTIFACT=${{ github.event.inputs.artifact }} BOOTSTRAP=${{ github.event.inputs.bootstrap }} MODE=${{ github.event.inputs.mode }} REQUIREMENTS=${{ github.event.inputs.requirements }} testapps-generic
+ - name: Copy produced artifacts from docker container (*.apk, *.aab)
+ if: github.event.inputs.bootstrap != 'service_library'
+ run: |
+ mkdir -p dist
+ docker cp p4a-latest:/home/user/app/testapps/on_device_unit_tests/${{ env.APK_ARTIFACT_FILENAME }} dist/ || true
+ docker cp p4a-latest:/home/user/app/testapps/on_device_unit_tests/${{ env.AAB_ARTIFACT_FILENAME }} dist/ || true
+ - name: Copy produced artifacts from docker container (*.aar)
+ if: github.event.inputs.bootstrap == 'service_library'
+ run: |
+ mkdir -p dist
+ docker cp p4a-latest:/home/user/app/testapps/on_device_unit_tests/${{ env.AAR_ARTIFACT_FILENAME }} dist/
+ - name: Rename artifacts to include the build platform name (*.apk, *.aab, *.aar)
+ run: |
+ if [ -f dist/${{ env.APK_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.APK_ARTIFACT_FILENAME }} dist/${{ github.event.inputs.os }}-${{ github.event.inputs.bootstrap }}-${{ env.APK_ARTIFACT_FILENAME }}; fi
+ if [ -f dist/${{ env.AAB_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.AAB_ARTIFACT_FILENAME }} dist/${{ github.event.inputs.os }}-${{ github.event.inputs.bootstrap }}-${{ env.AAB_ARTIFACT_FILENAME }}; fi
+ if [ -f dist/${{ env.AAR_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.AAR_ARTIFACT_FILENAME }} dist/${{ github.event.inputs.os }}-${{ github.event.inputs.bootstrap }}-${{ env.AAR_ARTIFACT_FILENAME }}; fi
+ - name: Upload artifacts
+ uses: actions/upload-artifact@v4
+ with:
+ name: ${{ github.event.inputs.os }}-${{ github.event.inputs.bootstrap }}-artifacts
+ path: dist
+ # Cleanup the container after all steps are done
+ - name: Cleanup Docker container
+ run: docker rm p4a-latest
+ if: always()
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 0cbad15afe..030828e657 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -17,7 +17,7 @@ jobs:
- name: Checkout python-for-android
uses: actions/checkout@v4
- name: Set up Python 3.x
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: 3.x
- name: Run flake8
@@ -38,7 +38,7 @@ jobs:
- name: Checkout python-for-android
uses: actions/checkout@v4
- name: Set up Python ${{ matrix.python-version }}
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: ${{ matrix.python-version }}
- name: Tox tests
diff --git a/.github/workflows/pypi-release.yml b/.github/workflows/pypi-release.yml
index a66a30567e..487a903016 100644
--- a/.github/workflows/pypi-release.yml
+++ b/.github/workflows/pypi-release.yml
@@ -5,9 +5,9 @@ jobs:
pypi_release:
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v3
+ - uses: actions/checkout@v4
- name: Set up Python 3.x
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: '3.x'
- name: Install dependencies
@@ -22,4 +22,4 @@ jobs:
uses: pypa/gh-action-pypi-publish@v1.4.2
with:
user: __token__
- password: ${{ secrets.pypi_password }}
\ No newline at end of file
+ password: ${{ secrets.pypi_password }}
diff --git a/Makefile b/Makefile
index 67d71e7bcb..03c4ff3e52 100644
--- a/Makefile
+++ b/Makefile
@@ -33,6 +33,20 @@ rebuild_updated_recipes: virtualenv
ANDROID_SDK_HOME=$(ANDROID_SDK_HOME) ANDROID_NDK_HOME=$(ANDROID_NDK_HOME) \
$(PYTHON) ci/rebuild_updated_recipes.py $(REBUILD_UPDATED_RECIPES_EXTRA_ARGS)
+# make ARCH=armeabi-v7a,arm64-v8a ARTIFACT=apk BOOTSTRAP=sdl2 MODE=debug REQUIREMENTS=python testapps-generic
+testapps-generic: virtualenv
+ @if [ -z "$(ARCH)" ]; then echo "ARCH is not set"; exit 1; fi
+ @if [ -z "$(ARTIFACT)" ]; then echo "ARTIFACT is not set"; exit 1; fi
+ @if [ -z "$(BOOTSTRAP)" ]; then echo "BOOTSTRAP is not set"; exit 1; fi
+ @if [ -z "$(MODE)" ]; then echo "MODE is not set"; exit 1; fi
+ @if [ -z "$(REQUIREMENTS)" ]; then echo "REQUIREMENTS is not set"; exit 1; fi
+ @ARCH_FLAGS=$$(echo "$(ARCH)" | tr ',' ' ' | sed 's/\([^ ]\+\)/--arch=\1/g'); \
+ . $(ACTIVATE) && cd testapps/on_device_unit_tests/ && \
+ python setup.py $(ARTIFACT) \
+ --sdk-dir $(ANDROID_SDK_HOME) \
+ --ndk-dir $(ANDROID_NDK_HOME) \
+ $$ARCH_FLAGS --bootstrap $(BOOTSTRAP) --$(MODE) --requirements $(REQUIREMENTS)
+
testapps-with-numpy: testapps-with-numpy/debug/apk testapps-with-numpy/release/aab
# testapps-with-numpy/MODE/ARTIFACT
From 7d41d58ff3e1dd38833eaddc0156daea6f41a766 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 08:49:12 -0700
Subject: [PATCH 083/158] add unit test
---
tests/test_recipe.py | 7 +++++++
1 file changed, 7 insertions(+)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index 006129f3a2..e313f0c204 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -326,3 +326,10 @@ def test_postarch_build(self, mock_install_stl_lib):
assert recipe.need_stl_shared, True
recipe.postbuild_arch(arch)
mock_install_stl_lib.assert_called_once_with(arch)
+
+ def test_recipe_download_headers(self):
+ """Download header can be created on the fly using environment variables."""
+ recipe = DummyRecipe()
+ with mock.patch.dict(os.environ, '[["header1","foo"],["header2", "bar"]]'):
+ download_headers = recipe.download_headers
+ assert download_headers == [["header1","foo"],["header2", "bar"]]
From 283ce7d1bf367dda020a1f9431b0d9e85dde142e Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 09:01:31 -0700
Subject: [PATCH 084/158] tidy
---
tests/test_recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index e313f0c204..5a983e9304 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -332,4 +332,4 @@ def test_recipe_download_headers(self):
recipe = DummyRecipe()
with mock.patch.dict(os.environ, '[["header1","foo"],["header2", "bar"]]'):
download_headers = recipe.download_headers
- assert download_headers == [["header1","foo"],["header2", "bar"]]
+ assert download_headers == [["header1", "foo"],["header2", "bar"]]
From f8802d30c8c282909534bdbeeb65f05411e61593 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 09:03:39 -0700
Subject: [PATCH 085/158] tidy
---
tests/test_recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index 5a983e9304..75c36a554c 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -332,4 +332,4 @@ def test_recipe_download_headers(self):
recipe = DummyRecipe()
with mock.patch.dict(os.environ, '[["header1","foo"],["header2", "bar"]]'):
download_headers = recipe.download_headers
- assert download_headers == [["header1", "foo"],["header2", "bar"]]
+ assert download_headers == [["header1", "foo"], ["header2", "bar"]]
From 4243b5501159e3c8f39eadbb28accb0a10548894 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 09:14:22 -0700
Subject: [PATCH 086/158] fix tests
---
tests/test_recipe.py | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index 75c36a554c..1a736ffda2 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -330,6 +330,7 @@ def test_postarch_build(self, mock_install_stl_lib):
def test_recipe_download_headers(self):
"""Download header can be created on the fly using environment variables."""
recipe = DummyRecipe()
- with mock.patch.dict(os.environ, '[["header1","foo"],["header2", "bar"]]'):
+ recipe.name = "dummy"
+ with mock.patch.dict(os.environ, {'DOWNLOAD_HEADERS_dummy': '[["header1","foo"],["header2", "bar"]]'}):
download_headers = recipe.download_headers
assert download_headers == [["header1", "foo"], ["header2", "bar"]]
From 920d4db8913dd19cd6582f258f96d881f3694cad Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 09:20:29 -0700
Subject: [PATCH 087/158] fix recipe name for python 3.8
---
tests/test_recipe.py | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index 1a736ffda2..4c87625611 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -330,7 +330,6 @@ def test_postarch_build(self, mock_install_stl_lib):
def test_recipe_download_headers(self):
"""Download header can be created on the fly using environment variables."""
recipe = DummyRecipe()
- recipe.name = "dummy"
- with mock.patch.dict(os.environ, {'DOWNLOAD_HEADERS_dummy': '[["header1","foo"],["header2", "bar"]]'}):
+ with mock.patch.dict(os.environ, {f'DOWNLOAD_HEADERS_{recipe.name}': '[["header1","foo"],["header2", "bar"]]'}):
download_headers = recipe.download_headers
assert download_headers == [["header1", "foo"], ["header2", "bar"]]
From 0935af8ac3c5162a61a2144de95bc96603f8021e Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 10:21:04 -0700
Subject: [PATCH 088/158] fix assert for headers
---
tests/test_recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index 4c87625611..b02a874e84 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -332,4 +332,4 @@ def test_recipe_download_headers(self):
recipe = DummyRecipe()
with mock.patch.dict(os.environ, {f'DOWNLOAD_HEADERS_{recipe.name}': '[["header1","foo"],["header2", "bar"]]'}):
download_headers = recipe.download_headers
- assert download_headers == [["header1", "foo"], ["header2", "bar"]]
+ assert download_headers == [("header1", "foo"), ("header2", "bar")]
From 33c378479b1c27b036119d94e446746de7319249 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 10:49:33 -0700
Subject: [PATCH 089/158] cancel in progress jobs as needed
---
.github/workflows/push.yml | 3 +++
1 file changed, 3 insertions(+)
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 0cbad15afe..dd6a48531c 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -8,6 +8,9 @@ env:
AAR_ARTIFACT_FILENAME: bdist_unit_tests_app-release-1.1.aar
PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE: 0
+concurrency:
+ cancel-in-progress: true
+
jobs:
flake8:
From 5bed48d5a629c825c7f2fd6ab0eb38607ee08789 Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Mon, 21 Oct 2024 10:51:52 -0700
Subject: [PATCH 090/158] add build property to concurrency
---
.github/workflows/push.yml | 1 +
1 file changed, 1 insertion(+)
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index dd6a48531c..0c18ea81f8 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -9,6 +9,7 @@ env:
PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE: 0
concurrency:
+ group: build-${{ github.ref }}
cancel-in-progress: true
jobs:
From b3394938bafc40a41692242a7dae5bbfc506f8bb Mon Sep 17 00:00:00 2001
From: Kartavya Shukla <87070473+Novfensec@users.noreply.github.com>
Date: Tue, 19 Nov 2024 16:07:45 +0530
Subject: [PATCH 091/158] Add bitarray recipe (#3084)
* add bitarray recipe
---
pythonforandroid/recipes/bitarray/__init__.py | 11 +++++++++++
1 file changed, 11 insertions(+)
create mode 100644 pythonforandroid/recipes/bitarray/__init__.py
diff --git a/pythonforandroid/recipes/bitarray/__init__.py b/pythonforandroid/recipes/bitarray/__init__.py
new file mode 100644
index 0000000000..7f2d8eaae5
--- /dev/null
+++ b/pythonforandroid/recipes/bitarray/__init__.py
@@ -0,0 +1,11 @@
+from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+
+
+class BitarrayRecipe(CppCompiledComponentsPythonRecipe):
+ stl_lib_name = "c++_shared"
+ version = "3.0.0"
+ url = "https://github.com/ilanschnell/bitarray/archive/refs/tags/{version}.tar.gz"
+ depends = ["setuptools"]
+
+
+recipe = BitarrayRecipe()
From 9d4f2f9335065640411fcffea028595fff8ca96a Mon Sep 17 00:00:00 2001
From: HyTurtle <81598434+HyTurtle@users.noreply.github.com>
Date: Tue, 26 Nov 2024 16:38:01 +0000
Subject: [PATCH 092/158] bump openssl to 3 (#3086)
* bump openssl to 3
As version 1.1 disabled
* Update test_prerequisites.py
bump test version of openssl to 3
---
pythonforandroid/prerequisites.py | 2 +-
tests/test_prerequisites.py | 4 ++--
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/pythonforandroid/prerequisites.py b/pythonforandroid/prerequisites.py
index e85991948f..6b592046ed 100644
--- a/pythonforandroid/prerequisites.py
+++ b/pythonforandroid/prerequisites.py
@@ -262,7 +262,7 @@ def darwin_installer(self):
class OpenSSLPrerequisite(Prerequisite):
name = "openssl"
- homebrew_formula_name = "openssl@1.1"
+ homebrew_formula_name = "openssl@3"
mandatory = dict(linux=False, darwin=True)
installer_is_supported = dict(linux=False, darwin=True)
diff --git a/tests/test_prerequisites.py b/tests/test_prerequisites.py
index 70ffa0c0d1..8d577fde1c 100644
--- a/tests/test_prerequisites.py
+++ b/tests/test_prerequisites.py
@@ -99,8 +99,8 @@ def setUp(self):
self.mandatory = dict(linux=False, darwin=True)
self.installer_is_supported = dict(linux=False, darwin=True)
self.prerequisite = OpenSSLPrerequisite()
- self.expected_homebrew_formula_name = "openssl@1.1"
- self.expected_homebrew_location_prefix = "/opt/homebrew/opt/openssl@1.1"
+ self.expected_homebrew_formula_name = "openssl@3"
+ self.expected_homebrew_location_prefix = "/opt/homebrew/opt/openssl@3"
@mock.patch(
"pythonforandroid.prerequisites.Prerequisite._darwin_get_brew_formula_location_prefix"
From 534e53a96a3d9dbf86735f8948ed1c7cc0ffec83 Mon Sep 17 00:00:00 2001
From: Robert Niederreiter <rnix@squarewave.at>
Date: Wed, 27 Nov 2024 11:35:11 +0100
Subject: [PATCH 093/158] Update pyopenssl
---
pythonforandroid/recipes/pyopenssl/__init__.py | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/pythonforandroid/recipes/pyopenssl/__init__.py b/pythonforandroid/recipes/pyopenssl/__init__.py
index 092a31059e..cb69aca3bd 100644
--- a/pythonforandroid/recipes/pyopenssl/__init__.py
+++ b/pythonforandroid/recipes/pyopenssl/__init__.py
@@ -3,9 +3,9 @@
class PyOpenSSLRecipe(PythonRecipe):
- version = '19.0.0'
+ version = '24.2.1'
url = 'https://pypi.python.org/packages/source/p/pyOpenSSL/pyOpenSSL-{version}.tar.gz'
- depends = ['openssl', 'setuptools']
+ depends = ['cffi', 'openssl', 'setuptools']
site_packages_name = 'OpenSSL'
call_hostpython_via_targetpython = False
From f0177c5e72f7a604d0ac0f6d36b2c9c97a0f5b42 Mon Sep 17 00:00:00 2001
From: Robert Niederreiter <rnix@squarewave.at>
Date: Wed, 27 Nov 2024 11:46:44 +0100
Subject: [PATCH 094/158] Try older version
---
pythonforandroid/recipes/pyopenssl/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/pyopenssl/__init__.py b/pythonforandroid/recipes/pyopenssl/__init__.py
index cb69aca3bd..2d4d7a893f 100644
--- a/pythonforandroid/recipes/pyopenssl/__init__.py
+++ b/pythonforandroid/recipes/pyopenssl/__init__.py
@@ -3,7 +3,7 @@
class PyOpenSSLRecipe(PythonRecipe):
- version = '24.2.1'
+ version = '24.1.0'
url = 'https://pypi.python.org/packages/source/p/pyOpenSSL/pyOpenSSL-{version}.tar.gz'
depends = ['cffi', 'openssl', 'setuptools']
site_packages_name = 'OpenSSL'
From 1544339b89f519392f51aa9a33bf570963aeec59 Mon Sep 17 00:00:00 2001
From: Evstifeev Roman <someuniquename@gmail.com>
Date: Wed, 11 Dec 2024 00:34:14 +0500
Subject: [PATCH 095/158] Fix rm: cannot remove 'CMakeFiles/': Is a directory
fixes #3089
---
pythonforandroid/recipes/jpeg/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/jpeg/__init__.py b/pythonforandroid/recipes/jpeg/__init__.py
index a81b82555c..436dc129ba 100644
--- a/pythonforandroid/recipes/jpeg/__init__.py
+++ b/pythonforandroid/recipes/jpeg/__init__.py
@@ -27,7 +27,7 @@ def build_arch(self, arch):
toolchain_file = join(self.ctx.ndk_dir,
'build/cmake/android.toolchain.cmake')
- shprint(sh.rm, '-f', 'CMakeCache.txt', 'CMakeFiles/')
+ shprint(sh.rm, '-rf', 'CMakeCache.txt', 'CMakeFiles/')
shprint(sh.cmake, '-G', 'Unix Makefiles',
'-DCMAKE_SYSTEM_NAME=Android',
'-DCMAKE_POSITION_INDEPENDENT_CODE=1',
From ceca3da75349a5286477311260236bedb98621a6 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Tue, 10 Dec 2024 22:51:34 +0100
Subject: [PATCH 096/158] :green_heart: Fix sphinx documentation build errors
The errors & warnings were:
```
doc/source/quickstart.rst:60: WARNING: Title underline too short.
Installing Prerequisites
~~~~~~~~~~~~~~~~~~~~~~~
doc/source/quickstart.rst:60: WARNING: Title underline too short.
Installing Prerequisites
~~~~~~~~~~~~~~~~~~~~~~~
doc/source/recipes.rst:77: ERROR: Unexpected indentation.
doc/source/recipes.rst:75: WARNING: Inline literal start-string without end-string.
doc/source/recipes.rst:75: WARNING: Inline interpreted text or phrase reference start-string without end-string.
doc/source/recipes.rst:78: WARNING: Block quote ends without a blank line; unexpected unindent.
doc/source/recipes.rst:78: WARNING: Inline literal start-string without end-string.
WARNING: autodoc: failed to import class 'Recipe' from module 'toolchain'; the following exception was raised:
```
And more errors popped as we fixed the above ones.
Full log:
https://github.com/kivy/python-for-android/actions/runs/12263397494/job/34218728454
---
doc/source/quickstart.rst | 2 +-
doc/source/recipes.rst | 5 +++--
pythonforandroid/recipe.py | 4 ++--
pythonforandroid/toolchain.py | 1 +
4 files changed, 7 insertions(+), 5 deletions(-)
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 81c860f888..61c33d6f16 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -57,7 +57,7 @@ You can also test the master branch from Github using::
pip install git+https://github.com/kivy/python-for-android.git
Installing Prerequisites
-~~~~~~~~~~~~~~~~~~~~~~~
+~~~~~~~~~~~~~~~~~~~~~~~~
p4a requires a few dependencies to be installed on your system to work
properly. While we're working on a way to automate pre-requisites checks,
diff --git a/doc/source/recipes.rst b/doc/source/recipes.rst
index bfe49ca717..b6e34319b3 100644
--- a/doc/source/recipes.rst
+++ b/doc/source/recipes.rst
@@ -73,9 +73,10 @@ For example, when downloading from a private github repository, you can specify
```
(For the DOWNLOAD_HEADERS_my-package-name environment variable - specify as a JSON formatted set of values)
-```
+.. code-block:: bash
+
[["Authorization","token <your personal access token>"],["Accept", "application/vnd.github+json"]]
-```
+
The actual build process takes place via three core methods::
def prebuild_arch(self, arch):
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 0cace3346e..44469aef2c 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -128,6 +128,7 @@ class Recipe(metaclass=RecipeMeta):
keys should be the generated libraries and the values the relative path of
the library inside his build folder. This dict will be used to perform
different operations:
+
- copy the library into the right location, depending on if it's shared
or static)
- check if we have to rebuild the library
@@ -571,7 +572,6 @@ def should_build(self, arch):
'''Should perform any necessary test and return True only if it needs
building again. Per default we implement a library test, in case that
we detect so.
-
'''
if self.built_libraries:
return not all(
@@ -591,7 +591,7 @@ def install_libraries(self, arch):
'''This method is always called after `build_arch`. In case that we
detect a library recipe, defined by the class attribute
`built_libraries`, we will copy all defined libraries into the
- right location.
+ right location.
'''
if not self.built_libraries:
return
diff --git a/pythonforandroid/toolchain.py b/pythonforandroid/toolchain.py
index e05bb3a8fe..3987647f9b 100644
--- a/pythonforandroid/toolchain.py
+++ b/pythonforandroid/toolchain.py
@@ -750,6 +750,7 @@ def recipes(self, args):
"""
Prints recipes basic info, e.g.
.. code-block:: bash
+
python3 3.7.1
depends: ['hostpython3', 'sqlite3', 'openssl', 'libffi']
conflicts: []
From 37d272cc42f1ca0d0143743ccd3949cca56dfd5d Mon Sep 17 00:00:00 2001
From: Brent Picasso <brent@autosportlabs.com>
Date: Fri, 13 Dec 2024 11:25:47 -0800
Subject: [PATCH 097/158] Add hardware acceleration codecs to ffmpeg recipe
(#3092)
* enable hardware acceleration codecs
* Reorder "enable-mediacodec" to fix libx264 issues
---------
Co-authored-by: Dexer <73297572+DexerBR@users.noreply.github.com>
---
pythonforandroid/recipes/ffmpeg/__init__.py | 6 ++++++
1 file changed, 6 insertions(+)
diff --git a/pythonforandroid/recipes/ffmpeg/__init__.py b/pythonforandroid/recipes/ffmpeg/__init__.py
index 3bc824834f..f7134b3384 100644
--- a/pythonforandroid/recipes/ffmpeg/__init__.py
+++ b/pythonforandroid/recipes/ffmpeg/__init__.py
@@ -28,6 +28,12 @@ def build_arch(self, arch):
cflags = []
ldflags = []
+ # enable hardware acceleration codecs
+ flags = [
+ '--enable-jni',
+ '--enable-mediacodec'
+ ]
+
if 'openssl' in self.ctx.recipe_build_order:
flags += [
'--enable-openssl',
From 10e2ba938229e0bae81bd418a38e4851cdc6b322 Mon Sep 17 00:00:00 2001
From: Evstifeev Roman <someuniquename@gmail.com>
Date: Sat, 14 Dec 2024 19:15:27 +0500
Subject: [PATCH 098/158] Update sdl2_ttf from 2.20.2 to 2.22.0
fixes #2963
fixes #2902
fixes https://github.com/kivy/buildozer/issues/1772
---
pythonforandroid/recipes/sdl2_ttf/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl2_ttf/__init__.py b/pythonforandroid/recipes/sdl2_ttf/__init__.py
index 9f97ae441c..c869e1fc25 100644
--- a/pythonforandroid/recipes/sdl2_ttf/__init__.py
+++ b/pythonforandroid/recipes/sdl2_ttf/__init__.py
@@ -2,7 +2,7 @@
class LibSDL2TTF(BootstrapNDKRecipe):
- version = '2.20.2'
+ version = '2.22.0'
url = 'https://github.com/libsdl-org/SDL_ttf/releases/download/release-{version}/SDL2_ttf-{version}.tar.gz'
dir_name = 'SDL2_ttf'
From ea9fb61bc367c7837f51078d424f372980b6adc7 Mon Sep 17 00:00:00 2001
From: Evstifeev Roman <someuniquename@gmail.com>
Date: Sun, 15 Dec 2024 22:56:55 +0500
Subject: [PATCH 099/158] kivy recipe: add filetype dependency
Original PR: https://github.com/kivy/kivy/pull/8889
Fixes #3098
---
pythonforandroid/recipes/kivy/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index 5cb56611e7..20113d88a0 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -27,7 +27,7 @@ class KivyRecipe(CythonRecipe):
name = 'kivy'
depends = ['sdl2', 'pyjnius', 'setuptools']
- python_depends = ['certifi', 'chardet', 'idna', 'requests', 'urllib3']
+ python_depends = ['certifi', 'chardet', 'idna', 'requests', 'urllib3', 'filetype']
# sdl-gl-swapwindow-nogil.patch is needed to avoid a deadlock.
# See: https://github.com/kivy/kivy/pull/8025
From 1d7c70147bd2a32949bbd04d9edd24881799b124 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Mon, 23 Dec 2024 17:54:55 -0300
Subject: [PATCH 100/158] Add `httpx` recipe (#3100)
* add httpx recipe
* flake8 fix
---
pythonforandroid/recipes/httpx/__init__.py | 13 +++++++++++++
1 file changed, 13 insertions(+)
create mode 100644 pythonforandroid/recipes/httpx/__init__.py
diff --git a/pythonforandroid/recipes/httpx/__init__.py b/pythonforandroid/recipes/httpx/__init__.py
new file mode 100644
index 0000000000..60b34a8c46
--- /dev/null
+++ b/pythonforandroid/recipes/httpx/__init__.py
@@ -0,0 +1,13 @@
+from pythonforandroid.recipe import PyProjectRecipe
+
+
+class HttpxRecipe(PyProjectRecipe):
+ name = "httpx"
+ version = "0.28.1"
+ url = (
+ "https://pypi.python.org/packages/source/h/httpx/httpx-{version}.tar.gz"
+ )
+ depends = ["httpcore", "h11", "certifi", "idna", "sniffio"]
+
+
+recipe = HttpxRecipe()
From 61ff0b69190204f147ec5df60c2c1f77f30f03d9 Mon Sep 17 00:00:00 2001
From: JasonnnW3000 <sufssl04@gmail.com>
Date: Wed, 1 Jan 2025 01:01:01 -0500
Subject: [PATCH 101/158] Update LICENSE, fix license year
Signed-off-by: JasonnnW3000 <sufssl04@gmail.com>
---
LICENSE | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/LICENSE b/LICENSE
index 4e3506010a..06f46c69cc 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,6 +1,6 @@
MIT License
-Copyright (c) 2010-2023 Kivy Team and other contributors
+Copyright (c) 2010-2025 Kivy Team and other contributors
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
From b6d734c3f431464ff846a2b575be1a7a60e8833f Mon Sep 17 00:00:00 2001
From: Evstifeev Roman <someuniquename@gmail.com>
Date: Wed, 1 Jan 2025 13:03:40 +0500
Subject: [PATCH 102/158] update kivy to 2.3.1
---
pythonforandroid/recipes/kivy/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index 20113d88a0..b28e03a7a5 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -22,7 +22,7 @@ def is_kivy_affected_by_deadlock_issue(recipe=None, arch=None):
class KivyRecipe(CythonRecipe):
- version = '2.3.0'
+ version = '2.3.1'
url = 'https://github.com/kivy/kivy/archive/{version}.zip'
name = 'kivy'
From 09684e82252a71d0fcde5374a30997210ca681ab Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Thu, 6 Mar 2025 10:40:03 +0100
Subject: [PATCH 103/158] :arrow_up: Migrate to actions/upload-artifact@v4
The v3 is deprecated, the error was:
```
This request has been automatically failed because it uses a deprecated
version of `actions/upload-artifact: v3`.
Learn more:
https://github.blog/changelog/2024-04-16-deprecation-notice-v3-of-the-artifact-actions/
```
Also fix the docker image deletion to fail gracefully on no image.
The error was:
```
"docker rmi" requires at least 1 argument.
```
---
.github/workflows/push.yml | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 1301975a8a..99f63de53a 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -102,7 +102,7 @@ jobs:
if [ -f dist/${{ env.AAB_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.AAB_ARTIFACT_FILENAME }} dist/${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-${{ env.AAB_ARTIFACT_FILENAME }}; fi
if [ -f dist/${{ env.AAR_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.AAR_ARTIFACT_FILENAME }} dist/${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-${{ env.AAR_ARTIFACT_FILENAME }}; fi
- name: Upload artifacts
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: ${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-artifacts
path: dist
@@ -157,7 +157,7 @@ jobs:
if [ -f dist/${{ env.APK_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.APK_ARTIFACT_FILENAME }} dist/${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-${{ env.APK_ARTIFACT_FILENAME }}; fi
if [ -f dist/${{ env.AAB_ARTIFACT_FILENAME }} ]; then mv dist/${{ env.AAB_ARTIFACT_FILENAME }} dist/${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-${{ env.AAB_ARTIFACT_FILENAME }}; fi
- name: Upload artifacts
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: ${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-artifacts
path: dist
@@ -184,7 +184,7 @@ jobs:
sudo swapoff -a
sudo rm -f /swapfile
sudo apt -y clean
- docker rmi $(docker image ls -aq)
+ docker images -q | xargs -r docker rmi
df -h
- name: Pull docker image
run: |
From ceed0494e1128b4c2b5cf0beaa94cbc53a7edf46 Mon Sep 17 00:00:00 2001
From: Miguel Risco-Castillo <mriscoc@users.noreply.github.com>
Date: Fri, 7 Mar 2025 03:52:48 -0500
Subject: [PATCH 104/158] :bug: fixes Kiwisolver build fails
The error was "Python.h not found", closes #3115
Note that the macos-14 build is still failing
---
pythonforandroid/recipes/kiwisolver/__init__.py | 8 ++++++++
1 file changed, 8 insertions(+)
diff --git a/pythonforandroid/recipes/kiwisolver/__init__.py b/pythonforandroid/recipes/kiwisolver/__init__.py
index c4c19ac257..d94af0f14b 100644
--- a/pythonforandroid/recipes/kiwisolver/__init__.py
+++ b/pythonforandroid/recipes/kiwisolver/__init__.py
@@ -8,5 +8,13 @@ class KiwiSolverRecipe(PyProjectRecipe):
depends = ['cppy']
need_stl_shared = True
+ # from https://github.com/kivy/python-for-android/issues/3115
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ flags = " -I" + self.ctx.python_recipe.include_root(arch.arch)
+ env["CFLAGS"] = flags
+ env["CPPFLAGS"] = flags
+ return env
+
recipe = KiwiSolverRecipe()
From e710cfc581adaf49f6adfdc64d2540c9201dd7ed Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 22 Mar 2025 15:32:44 +0100
Subject: [PATCH 105/158] :bug: Fix the kiwisolver build on macOS, fixes #3122
---
pythonforandroid/recipes/kiwisolver/__init__.py | 7 ++++---
1 file changed, 4 insertions(+), 3 deletions(-)
diff --git a/pythonforandroid/recipes/kiwisolver/__init__.py b/pythonforandroid/recipes/kiwisolver/__init__.py
index d94af0f14b..3ccfc2d432 100644
--- a/pythonforandroid/recipes/kiwisolver/__init__.py
+++ b/pythonforandroid/recipes/kiwisolver/__init__.py
@@ -8,12 +8,13 @@ class KiwiSolverRecipe(PyProjectRecipe):
depends = ['cppy']
need_stl_shared = True
- # from https://github.com/kivy/python-for-android/issues/3115
def get_recipe_env(self, arch, **kwargs):
+ """Override compile and linker flags, refs: #3115 and #3122"""
env = super().get_recipe_env(arch, **kwargs)
flags = " -I" + self.ctx.python_recipe.include_root(arch.arch)
- env["CFLAGS"] = flags
- env["CPPFLAGS"] = flags
+ env["CFLAGS"] += flags
+ env["CPPFLAGS"] += flags
+ env["LDFLAGS"] += " -shared"
return env
From 44aabede4a375cb7ddfd57e5961890bdaa2150f0 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 22 Mar 2025 22:10:44 +0100
Subject: [PATCH 106/158] :bug: Update the greenlet recipe, closes #2806
Fixes build time errors, but runtime not tested.
---
pythonforandroid/recipes/greenlet/__init__.py | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/pythonforandroid/recipes/greenlet/__init__.py b/pythonforandroid/recipes/greenlet/__init__.py
index 3f2043d57d..d9b208476f 100644
--- a/pythonforandroid/recipes/greenlet/__init__.py
+++ b/pythonforandroid/recipes/greenlet/__init__.py
@@ -1,8 +1,8 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class GreenletRecipe(CompiledComponentsPythonRecipe):
- version = '0.4.15'
+class GreenletRecipe(PyProjectRecipe):
+ version = '3.1.1'
url = 'https://pypi.python.org/packages/source/g/greenlet/greenlet-{version}.tar.gz'
depends = ['setuptools']
call_hostpython_via_targetpython = False
From d6eeea1d9f9c912ea69dd25541e846cd255bdec3 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 22 Mar 2025 21:26:48 +0100
Subject: [PATCH 107/158] :bug: Update and fix atom recipe, closes #2802
Fixes build time errors, but runtime not tested.
---
pythonforandroid/recipes/atom/__init__.py | 13 +++++++------
pythonforandroid/recipes/atom/pyproject.toml.patch | 12 ++++++++++++
2 files changed, 19 insertions(+), 6 deletions(-)
create mode 100644 pythonforandroid/recipes/atom/pyproject.toml.patch
diff --git a/pythonforandroid/recipes/atom/__init__.py b/pythonforandroid/recipes/atom/__init__.py
index 51923d5487..22fec4cd57 100644
--- a/pythonforandroid/recipes/atom/__init__.py
+++ b/pythonforandroid/recipes/atom/__init__.py
@@ -1,11 +1,12 @@
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class AtomRecipe(CppCompiledComponentsPythonRecipe):
- site_packages_name = 'atom'
- version = '0.3.10'
- url = 'https://github.com/nucleic/atom/archive/master.zip'
- depends = ['setuptools']
+class AtomRecipe(PyProjectRecipe):
+ site_packages_name = "atom"
+ version = "0.11.0"
+ url = "https://files.pythonhosted.org/packages/source/a/atom/atom-{version}.tar.gz"
+ depends = ["setuptools"]
+ patches = ["pyproject.toml.patch"]
recipe = AtomRecipe()
diff --git a/pythonforandroid/recipes/atom/pyproject.toml.patch b/pythonforandroid/recipes/atom/pyproject.toml.patch
new file mode 100644
index 0000000000..ebf8cbc454
--- /dev/null
+++ b/pythonforandroid/recipes/atom/pyproject.toml.patch
@@ -0,0 +1,12 @@
+diff --git a/pyproject.toml b/pyproject.toml
+index d41287f..c83b053 100644
+--- a/pyproject.toml
++++ b/pyproject.toml
+@@ -40,6 +40,7 @@
+ [tool.setuptools]
+ include-package-data = false
+ package-data = { atom = ["py.typed", "*.pyi"] }
++ packages = ["atom"]
+
+ [tool.setuptools_scm]
+ write_to = "atom/version.py"
From 798e2c2fb0c95d4490e3f28b8996065c92d509f9 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 22 Mar 2025 17:26:20 +0100
Subject: [PATCH 108/158] :bug: Upgrade and fix gevent recipe, closes #2805
Update the gevent recipe from 1.4.0 (2019/01) to the latest version
24.11.1 (2024/11) and fix the build errors.
Note that the build fails on macOS, the error is:
```
deps/libuv/src/unix/bsd-ifaddrs.c:31:10: fatal error: 'net/if_dl.h' file not found
^~~~~~~~~~~~~
```
---
pythonforandroid/recipes/gevent/__init__.py | 27 ++++++++++++-----
.../recipes/gevent/cross_compiling.patch | 30 +++++++++----------
tests/recipes/test_gevent.py | 6 ++--
3 files changed, 39 insertions(+), 24 deletions(-)
diff --git a/pythonforandroid/recipes/gevent/__init__.py b/pythonforandroid/recipes/gevent/__init__.py
index 7958a5480f..3206603e82 100644
--- a/pythonforandroid/recipes/gevent/__init__.py
+++ b/pythonforandroid/recipes/gevent/__init__.py
@@ -1,22 +1,33 @@
+"""
+Note that this recipe doesn't yet build on macOS, the error is:
+```
+deps/libuv/src/unix/bsd-ifaddrs.c:31:10: fatal error: 'net/if_dl.h' file not found
+#include <net/if_dl.h>
+ ^~~~~~~~~~~~~
+1 error generated.
+error: command '/Users/runner/.android/android-ndk/toolchains/llvm/prebuilt/darwin-x86_64/bin/clang' failed with exit code 1
+```
+"""
import re
from pythonforandroid.logger import info
-from pythonforandroid.recipe import CythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class GeventRecipe(CythonRecipe):
- version = '1.4.0'
- url = 'https://pypi.python.org/packages/source/g/gevent/gevent-{version}.tar.gz'
+class GeventRecipe(PyProjectRecipe):
+ version = '24.11.1'
+ url = 'https://github.com/gevent/gevent/archive/refs/tags/{version}.tar.gz'
depends = ['librt', 'setuptools']
patches = ["cross_compiling.patch"]
- def get_recipe_env(self, arch=None, with_flags_in_cc=True):
+ def get_recipe_env(self, arch, **kwargs):
"""
- Moves all -I<inc> -D<macro> from CFLAGS to CPPFLAGS environment.
- Moves all -l<lib> from LDFLAGS to LIBS environment.
- Copies all -l<lib> from LDLIBS to LIBS environment.
- - Fixes linker name (use cross compiler) and flags (appends LIBS)
+ - Fixes linker name (use cross compiler) and flags (appends LIBS).
+ - Feds the command prefix for the configure --host flag.
"""
- env = super().get_recipe_env(arch, with_flags_in_cc)
+ env = super().get_recipe_env(arch, **kwargs)
# CFLAGS may only be used to specify C compiler flags, for macro definitions use CPPFLAGS
regex = re.compile(r'(?:\s|^)-[DI][\S]+')
env['CPPFLAGS'] = ''.join(re.findall(regex, env['CFLAGS'])).strip()
@@ -28,6 +39,8 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
env['LIBS'] += ' {}'.format(''.join(re.findall(regex, env['LDLIBS'])).strip())
env['LDFLAGS'] = re.sub(regex, '', env['LDFLAGS'])
info('Moved "{}" from LDFLAGS to LIBS.'.format(env['LIBS']))
+ # used with the `./configure --host` flag for cross compiling, refs #2805
+ env['COMMAND_PREFIX'] = arch.command_prefix
return env
diff --git a/pythonforandroid/recipes/gevent/cross_compiling.patch b/pythonforandroid/recipes/gevent/cross_compiling.patch
index 01e55d8c00..6cafbb9f05 100644
--- a/pythonforandroid/recipes/gevent/cross_compiling.patch
+++ b/pythonforandroid/recipes/gevent/cross_compiling.patch
@@ -1,26 +1,26 @@
diff --git a/_setupares.py b/_setupares.py
-index dd184de6..bb16bebe 100644
+index c42fe369..cd8854df 100644
--- a/_setupares.py
+++ b/_setupares.py
-@@ -43,7 +43,7 @@ else:
+@@ -42,7 +42,7 @@ cflags = ('CFLAGS="%s"' % (cflags,)) if cflags else ''
ares_configure_command = ' '.join([
"(cd ", quoted_dep_abspath('c-ares'),
- " && if [ -r ares_build.h ]; then cp ares_build.h ares_build.h.orig; fi ",
-- " && sh ./configure --disable-dependency-tracking " + _m32 + "CONFIG_COMMANDS= ",
-+ " && sh ./configure --host={} --disable-dependency-tracking ".format(os.environ['TOOLCHAIN_PREFIX']) + _m32 + "CONFIG_COMMANDS= ",
- " && cp ares_config.h ares_build.h \"$OLDPWD\" ",
- " && cat ares_build.h ",
- " && if [ -r ares_build.h.orig ]; then mv ares_build.h.orig ares_build.h; fi)",
+ " && if [ -r include/ares_build.h ]; then cp include/ares_build.h include/ares_build.h.orig; fi ",
+- " && sh ./configure --disable-dependency-tracking --disable-tests -C " + cflags,
++ " && sh ./configure --host={} --disable-dependency-tracking --disable-tests -C ".format(os.environ['COMMAND_PREFIX']) + cflags,
+ " && cp src/lib/ares_config.h include/ares_build.h \"$OLDPWD\" ",
+ " && cat include/ares_build.h ",
+ " && if [ -r include/ares_build.h.orig ]; then mv include/ares_build.h.orig include/ares_build.h; fi)",
diff --git a/_setuplibev.py b/_setuplibev.py
-index 2a5841bf..b6433c94 100644
+index f05c2fe9..32f9bd81 100644
--- a/_setuplibev.py
+++ b/_setuplibev.py
-@@ -31,7 +31,7 @@ LIBEV_EMBED = should_embed('libev')
- # and the PyPy branch will clean it up.
+@@ -28,7 +28,7 @@ LIBEV_EMBED = should_embed('libev')
+ # Configure libev in place
libev_configure_command = ' '.join([
"(cd ", quoted_dep_abspath('libev'),
-- " && sh ./configure ",
-+ " && sh ./configure --host={} ".format(os.environ['TOOLCHAIN_PREFIX']),
- " && cp config.h \"$OLDPWD\"",
+- " && sh ./configure -C > configure-output.txt",
++ " && sh ./configure --host={} -C > configure-output.txt".format(os.environ['COMMAND_PREFIX']),
")",
- '> configure-output.txt'
+ ])
+
diff --git a/tests/recipes/test_gevent.py b/tests/recipes/test_gevent.py
index 8c6601e255..c434489fe8 100644
--- a/tests/recipes/test_gevent.py
+++ b/tests/recipes/test_gevent.py
@@ -35,9 +35,9 @@ def test_get_recipe_env(self):
'LDFLAGS': mocked_ldflags,
'LDLIBS': mocked_ldlibs,
}
- with patch('pythonforandroid.recipe.CythonRecipe.get_recipe_env') as m_get_recipe_env:
+ with patch('pythonforandroid.recipe.PyProjectRecipe.get_recipe_env') as m_get_recipe_env:
m_get_recipe_env.return_value = mocked_env
- env = self.recipe.get_recipe_env()
+ env = self.recipe.get_recipe_env(self.arch)
expected_cflags = (
' -fomit-frame-pointer -mandroid -isystem /path/to/isystem'
' -isysroot /path/to/sysroot'
@@ -57,11 +57,13 @@ def test_get_recipe_env(self):
)
expected_ldlibs = mocked_ldlibs
expected_libs = '-lm -lpython3.7m -lm'
+ expected_command_prefix = 'aarch64-linux-android'
expected_env = {
'CFLAGS': expected_cflags,
'CPPFLAGS': expected_cppflags,
'LDFLAGS': expected_ldflags,
'LDLIBS': expected_ldlibs,
'LIBS': expected_libs,
+ 'COMMAND_PREFIX': expected_command_prefix,
}
self.assertEqual(expected_env, env)
From e9dbcfd10f8b7786f56125d0a13106484f299bca Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 22 Mar 2025 21:39:19 +0100
Subject: [PATCH 109/158] :fire: Drop libmysqlclient recipe, closes #2808
The recipe was introduced in #587 but has multiple issues and doesn't build.
Notable problems:
- Never updated since initial commit (aside from formatting)
- Uses `master` branch instead of pinning a version
- Contains lots of commented-out code
- Relies on https://github.com/0x-ff/libmysql-android, which:
- Is not an official MySQL repository
- Hasn't been updated since 2013
- Lacks English documentation (only available in Russian)
---
ci/constants.py | 2 -
.../recipes/libmysqlclient/Linux.cmake | 5 --
.../recipes/libmysqlclient/__init__.py | 67 -------------------
.../libmysqlclient/add-custom-platform.patch | 8 ---
.../libmysqlclient/disable-soname.patch | 11 ---
.../libmysqlclient/disable-soversion.patch | 12 ----
.../recipes/libmysqlclient/p4a.cmake | 3 -
tests/recipes/test_libmysqlclient.py | 30 ---------
8 files changed, 138 deletions(-)
delete mode 100644 pythonforandroid/recipes/libmysqlclient/Linux.cmake
delete mode 100644 pythonforandroid/recipes/libmysqlclient/__init__.py
delete mode 100644 pythonforandroid/recipes/libmysqlclient/add-custom-platform.patch
delete mode 100644 pythonforandroid/recipes/libmysqlclient/disable-soname.patch
delete mode 100644 pythonforandroid/recipes/libmysqlclient/disable-soversion.patch
delete mode 100644 pythonforandroid/recipes/libmysqlclient/p4a.cmake
delete mode 100644 tests/recipes/test_libmysqlclient.py
diff --git a/ci/constants.py b/ci/constants.py
index cc1d9ea70a..382a4a0bfe 100644
--- a/ci/constants.py
+++ b/ci/constants.py
@@ -33,8 +33,6 @@ class TargetPython(Enum):
'twisted',
# genericndkbuild is incompatible with sdl2 (which is build by default when targeting sdl2 bootstrap)
'genericndkbuild',
- # libmysqlclient gives a linker failure (See issue #2808)
- 'libmysqlclient',
# boost gives errors (requires numpy? syntax error in .jam?)
'boost',
# libtorrent gives errors (requires boost. Also, see issue #2809, to start with)
diff --git a/pythonforandroid/recipes/libmysqlclient/Linux.cmake b/pythonforandroid/recipes/libmysqlclient/Linux.cmake
deleted file mode 100644
index 42cf0694fd..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/Linux.cmake
+++ /dev/null
@@ -1,5 +0,0 @@
-asdgasdgasdg
-asdg
-asdg
-include(${CMAKE_ROOT}/Modules/Platform/Linux.cmake)
-set(CMAKE_SHARED_LIBRARY_SONAME_C_FLAG "")
diff --git a/pythonforandroid/recipes/libmysqlclient/__init__.py b/pythonforandroid/recipes/libmysqlclient/__init__.py
deleted file mode 100644
index 84fd8d30ac..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/__init__.py
+++ /dev/null
@@ -1,67 +0,0 @@
-from pythonforandroid.logger import shprint
-from pythonforandroid.recipe import Recipe
-from pythonforandroid.util import current_directory
-import sh
-from os.path import join
-
-
-class LibmysqlclientRecipe(Recipe):
- name = 'libmysqlclient'
- version = 'master'
- url = 'https://github.com/0x-ff/libmysql-android/archive/{version}.zip'
- # version = '5.5.47'
- # url = 'http://dev.mysql.com/get/Downloads/MySQL-5.5/mysql-{version}.tar.gz'
- #
- # depends = ['ncurses']
- #
-
- # patches = ['add-custom-platform.patch']
-
- patches = ['disable-soversion.patch']
-
- def should_build(self, arch):
- return not self.has_libs(arch, 'libmysql.so')
-
- def build_arch(self, arch):
- env = self.get_recipe_env(arch)
- with current_directory(join(self.get_build_dir(arch.arch), 'libmysqlclient')):
- shprint(sh.cp, '-t', '.', join(self.get_recipe_dir(), 'p4a.cmake'))
- # ensure_dir('Platform')
- # shprint(sh.cp, '-t', 'Platform', join(self.get_recipe_dir(), 'Linux.cmake'))
- shprint(sh.rm, '-f', 'CMakeCache.txt')
- shprint(sh.cmake, '-G', 'Unix Makefiles',
- # '-DCMAKE_MODULE_PATH=' + join(self.get_build_dir(arch.arch), 'libmysqlclient'),
- '-DCMAKE_INSTALL_PREFIX=./install',
- '-DCMAKE_TOOLCHAIN_FILE=p4a.cmake', _env=env)
- shprint(sh.make, _env=env)
-
- self.install_libs(arch, join('libmysql', 'libmysql.so'))
-
- # def get_recipe_env(self, arch=None):
- # env = super().get_recipe_env(arch)
- # env['WITHOUT_SERVER'] = 'ON'
- # ncurses = self.get_recipe('ncurses', self)
- # # env['CFLAGS'] += ' -I' + join(ncurses.get_build_dir(arch.arch),
- # # 'include')
- # env['CURSES_LIBRARY'] = join(self.ctx.get_libs_dir(arch.arch), 'libncurses.so')
- # env['CURSES_INCLUDE_PATH'] = join(ncurses.get_build_dir(arch.arch),
- # 'include')
- # return env
- #
- # def build_arch(self, arch):
- # env = self.get_recipe_env(arch)
- # with current_directory(self.get_build_dir(arch.arch)):
- # # configure = sh.Command('./configure')
- # # TODO: should add openssl as an optional dep and compile support
- # # shprint(configure, '--enable-shared', '--enable-assembler',
- # # '--enable-thread-safe-client', '--with-innodb',
- # # '--without-server', _env=env)
- # # shprint(sh.make, _env=env)
- # shprint(sh.cmake, '.', '-DCURSES_LIBRARY=' + env['CURSES_LIBRARY'],
- # '-DCURSES_INCLUDE_PATH=' + env['CURSES_INCLUDE_PATH'], _env=env)
- # shprint(sh.make, _env=env)
- #
- # self.install_libs(arch, 'libmysqlclient.so')
-
-
-recipe = LibmysqlclientRecipe()
diff --git a/pythonforandroid/recipes/libmysqlclient/add-custom-platform.patch b/pythonforandroid/recipes/libmysqlclient/add-custom-platform.patch
deleted file mode 100644
index e76c69a723..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/add-custom-platform.patch
+++ /dev/null
@@ -1,8 +0,0 @@
---- libmysqlclient/libmysqlclient/libmysql/CMakeLists.txt 2013-02-27 00:25:45.000000000 -0600
-+++ b/libmysqlclient/libmysql/CMakeLists.txt 2016-01-11 13:28:51.142356988 -0600
-@@ -152,3 +152,5 @@
- ${CMAKE_SOURCE_DIR}/libmysql/libmysqlclient_r${CMAKE_SHARED_LIBRARY_SUFFIX}
- DESTINATION "lib")
- ENDIF(WIN32)
-+
-+LIST(APPEND CMAKE_MODULE_PATH "${CMAKE_SOURCE_PREFIX}")
diff --git a/pythonforandroid/recipes/libmysqlclient/disable-soname.patch b/pythonforandroid/recipes/libmysqlclient/disable-soname.patch
deleted file mode 100644
index 5a4dbf2639..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/disable-soname.patch
+++ /dev/null
@@ -1,11 +0,0 @@
---- libmysqlclient/libmysqlclient/CMakeLists.txt 2013-02-27 00:25:45.000000000 -0600
-+++ b/libmysqlclient/CMakeLists.txt 2016-01-11 13:48:41.672323738 -0600
-@@ -24,6 +24,8 @@
- SET(CMAKE_BUILD_TYPE "Release")
- ENDIF(NOT CMAKE_BUILD_TYPE)
-
-+SET(CMAKE_SHARED_LIBRARY_SONAME_C_FLAG "")
-+
- # This reads user configuration, generated by configure.js.
- IF(WIN32 AND EXISTS ${CMAKE_SOURCE_DIR}/win/configure.data)
- INCLUDE(${CMAKE_SOURCE_DIR}/win/configure.data)
diff --git a/pythonforandroid/recipes/libmysqlclient/disable-soversion.patch b/pythonforandroid/recipes/libmysqlclient/disable-soversion.patch
deleted file mode 100644
index d6353de1cb..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/disable-soversion.patch
+++ /dev/null
@@ -1,12 +0,0 @@
---- libmysqlclient/libmysqlclient/libmysql/CMakeLists.txt 2013-02-27 00:25:45.000000000 -0600
-+++ b/libmysqlclient/libmysql/CMakeLists.txt 2016-01-11 14:00:26.729332913 -0600
-@@ -97,9 +97,6 @@
- ADD_LIBRARY(libmysql SHARED ${CLIENT_SOURCES} libmysql.def)
- TARGET_LINK_LIBRARIES(libmysql ${CMAKE_THREAD_LIBS_INIT})
- STRING(REGEX REPLACE "\\..+" "" LIBMYSQL_SOVERSION ${SHARED_LIB_VERSION})
--SET_TARGET_PROPERTIES(libmysql
-- PROPERTIES VERSION ${SHARED_LIB_VERSION}
-- SOVERSION ${LIBMYSQL_SOVERSION})
- IF(OPENSSL_LIBRARIES)
- TARGET_LINK_LIBRARIES(libmysql ${OPENSSL_LIBRARIES} ${OPENSSL_LIBCRYPTO})
- ENDIF(OPENSSL_LIBRARIES)
diff --git a/pythonforandroid/recipes/libmysqlclient/p4a.cmake b/pythonforandroid/recipes/libmysqlclient/p4a.cmake
deleted file mode 100644
index 9e4c34339d..0000000000
--- a/pythonforandroid/recipes/libmysqlclient/p4a.cmake
+++ /dev/null
@@ -1,3 +0,0 @@
-SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM BOTH)
-SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
-SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
diff --git a/tests/recipes/test_libmysqlclient.py b/tests/recipes/test_libmysqlclient.py
deleted file mode 100644
index 4c85dc92e2..0000000000
--- a/tests/recipes/test_libmysqlclient.py
+++ /dev/null
@@ -1,30 +0,0 @@
-import unittest
-from unittest import mock
-from tests.recipes.recipe_lib_test import BaseTestForCmakeRecipe
-
-
-class TestLibmysqlclientRecipe(BaseTestForCmakeRecipe, unittest.TestCase):
- """
- An unittest for recipe :mod:`~pythonforandroid.recipes.libmysqlclient`
- """
- recipe_name = "libmysqlclient"
-
- @mock.patch("pythonforandroid.recipes.libmysqlclient.sh.rm")
- @mock.patch("pythonforandroid.recipes.libmysqlclient.sh.cp")
- @mock.patch("pythonforandroid.util.chdir")
- @mock.patch("pythonforandroid.build.ensure_dir")
- @mock.patch("shutil.which")
- def test_build_arch(
- self,
- mock_shutil_which,
- mock_ensure_dir,
- mock_current_directory,
- mock_sh_cp,
- mock_sh_rm,
- ):
- # We overwrite the base test method because we need
- # to mock a little more (`sh.cp` and rmdir)
- super().test_build_arch()
- # make sure that the mocked methods are actually called
- mock_sh_cp.assert_called()
- mock_sh_rm.assert_called()
From d9ded99b193c51f4d5fd0ac269bc8da48a778914 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Wed, 26 Mar 2025 09:52:12 +0530
Subject: [PATCH 110/158] `sdl2_image`: fix downloading
---
.../recipes/sdl2_image/__init__.py | 28 +++++++++++++++----
1 file changed, 22 insertions(+), 6 deletions(-)
diff --git a/pythonforandroid/recipes/sdl2_image/__init__.py b/pythonforandroid/recipes/sdl2_image/__init__.py
index b3ac504fbf..8cea604e59 100644
--- a/pythonforandroid/recipes/sdl2_image/__init__.py
+++ b/pythonforandroid/recipes/sdl2_image/__init__.py
@@ -2,11 +2,10 @@
import sh
from pythonforandroid.logger import shprint
from pythonforandroid.recipe import BootstrapNDKRecipe
-from pythonforandroid.util import current_directory
class LibSDL2Image(BootstrapNDKRecipe):
- version = '2.8.0'
+ version = '2.8.2'
url = 'https://github.com/libsdl-org/SDL_image/releases/download/release-{version}/SDL2_image-{version}.tar.gz'
dir_name = 'SDL2_image'
@@ -20,10 +19,27 @@ def get_include_dirs(self, arch):
def prebuild_arch(self, arch):
# We do not have a folder for each arch on BootstrapNDKRecipe, so we
# need to skip the external deps download if we already have done it.
- external_deps_dir = os.path.join(self.get_build_dir(arch.arch), "external")
- if not os.path.exists(os.path.join(external_deps_dir, "libwebp")):
- with current_directory(external_deps_dir):
- shprint(sh.Command("./download.sh"))
+
+ build_dir = self.get_build_dir(arch.arch)
+
+ with open(os.path.join(build_dir, ".gitmodules"), "r") as file:
+ for section in file.read().split('[submodule "')[1:]:
+ line_split = section.split(" = ")
+ # Parse .gitmoulde section
+ clone_path, url, branch = (
+ os.path.join(build_dir, line_split[1].split("\n")[0].strip()),
+ line_split[2].split("\n")[0].strip(),
+ line_split[-1].strip()
+ )
+ # Clone if needed
+ if not os.path.exists(clone_path) or os.listdir(clone_path) == 0:
+ shprint(
+ sh.git, "clone", url,
+ "--depth", "1", "-b",
+ branch, clone_path, "--recursive"
+ )
+ file.close()
+
super().prebuild_arch(arch)
From 58d148bf81e312ae890cb7762ccb9f0f877ee7a4 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Sat, 12 Apr 2025 15:55:55 +0200
Subject: [PATCH 111/158] Add `SDL3` bootstrap (alongside `SDL3`, `SDL3_ttf`,
`SDL3_mixer`, `SDL3_image` recipes) for Kivy `3.0.0` (#3125)
* Add SDL3 bootstrap
* Avoid some DRY issues + minor fixes + version bump
---
pythonforandroid/bootstrap.py | 25 +-
.../bootstraps/_sdl_common/__init__.py | 49 ++
.../{sdl2 => _sdl_common}/build/.gitignore | 0
.../{sdl2 => _sdl_common}/build/blacklist.txt | 0
.../build/jni/Application.mk | 0
.../build/src/main/assets/.gitkeep | 0
.../build/src/main/java}/.gitkeep | 0
.../android/GenericBroadcastReceiver.java | 0
.../GenericBroadcastReceiverCallback.java | 0
.../org/kivy/android/launcher/Project.java | 0
.../kivy/android/launcher/ProjectAdapter.java | 0
.../kivy/android/launcher/ProjectChooser.java | 0
.../build/src/main/jniLibs}/.gitkeep | 0
.../build/src/main/libs}/.gitkeep | 0
.../main/res/drawable-hdpi/ic_launcher.png | Bin
.../main/res/drawable-mdpi/ic_launcher.png | Bin
.../main/res/drawable-xhdpi/ic_launcher.png | Bin
.../main/res/drawable-xxhdpi/ic_launcher.png | Bin
.../build/src/main/res/drawable}/.gitkeep | 0
.../src/main/res/layout/chooser_item.xml | 0
.../build/src/main/res/layout/main.xml | 0
.../src/main/res/layout/project_chooser.xml | 0
.../src/main/res/layout/project_empty.xml | 0
.../src/main/res/mipmap-anydpi-v26}/.gitkeep | 0
.../build/src/main/res/mipmap/.gitkeep | 0
.../build/templates/AndroidManifest.tmpl.xml | 0
.../build/templates/strings.tmpl.xml | 0
.../bootstraps/common/build/build.py | 22 +-
.../build/jni/application/src/Android.mk | 5 +-
.../common/build/jni/application/src/start.c | 8 +-
.../java/org/kivy/android/PythonUtil.java | 4 +
.../jni/application/src/bootstrap_name.h | 1 -
pythonforandroid/bootstraps/sdl2/__init__.py | 50 +-
.../sdl2/build/jni/application/src/Android.mk | 22 +
pythonforandroid/bootstraps/sdl3/__init__.py | 12 +
.../sdl3/build/jni/application/src/Android.mk | 22 +
.../jni/application/src/Android_static.mk | 13 +
.../jni/application/src/bootstrap_name.h | 5 +
.../java/org/kivy/android/PythonActivity.java | 645 ++++++++++++++++++
.../build/src/patches/SDLActivity.java.patch | 49 ++
.../jni/application/src/bootstrap_name.h | 1 -
.../jni/application/src/bootstrap_name.h | 1 -
.../jni/application/src/bootstrap_name.h | 1 -
pythonforandroid/build.py | 4 +
pythonforandroid/recipes/android/__init__.py | 15 +-
.../recipes/android/src/android/_android.pyx | 2 +-
pythonforandroid/recipes/android/src/setup.py | 3 +-
.../recipes/genericndkbuild/__init__.py | 2 +-
pythonforandroid/recipes/kivy/__init__.py | 20 +-
pythonforandroid/recipes/pyjnius/__init__.py | 7 +-
.../recipes/pyjnius/sdl3_jnienv_getter.patch | 24 +
pythonforandroid/recipes/sdl2/__init__.py | 2 +
pythonforandroid/recipes/sdl3/__init__.py | 59 ++
.../recipes/sdl3_image/__init__.py | 41 ++
.../recipes/sdl3_image/enable-webp.patch | 12 +
.../recipes/sdl3_mixer/__init__.py | 45 ++
.../recipes/sdl3_mixer/disable-libgme.patch | 12 +
pythonforandroid/recipes/sdl3_ttf/__init__.py | 39 ++
tests/test_bootstrap.py | 44 +-
tests/test_build.py | 2 +-
tests/test_graph.py | 4 +-
61 files changed, 1177 insertions(+), 95 deletions(-)
create mode 100644 pythonforandroid/bootstraps/_sdl_common/__init__.py
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/.gitignore (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/blacklist.txt (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/jni/Application.mk (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/assets/.gitkeep (100%)
rename pythonforandroid/bootstraps/{sdl2/build/src/main/jniLibs => _sdl_common/build/src/main/java}/.gitkeep (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/java/org/kivy/android/GenericBroadcastReceiver.java (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/java/org/kivy/android/GenericBroadcastReceiverCallback.java (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/java/org/kivy/android/launcher/Project.java (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/java/org/kivy/android/launcher/ProjectAdapter.java (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/java/org/kivy/android/launcher/ProjectChooser.java (100%)
rename pythonforandroid/bootstraps/{sdl2/build/src/main/libs => _sdl_common/build/src/main/jniLibs}/.gitkeep (100%)
rename pythonforandroid/bootstraps/{sdl2/build/src/main/res/drawable => _sdl_common/build/src/main/libs}/.gitkeep (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/drawable-hdpi/ic_launcher.png (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/drawable-mdpi/ic_launcher.png (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/drawable-xhdpi/ic_launcher.png (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/drawable-xxhdpi/ic_launcher.png (100%)
rename pythonforandroid/bootstraps/{sdl2/build/src/main/res/mipmap-anydpi-v26 => _sdl_common/build/src/main/res/drawable}/.gitkeep (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/layout/chooser_item.xml (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/layout/main.xml (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/layout/project_chooser.xml (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/src/main/res/layout/project_empty.xml (100%)
rename pythonforandroid/bootstraps/{sdl2/build/src/main/res/mipmap => _sdl_common/build/src/main/res/mipmap-anydpi-v26}/.gitkeep (100%)
create mode 100644 pythonforandroid/bootstraps/_sdl_common/build/src/main/res/mipmap/.gitkeep
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/templates/AndroidManifest.tmpl.xml (100%)
rename pythonforandroid/bootstraps/{sdl2 => _sdl_common}/build/templates/strings.tmpl.xml (100%)
create mode 100644 pythonforandroid/bootstraps/sdl2/build/jni/application/src/Android.mk
create mode 100644 pythonforandroid/bootstraps/sdl3/__init__.py
create mode 100644 pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android.mk
create mode 100644 pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android_static.mk
create mode 100644 pythonforandroid/bootstraps/sdl3/build/jni/application/src/bootstrap_name.h
create mode 100644 pythonforandroid/bootstraps/sdl3/build/src/main/java/org/kivy/android/PythonActivity.java
create mode 100644 pythonforandroid/bootstraps/sdl3/build/src/patches/SDLActivity.java.patch
create mode 100644 pythonforandroid/recipes/pyjnius/sdl3_jnienv_getter.patch
create mode 100644 pythonforandroid/recipes/sdl3/__init__.py
create mode 100644 pythonforandroid/recipes/sdl3_image/__init__.py
create mode 100644 pythonforandroid/recipes/sdl3_image/enable-webp.patch
create mode 100644 pythonforandroid/recipes/sdl3_mixer/__init__.py
create mode 100644 pythonforandroid/recipes/sdl3_mixer/disable-libgme.patch
create mode 100644 pythonforandroid/recipes/sdl3_ttf/__init__.py
diff --git a/pythonforandroid/bootstrap.py b/pythonforandroid/bootstrap.py
index 8bbbcf0eb6..5f4b3e7ab9 100755
--- a/pythonforandroid/bootstrap.py
+++ b/pythonforandroid/bootstrap.py
@@ -14,6 +14,8 @@
rmdir, move)
from pythonforandroid.recipe import Recipe
+SDL_BOOTSTRAPS = ("sdl2", "sdl3")
+
def copy_files(src_root, dest_root, override=True, symlink=False):
for root, dirnames, filenames in walk(src_root):
@@ -39,7 +41,7 @@ def copy_files(src_root, dest_root, override=True, symlink=False):
default_recipe_priorities = [
- "webview", "sdl2", "service_only" # last is highest
+ "webview", "sdl2", "sdl3", "service_only" # last is highest
]
# ^^ NOTE: these are just the default priorities if no special rules
# apply (which you can find in the code below), so basically if no
@@ -150,18 +152,18 @@ def get_bootstrap_dirs(self):
return bootstrap_dirs
def _copy_in_final_files(self):
- if self.name == "sdl2":
- # Get the paths for copying SDL2's java source code:
- sdl2_recipe = Recipe.get_recipe("sdl2", self.ctx)
- sdl2_build_dir = sdl2_recipe.get_jni_dir()
- src_dir = join(sdl2_build_dir, "SDL", "android-project",
+ if self.name in SDL_BOOTSTRAPS:
+ # Get the paths for copying SDL's java source code:
+ sdl_recipe = Recipe.get_recipe(self.name, self.ctx)
+ sdl_build_dir = sdl_recipe.get_jni_dir()
+ src_dir = join(sdl_build_dir, "SDL", "android-project",
"app", "src", "main", "java",
"org", "libsdl", "app")
target_dir = join(self.dist_dir, 'src', 'main', 'java', 'org',
'libsdl', 'app')
# Do actual copying:
- info('Copying in SDL2 .java files from: ' + str(src_dir))
+ info('Copying in SDL .java files from: ' + str(src_dir))
if not os.path.exists(target_dir):
os.makedirs(target_dir)
copy_files(src_dir, target_dir, override=True)
@@ -193,7 +195,7 @@ def assemble_distribution(self):
@classmethod
def all_bootstraps(cls):
'''Find all the available bootstraps and return them.'''
- forbidden_dirs = ('__pycache__', 'common')
+ forbidden_dirs = ('__pycache__', 'common', '_sdl_common')
bootstraps_dir = join(dirname(__file__), 'bootstraps')
result = set()
for name in listdir(bootstraps_dir):
@@ -272,6 +274,13 @@ def have_dependency_in_recipes(dep):
info('Using sdl2 bootstrap since it is in dependencies')
return cls.get_bootstrap("sdl2", ctx)
+ # Special rule: return SDL3 bootstrap if there's an sdl3 dep:
+ if (have_dependency_in_recipes("sdl3") and
+ "sdl3" in [b.name for b in acceptable_bootstraps]
+ ):
+ info('Using sdl3 bootstrap since it is in dependencies')
+ return cls.get_bootstrap("sdl3", ctx)
+
# Special rule: return "webview" if we depend on common web recipe:
for possible_web_dep in known_web_packages:
if have_dependency_in_recipes(possible_web_dep):
diff --git a/pythonforandroid/bootstraps/_sdl_common/__init__.py b/pythonforandroid/bootstraps/_sdl_common/__init__.py
new file mode 100644
index 0000000000..034e52c7c1
--- /dev/null
+++ b/pythonforandroid/bootstraps/_sdl_common/__init__.py
@@ -0,0 +1,49 @@
+from os.path import join
+
+import sh
+
+from pythonforandroid.toolchain import (
+ Bootstrap, shprint, current_directory, info, info_main)
+from pythonforandroid.util import ensure_dir, rmdir
+
+
+class SDLGradleBootstrap(Bootstrap):
+ name = "_sdl_common"
+
+ recipe_depends = []
+
+ def assemble_distribution(self):
+ info_main("# Creating Android project ({})".format(self.name))
+
+ rmdir(self.dist_dir)
+ info("Copying SDL/gradle build")
+ shprint(sh.cp, "-r", self.build_dir, self.dist_dir)
+
+ # either the build use environment variable (ANDROID_HOME)
+ # or the local.properties if exists
+ with current_directory(self.dist_dir):
+ with open('local.properties', 'w') as fileh:
+ fileh.write('sdk.dir={}'.format(self.ctx.sdk_dir))
+
+ with current_directory(self.dist_dir):
+ info("Copying Python distribution")
+
+ self.distribute_javaclasses(self.ctx.javaclass_dir,
+ dest_dir=join("src", "main", "java"))
+
+ for arch in self.ctx.archs:
+ python_bundle_dir = join(f'_python_bundle__{arch.arch}', '_python_bundle')
+ ensure_dir(python_bundle_dir)
+
+ self.distribute_libs(arch, [self.ctx.get_libs_dir(arch.arch)])
+ site_packages_dir = self.ctx.python_recipe.create_python_bundle(
+ join(self.dist_dir, python_bundle_dir), arch)
+ if not self.ctx.with_debug_symbols:
+ self.strip_libraries(arch)
+ self.fry_eggs(site_packages_dir)
+
+ if 'sqlite3' not in self.ctx.recipe_build_order:
+ with open('blacklist.txt', 'a') as fileh:
+ fileh.write('\nsqlite3/*\nlib-dynload/_sqlite3.so\n')
+
+ super().assemble_distribution()
diff --git a/pythonforandroid/bootstraps/sdl2/build/.gitignore b/pythonforandroid/bootstraps/_sdl_common/build/.gitignore
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/.gitignore
rename to pythonforandroid/bootstraps/_sdl_common/build/.gitignore
diff --git a/pythonforandroid/bootstraps/sdl2/build/blacklist.txt b/pythonforandroid/bootstraps/_sdl_common/build/blacklist.txt
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/blacklist.txt
rename to pythonforandroid/bootstraps/_sdl_common/build/blacklist.txt
diff --git a/pythonforandroid/bootstraps/sdl2/build/jni/Application.mk b/pythonforandroid/bootstraps/_sdl_common/build/jni/Application.mk
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/jni/Application.mk
rename to pythonforandroid/bootstraps/_sdl_common/build/jni/Application.mk
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/assets/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/assets/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/assets/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/assets/.gitkeep
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/jniLibs/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/jniLibs/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/.gitkeep
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/GenericBroadcastReceiver.java b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/GenericBroadcastReceiver.java
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/GenericBroadcastReceiver.java
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/GenericBroadcastReceiver.java
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/GenericBroadcastReceiverCallback.java b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/GenericBroadcastReceiverCallback.java
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/GenericBroadcastReceiverCallback.java
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/GenericBroadcastReceiverCallback.java
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/Project.java b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/Project.java
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/Project.java
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/Project.java
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/ProjectAdapter.java b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/ProjectAdapter.java
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/ProjectAdapter.java
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/ProjectAdapter.java
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/ProjectChooser.java b/pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/ProjectChooser.java
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/java/org/kivy/android/launcher/ProjectChooser.java
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/java/org/kivy/android/launcher/ProjectChooser.java
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/libs/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/jniLibs/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/libs/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/jniLibs/.gitkeep
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/libs/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/libs/.gitkeep
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-hdpi/ic_launcher.png b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-hdpi/ic_launcher.png
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-hdpi/ic_launcher.png
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-hdpi/ic_launcher.png
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-mdpi/ic_launcher.png b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-mdpi/ic_launcher.png
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-mdpi/ic_launcher.png
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-mdpi/ic_launcher.png
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-xhdpi/ic_launcher.png b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-xhdpi/ic_launcher.png
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-xhdpi/ic_launcher.png
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-xhdpi/ic_launcher.png
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-xxhdpi/ic_launcher.png b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-xxhdpi/ic_launcher.png
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/drawable-xxhdpi/ic_launcher.png
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable-xxhdpi/ic_launcher.png
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/mipmap-anydpi-v26/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/mipmap-anydpi-v26/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/drawable/.gitkeep
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/chooser_item.xml b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/chooser_item.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/chooser_item.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/chooser_item.xml
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/main.xml b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/main.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/main.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/main.xml
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/project_chooser.xml b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/project_chooser.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/project_chooser.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/project_chooser.xml
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/project_empty.xml b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/project_empty.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/layout/project_empty.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/layout/project_empty.xml
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/main/res/mipmap/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/mipmap-anydpi-v26/.gitkeep
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/src/main/res/mipmap/.gitkeep
rename to pythonforandroid/bootstraps/_sdl_common/build/src/main/res/mipmap-anydpi-v26/.gitkeep
diff --git a/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/mipmap/.gitkeep b/pythonforandroid/bootstraps/_sdl_common/build/src/main/res/mipmap/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml b/pythonforandroid/bootstraps/_sdl_common/build/templates/AndroidManifest.tmpl.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/templates/AndroidManifest.tmpl.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/templates/AndroidManifest.tmpl.xml
diff --git a/pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml b/pythonforandroid/bootstraps/_sdl_common/build/templates/strings.tmpl.xml
similarity index 100%
rename from pythonforandroid/bootstraps/sdl2/build/templates/strings.tmpl.xml
rename to pythonforandroid/bootstraps/_sdl_common/build/templates/strings.tmpl.xml
diff --git a/pythonforandroid/bootstraps/common/build/build.py b/pythonforandroid/bootstraps/common/build/build.py
index 94382049bf..99c4ea24ab 100644
--- a/pythonforandroid/bootstraps/common/build/build.py
+++ b/pythonforandroid/bootstraps/common/build/build.py
@@ -20,6 +20,7 @@
from fnmatch import fnmatch
import jinja2
+from pythonforandroid.bootstrap import SDL_BOOTSTRAPS
from pythonforandroid.util import rmdir, ensure_dir, max_build_tool_version
@@ -83,7 +84,7 @@ def get_bootstrap_name():
if PYTHON is not None and not exists(PYTHON):
PYTHON = None
-if _bootstrap_name in ('sdl2', 'webview', 'service_only', 'qt'):
+if _bootstrap_name in ('sdl2', 'sdl3', 'webview', 'service_only', 'qt'):
WHITELIST_PATTERNS.append('pyconfig.h')
environment = jinja2.Environment(loader=jinja2.FileSystemLoader(
@@ -220,6 +221,10 @@ def compile_py_file(python_file, optimize_python=True):
return ".".join([os.path.splitext(python_file)[0], "pyc"])
+def is_sdl_bootstrap():
+ return get_bootstrap_name() in SDL_BOOTSTRAPS
+
+
def make_package(args):
# If no launcher is specified, require a main.py/main.pyc:
if (get_bootstrap_name() != "sdl" or args.launcher is None) and \
@@ -541,7 +546,7 @@ def make_package(args):
"debug": "debug" in args.build_mode,
"native_services": args.native_services
}
- if get_bootstrap_name() == "sdl2":
+ if is_sdl_bootstrap():
render_args["url_scheme"] = url_scheme
render(
@@ -596,7 +601,7 @@ def make_package(args):
"args": args,
"private_version": hashlib.sha1(private_version.encode()).hexdigest()
}
- if get_bootstrap_name() == "sdl2":
+ if is_sdl_bootstrap():
render_args["url_scheme"] = url_scheme
render(
'strings.tmpl.xml',
@@ -769,7 +774,7 @@ def create_argument_parser():
ap.add_argument('--private', dest='private',
help='the directory with the app source code files' +
' (containing your main.py entrypoint)',
- required=(get_bootstrap_name() != "sdl2"))
+ required=(not is_sdl_bootstrap()))
ap.add_argument('--package', dest='package',
help=('The name of the java package the project will be'
' packaged under.'),
@@ -787,7 +792,7 @@ def create_argument_parser():
'same number of groups of numbers as previous '
'versions.'),
required=True)
- if get_bootstrap_name() == "sdl2":
+ if is_sdl_bootstrap():
ap.add_argument('--launcher', dest='launcher', action='store_true',
help=('Provide this argument to build a multi-app '
'launcher, rather than a single app.'))
@@ -1044,7 +1049,7 @@ def _read_configuration():
args.orientation, args.manifest_orientation
)
- if get_bootstrap_name() == "sdl2":
+ if is_sdl_bootstrap():
args.sdl_orientation_hint = get_sdl_orientation_hint(args.orientation)
if args.res_xmls and isinstance(args.res_xmls[0], list):
@@ -1073,10 +1078,9 @@ def _read_configuration():
if x.strip() and not x.strip().startswith('#')]
WHITELIST_PATTERNS += patterns
- if args.private is None and \
- get_bootstrap_name() == 'sdl2' and args.launcher is None:
+ if args.private is None and is_sdl_bootstrap() and args.launcher is None:
print('Need --private directory or ' +
- '--launcher (SDL2 bootstrap only)' +
+ '--launcher (SDL2/SDL3 bootstrap only)' +
'to have something to launch inside the .apk!')
sys.exit(1)
make_package(args)
diff --git a/pythonforandroid/bootstraps/common/build/jni/application/src/Android.mk b/pythonforandroid/bootstraps/common/build/jni/application/src/Android.mk
index fb2b17719d..eced58db08 100644
--- a/pythonforandroid/bootstraps/common/build/jni/application/src/Android.mk
+++ b/pythonforandroid/bootstraps/common/build/jni/application/src/Android.mk
@@ -9,12 +9,11 @@ SDL_PATH := ../../SDL
LOCAL_C_INCLUDES := $(LOCAL_PATH)/$(SDL_PATH)/include
# Add your application source files here...
-LOCAL_SRC_FILES := $(SDL_PATH)/src/main/android/SDL_android_main.c \
- start.c
+LOCAL_SRC_FILES := start.c
LOCAL_CFLAGS += -I$(PYTHON_INCLUDE_ROOT) $(EXTRA_CFLAGS)
-LOCAL_SHARED_LIBRARIES := SDL2 python_shared
+LOCAL_SHARED_LIBRARIES := python_shared
LOCAL_LDLIBS := -lGLESv1_CM -lGLESv2 -llog $(EXTRA_LDLIBS)
diff --git a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
index 88999faf98..ef910cab3d 100644
--- a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
+++ b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
@@ -16,10 +16,16 @@
#include "bootstrap_name.h"
-#ifndef BOOTSTRAP_USES_NO_SDL_HEADERS
+#ifdef BOOTSTRAP_NAME_SDL2
#include "SDL.h"
#include "SDL_opengles2.h"
#endif
+
+#ifdef BOOTSTRAP_NAME_SDL3
+#include "SDL3/SDL.h"
+#include "SDL3/SDL_main.h"
+#endif
+
#include "android/log.h"
#define ENTRYPOINT_MAXLEN 128
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
index cc04d83f6b..83d11639bb 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
@@ -49,6 +49,10 @@ protected static ArrayList<String> getLibraries(File libsDir) {
addLibraryIfExists(libsList, "SDL2_image", libsDir);
addLibraryIfExists(libsList, "SDL2_mixer", libsDir);
addLibraryIfExists(libsList, "SDL2_ttf", libsDir);
+ addLibraryIfExists(libsList, "SDL3", libsDir);
+ addLibraryIfExists(libsList, "SDL3_image", libsDir);
+ addLibraryIfExists(libsList, "SDL3_mixer", libsDir);
+ addLibraryIfExists(libsList, "SDL3_ttf", libsDir);
libsList.add("python3.5m");
libsList.add("python3.6m");
libsList.add("python3.7m");
diff --git a/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
index 8a4d8aa464..76709f02c9 100644
--- a/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
+++ b/pythonforandroid/bootstraps/qt/build/jni/application/src/bootstrap_name.h
@@ -1,4 +1,3 @@
-#define BOOTSTRAP_USES_NO_SDL_HEADERS
const char bootstrap_name[] = "qt";
diff --git a/pythonforandroid/bootstraps/sdl2/__init__.py b/pythonforandroid/bootstraps/sdl2/__init__.py
index 9334724a33..0be9f9a23b 100644
--- a/pythonforandroid/bootstraps/sdl2/__init__.py
+++ b/pythonforandroid/bootstraps/sdl2/__init__.py
@@ -1,54 +1,12 @@
-from os.path import join
+from pythonforandroid.bootstraps._sdl_common import SDLGradleBootstrap
-import sh
-from pythonforandroid.toolchain import (
- Bootstrap, shprint, current_directory, info, info_main)
-from pythonforandroid.util import ensure_dir, rmdir
-
-
-class SDL2GradleBootstrap(Bootstrap):
- name = 'sdl2'
+class SDL2GradleBootstrap(SDLGradleBootstrap):
+ name = "sdl2"
recipe_depends = list(
- set(Bootstrap.recipe_depends).union({'sdl2'})
+ set(SDLGradleBootstrap.recipe_depends).union({"sdl2"})
)
- def assemble_distribution(self):
- info_main("# Creating Android project ({})".format(self.name))
-
- rmdir(self.dist_dir)
- info("Copying SDL2/gradle build")
- shprint(sh.cp, "-r", self.build_dir, self.dist_dir)
-
- # either the build use environment variable (ANDROID_HOME)
- # or the local.properties if exists
- with current_directory(self.dist_dir):
- with open('local.properties', 'w') as fileh:
- fileh.write('sdk.dir={}'.format(self.ctx.sdk_dir))
-
- with current_directory(self.dist_dir):
- info("Copying Python distribution")
-
- self.distribute_javaclasses(self.ctx.javaclass_dir,
- dest_dir=join("src", "main", "java"))
-
- for arch in self.ctx.archs:
- python_bundle_dir = join(f'_python_bundle__{arch.arch}', '_python_bundle')
- ensure_dir(python_bundle_dir)
-
- self.distribute_libs(arch, [self.ctx.get_libs_dir(arch.arch)])
- site_packages_dir = self.ctx.python_recipe.create_python_bundle(
- join(self.dist_dir, python_bundle_dir), arch)
- if not self.ctx.with_debug_symbols:
- self.strip_libraries(arch)
- self.fry_eggs(site_packages_dir)
-
- if 'sqlite3' not in self.ctx.recipe_build_order:
- with open('blacklist.txt', 'a') as fileh:
- fileh.write('\nsqlite3/*\nlib-dynload/_sqlite3.so\n')
-
- super().assemble_distribution()
-
bootstrap = SDL2GradleBootstrap()
diff --git a/pythonforandroid/bootstraps/sdl2/build/jni/application/src/Android.mk b/pythonforandroid/bootstraps/sdl2/build/jni/application/src/Android.mk
new file mode 100644
index 0000000000..09fb3b212e
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl2/build/jni/application/src/Android.mk
@@ -0,0 +1,22 @@
+LOCAL_PATH := $(call my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_MODULE := main
+
+SDL_PATH := ../../SDL
+
+LOCAL_C_INCLUDES := $(LOCAL_PATH)/$(SDL_PATH)/include
+
+# Add your application source files here...
+LOCAL_SRC_FILES := start.c
+
+LOCAL_CFLAGS += -I$(PYTHON_INCLUDE_ROOT) $(EXTRA_CFLAGS)
+
+LOCAL_SHARED_LIBRARIES := SDL2 python_shared
+
+LOCAL_LDLIBS := -lGLESv1_CM -lGLESv2 -llog $(EXTRA_LDLIBS)
+
+LOCAL_LDFLAGS += -L$(PYTHON_LINK_ROOT) $(APPLICATION_ADDITIONAL_LDFLAGS)
+
+include $(BUILD_SHARED_LIBRARY)
diff --git a/pythonforandroid/bootstraps/sdl3/__init__.py b/pythonforandroid/bootstraps/sdl3/__init__.py
new file mode 100644
index 0000000000..83f50493f7
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/__init__.py
@@ -0,0 +1,12 @@
+from pythonforandroid.bootstraps._sdl_common import SDLGradleBootstrap
+
+
+class SDL3GradleBootstrap(SDLGradleBootstrap):
+ name = "sdl3"
+
+ recipe_depends = list(
+ set(SDLGradleBootstrap.recipe_depends).union({"sdl3"})
+ )
+
+
+bootstrap = SDL3GradleBootstrap()
diff --git a/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android.mk b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android.mk
new file mode 100644
index 0000000000..14b4e0ed66
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android.mk
@@ -0,0 +1,22 @@
+LOCAL_PATH := $(call my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_MODULE := main
+
+SDL_PATH := ../../SDL
+
+LOCAL_C_INCLUDES := $(LOCAL_PATH)/$(SDL_PATH)/include
+
+# Add your application source files here...
+LOCAL_SRC_FILES := start.c
+
+LOCAL_CFLAGS += -I$(PYTHON_INCLUDE_ROOT) $(EXTRA_CFLAGS)
+
+LOCAL_SHARED_LIBRARIES := SDL3 python_shared
+
+LOCAL_LDLIBS := -lGLESv1_CM -lGLESv2 -llog $(EXTRA_LDLIBS)
+
+LOCAL_LDFLAGS += -L$(PYTHON_LINK_ROOT) $(APPLICATION_ADDITIONAL_LDFLAGS)
+
+include $(BUILD_SHARED_LIBRARY)
diff --git a/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android_static.mk b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android_static.mk
new file mode 100644
index 0000000000..f4ff2462e6
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/Android_static.mk
@@ -0,0 +1,13 @@
+LOCAL_PATH := $(call my-dir)
+
+include $(CLEAR_VARS)
+
+LOCAL_MODULE := main
+
+LOCAL_SRC_FILES := start.c
+
+LOCAL_STATIC_LIBRARIES := SDL3_static
+
+
+include $(BUILD_SHARED_LIBRARY)
+$(call import-module,SDL)LOCAL_PATH := $(call my-dir)
diff --git a/pythonforandroid/bootstraps/sdl3/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/bootstrap_name.h
new file mode 100644
index 0000000000..55096a4aad
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/jni/application/src/bootstrap_name.h
@@ -0,0 +1,5 @@
+
+#define BOOTSTRAP_NAME_SDL3
+
+const char bootstrap_name[] = "SDL3"; // capitalized for historic reasons
+
diff --git a/pythonforandroid/bootstraps/sdl3/build/src/main/java/org/kivy/android/PythonActivity.java b/pythonforandroid/bootstraps/sdl3/build/src/main/java/org/kivy/android/PythonActivity.java
new file mode 100644
index 0000000000..0a9c884cc4
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/src/main/java/org/kivy/android/PythonActivity.java
@@ -0,0 +1,645 @@
+package org.kivy.android;
+
+import java.io.InputStream;
+import java.io.FileWriter;
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.InvocationTargetException;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Timer;
+import java.util.TimerTask;
+
+import android.app.Activity;
+import android.content.Context;
+import android.content.Intent;
+import android.content.pm.ActivityInfo;
+import android.content.pm.PackageManager;
+import android.graphics.Bitmap;
+import android.graphics.BitmapFactory;
+import android.graphics.Color;
+import android.graphics.PixelFormat;
+import android.os.AsyncTask;
+import android.os.Bundle;
+import android.os.PowerManager;
+import android.util.Log;
+import android.view.inputmethod.InputMethodManager;
+import android.view.SurfaceView;
+import android.view.ViewGroup;
+import android.view.View;
+import android.widget.ImageView;
+import android.widget.Toast;
+import android.content.res.Resources.NotFoundException;
+
+import org.libsdl.app.SDLActivity;
+
+import org.kivy.android.launcher.Project;
+
+import org.renpy.android.ResourceManager;
+
+
+public class PythonActivity extends SDLActivity {
+ private static final String TAG = "PythonActivity";
+
+ public static PythonActivity mActivity = null;
+
+ private ResourceManager resourceManager = null;
+ private Bundle mMetaData = null;
+ private PowerManager.WakeLock mWakeLock = null;
+
+ public String getAppRoot() {
+ String app_root = getFilesDir().getAbsolutePath() + "/app";
+ return app_root;
+ }
+
+ @Override
+ protected void onCreate(Bundle savedInstanceState) {
+ Log.v(TAG, "PythonActivity onCreate running");
+ resourceManager = new ResourceManager(this);
+
+ Log.v(TAG, "About to do super onCreate");
+ super.onCreate(savedInstanceState);
+ Log.v(TAG, "Did super onCreate");
+
+ this.mActivity = this;
+ this.showLoadingScreen(this.getLoadingScreen());
+
+ new UnpackFilesTask().execute(getAppRoot());
+ }
+
+ public void loadLibraries() {
+ String app_root = new String(getAppRoot());
+ File app_root_file = new File(app_root);
+ PythonUtil.loadLibraries(app_root_file,
+ new File(getApplicationInfo().nativeLibraryDir));
+ }
+
+ /**
+ * Show an error using a toast. (Only makes sense from non-UI
+ * threads.)
+ */
+ public void toastError(final String msg) {
+
+ final Activity thisActivity = this;
+
+ runOnUiThread(new Runnable () {
+ public void run() {
+ Toast.makeText(thisActivity, msg, Toast.LENGTH_LONG).show();
+ }
+ });
+
+ // Wait to show the error.
+ synchronized (this) {
+ try {
+ this.wait(1000);
+ } catch (InterruptedException e) {
+ }
+ }
+ }
+
+ private class UnpackFilesTask extends AsyncTask<String, Void, String> {
+ @Override
+ protected String doInBackground(String... params) {
+ File app_root_file = new File(params[0]);
+ Log.v(TAG, "Ready to unpack");
+ PythonUtil.unpackAsset(mActivity, "private", app_root_file, true);
+ PythonUtil.unpackPyBundle(mActivity, getApplicationInfo().nativeLibraryDir + "/" + "libpybundle", app_root_file, false);
+ return null;
+ }
+
+ @Override
+ protected void onPostExecute(String result) {
+ // Figure out the directory where the game is. If the game was
+ // given to us via an intent, then we use the scheme-specific
+ // part of that intent to determine the file to launch. We
+ // also use the android.txt file to determine the orientation.
+ //
+ // Otherwise, we use the public data, if we have it, or the
+ // private data if we do not.
+ mActivity.finishLoad();
+
+ // finishLoad called setContentView with the SDL view, which
+ // removed the loading screen. However, we still need it to
+ // show until the app is ready to render, so pop it back up
+ // on top of the SDL view.
+ mActivity.showLoadingScreen(getLoadingScreen());
+
+ String app_root_dir = getAppRoot();
+ if (getIntent() != null && getIntent().getAction() != null &&
+ getIntent().getAction().equals("org.kivy.LAUNCH")) {
+ File path = new File(getIntent().getData().getSchemeSpecificPart());
+
+ Project p = Project.scanDirectory(path);
+ String entry_point = getEntryPoint(p.dir);
+ SDLActivity.nativeSetenv("ANDROID_ENTRYPOINT", p.dir + "/" + entry_point);
+ SDLActivity.nativeSetenv("ANDROID_ARGUMENT", p.dir);
+ SDLActivity.nativeSetenv("ANDROID_APP_PATH", p.dir);
+
+ if (p != null) {
+ if (p.landscape) {
+ setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
+ } else {
+ setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
+ }
+ }
+
+ // Let old apps know they started.
+ try {
+ FileWriter f = new FileWriter(new File(path, ".launch"));
+ f.write("started");
+ f.close();
+ } catch (IOException e) {
+ // pass
+ }
+ } else {
+ String entry_point = getEntryPoint(app_root_dir);
+ SDLActivity.nativeSetenv("ANDROID_ENTRYPOINT", entry_point);
+ SDLActivity.nativeSetenv("ANDROID_ARGUMENT", app_root_dir);
+ SDLActivity.nativeSetenv("ANDROID_APP_PATH", app_root_dir);
+ }
+
+ String mFilesDirectory = mActivity.getFilesDir().getAbsolutePath();
+ Log.v(TAG, "Setting env vars for start.c and Python to use");
+ SDLActivity.nativeSetenv("ANDROID_PRIVATE", mFilesDirectory);
+ SDLActivity.nativeSetenv("ANDROID_UNPACK", app_root_dir);
+ SDLActivity.nativeSetenv("PYTHONHOME", app_root_dir);
+ SDLActivity.nativeSetenv("PYTHONPATH", app_root_dir + ":" + app_root_dir + "/lib");
+ SDLActivity.nativeSetenv("PYTHONOPTIMIZE", "2");
+
+ try {
+ Log.v(TAG, "Access to our meta-data...");
+ mActivity.mMetaData = mActivity.getPackageManager().getApplicationInfo(
+ mActivity.getPackageName(), PackageManager.GET_META_DATA).metaData;
+
+ PowerManager pm = (PowerManager) mActivity.getSystemService(Context.POWER_SERVICE);
+ if ( mActivity.mMetaData.getInt("wakelock") == 1 ) {
+ mActivity.mWakeLock = pm.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "Screen On");
+ mActivity.mWakeLock.acquire();
+ }
+ if ( mActivity.mMetaData.getInt("surface.transparent") != 0 ) {
+ Log.v(TAG, "Surface will be transparent.");
+ getSurface().setZOrderOnTop(true);
+ getSurface().getHolder().setFormat(PixelFormat.TRANSPARENT);
+ } else {
+ Log.i(TAG, "Surface will NOT be transparent");
+ }
+ } catch (PackageManager.NameNotFoundException e) {
+ }
+
+ // Launch app if that hasn't been done yet:
+ if (mActivity.mHasFocus && (
+ // never went into proper resume state:
+ mActivity.mCurrentNativeState == NativeState.INIT ||
+ (
+ // resumed earlier but wasn't ready yet
+ mActivity.mCurrentNativeState == NativeState.RESUMED &&
+ mActivity.mSDLThread == null
+ ))) {
+ // Because sometimes the app will get stuck here and never
+ // actually run, ensure that it gets launched if we're active:
+ mActivity.resumeNativeThread();
+ }
+ }
+
+ @Override
+ protected void onPreExecute() {
+ }
+
+ @Override
+ protected void onProgressUpdate(Void... values) {
+ }
+ }
+
+ public static ViewGroup getLayout() {
+ return mLayout;
+ }
+
+ public static SurfaceView getSurface() {
+ return mSurface;
+ }
+
+ //----------------------------------------------------------------------------
+ // Listener interface for onNewIntent
+ //
+
+ public interface NewIntentListener {
+ void onNewIntent(Intent intent);
+ }
+
+ private List<NewIntentListener> newIntentListeners = null;
+
+ public void registerNewIntentListener(NewIntentListener listener) {
+ if ( this.newIntentListeners == null )
+ this.newIntentListeners = Collections.synchronizedList(new ArrayList<NewIntentListener>());
+ this.newIntentListeners.add(listener);
+ }
+
+ public void unregisterNewIntentListener(NewIntentListener listener) {
+ if ( this.newIntentListeners == null )
+ return;
+ this.newIntentListeners.remove(listener);
+ }
+
+ @Override
+ protected void onNewIntent(Intent intent) {
+ if ( this.newIntentListeners == null )
+ return;
+ this.onResume();
+ synchronized ( this.newIntentListeners ) {
+ Iterator<NewIntentListener> iterator = this.newIntentListeners.iterator();
+ while ( iterator.hasNext() ) {
+ (iterator.next()).onNewIntent(intent);
+ }
+ }
+ }
+
+ //----------------------------------------------------------------------------
+ // Listener interface for onActivityResult
+ //
+
+ public interface ActivityResultListener {
+ void onActivityResult(int requestCode, int resultCode, Intent data);
+ }
+
+ private List<ActivityResultListener> activityResultListeners = null;
+
+ public void registerActivityResultListener(ActivityResultListener listener) {
+ if ( this.activityResultListeners == null )
+ this.activityResultListeners = Collections.synchronizedList(new ArrayList<ActivityResultListener>());
+ this.activityResultListeners.add(listener);
+ }
+
+ public void unregisterActivityResultListener(ActivityResultListener listener) {
+ if ( this.activityResultListeners == null )
+ return;
+ this.activityResultListeners.remove(listener);
+ }
+
+ @Override
+ protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
+ if ( this.activityResultListeners == null )
+ return;
+ this.onResume();
+ synchronized ( this.activityResultListeners ) {
+ Iterator<ActivityResultListener> iterator = this.activityResultListeners.iterator();
+ while ( iterator.hasNext() )
+ (iterator.next()).onActivityResult(requestCode, resultCode, intent);
+ }
+ }
+
+ public static void start_service(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument
+ ) {
+ _do_start_service(
+ serviceTitle, serviceDescription, pythonServiceArgument, true
+ );
+ }
+
+ public static void start_service_not_as_foreground(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument
+ ) {
+ _do_start_service(
+ serviceTitle, serviceDescription, pythonServiceArgument, false
+ );
+ }
+
+ public static void _do_start_service(
+ String serviceTitle,
+ String serviceDescription,
+ String pythonServiceArgument,
+ boolean showForegroundNotification
+ ) {
+ Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
+ String argument = PythonActivity.mActivity.getFilesDir().getAbsolutePath();
+ String app_root_dir = PythonActivity.mActivity.getAppRoot();
+ String entry_point = PythonActivity.mActivity.getEntryPoint(app_root_dir + "/service");
+ serviceIntent.putExtra("androidPrivate", argument);
+ serviceIntent.putExtra("androidArgument", app_root_dir);
+ serviceIntent.putExtra("serviceEntrypoint", "service/" + entry_point);
+ serviceIntent.putExtra("pythonName", "python");
+ serviceIntent.putExtra("pythonHome", app_root_dir);
+ serviceIntent.putExtra("pythonPath", app_root_dir + ":" + app_root_dir + "/lib");
+ serviceIntent.putExtra("serviceStartAsForeground",
+ (showForegroundNotification ? "true" : "false")
+ );
+ serviceIntent.putExtra("serviceTitle", serviceTitle);
+ serviceIntent.putExtra("serviceDescription", serviceDescription);
+ serviceIntent.putExtra("pythonServiceArgument", pythonServiceArgument);
+ PythonActivity.mActivity.startService(serviceIntent);
+ }
+
+ public static void stop_service() {
+ Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
+ PythonActivity.mActivity.stopService(serviceIntent);
+ }
+
+ /** Loading screen view **/
+ public static ImageView mImageView = null;
+ public static View mLottieView = null;
+ /** Whether main routine/actual app has started yet **/
+ protected boolean mAppConfirmedActive = false;
+ /** Timer for delayed loading screen removal. **/
+ protected Timer loadingScreenRemovalTimer = null;
+
+ // Overridden since it's called often, to check whether to remove the
+ // loading screen:
+ @Override
+ protected boolean sendCommand(int command, Object data) {
+ boolean result = super.sendCommand(command, data);
+ considerLoadingScreenRemoval();
+ return result;
+ }
+
+ /** Confirm that the app's main routine has been launched.
+ **/
+ @Override
+ public void appConfirmedActive() {
+ if (!mAppConfirmedActive) {
+ Log.v(TAG, "appConfirmedActive() -> preparing loading screen removal");
+ mAppConfirmedActive = true;
+ considerLoadingScreenRemoval();
+ }
+ }
+
+ /** This is called from various places to check whether the app's main
+ * routine has been launched already, and if it has, then the loading
+ * screen will be removed.
+ **/
+ public void considerLoadingScreenRemoval() {
+ if (loadingScreenRemovalTimer != null)
+ return;
+ runOnUiThread(new Runnable() {
+ public void run() {
+ if (((PythonActivity)PythonActivity.mSingleton).mAppConfirmedActive &&
+ loadingScreenRemovalTimer == null) {
+ // Remove loading screen but with a delay.
+ // (app can use p4a's android.loadingscreen module to
+ // do it quicker if it wants to)
+ // get a handler (call from main thread)
+ // this will run when timer elapses
+ TimerTask removalTask = new TimerTask() {
+ @Override
+ public void run() {
+ // post a runnable to the handler
+ runOnUiThread(new Runnable() {
+ @Override
+ public void run() {
+ PythonActivity activity =
+ ((PythonActivity)PythonActivity.mSingleton);
+ if (activity != null)
+ activity.removeLoadingScreen();
+ }
+ });
+ }
+ };
+ loadingScreenRemovalTimer = new Timer();
+ loadingScreenRemovalTimer.schedule(removalTask, 5000);
+ }
+ }
+ });
+ }
+
+ public void removeLoadingScreen() {
+ runOnUiThread(new Runnable() {
+ public void run() {
+ View view = mLottieView != null ? mLottieView : mImageView;
+ if (view != null && view.getParent() != null) {
+ ((ViewGroup)view.getParent()).removeView(view);
+ mLottieView = null;
+ mImageView = null;
+ }
+ }
+ });
+ }
+
+ public String getEntryPoint(String search_dir) {
+ /* Get the main file (.pyc|.py) depending on if we
+ * have a compiled version or not.
+ */
+ List<String> entryPoints = new ArrayList<String>();
+ entryPoints.add("main.pyc"); // python 3 compiled files
+ for (String value : entryPoints) {
+ File mainFile = new File(search_dir + "/" + value);
+ if (mainFile.exists()) {
+ return value;
+ }
+ }
+ return "main.py";
+ }
+
+ protected void showLoadingScreen(View view) {
+ try {
+ if (mLayout == null) {
+ setContentView(view);
+ } else if (view.getParent() == null) {
+ mLayout.addView(view);
+ }
+ } catch (IllegalStateException e) {
+ // The loading screen can be attempted to be applied twice if app
+ // is tabbed in/out, quickly.
+ // (Gives error "The specified child already has a parent.
+ // You must call removeView() on the child's parent first.")
+ }
+ }
+
+ protected void setBackgroundColor(View view) {
+ /*
+ * Set the presplash loading screen background color
+ * https://developer.android.com/reference/android/graphics/Color.html
+ * Parse the color string, and return the corresponding color-int.
+ * If the string cannot be parsed, throws an IllegalArgumentException exception.
+ * Supported formats are: #RRGGBB #AARRGGBB or one of the following names:
+ * 'red', 'blue', 'green', 'black', 'white', 'gray', 'cyan', 'magenta', 'yellow',
+ * 'lightgray', 'darkgray', 'grey', 'lightgrey', 'darkgrey', 'aqua', 'fuchsia',
+ * 'lime', 'maroon', 'navy', 'olive', 'purple', 'silver', 'teal'.
+ */
+ String backgroundColor = resourceManager.getString("presplash_color");
+ if (backgroundColor != null) {
+ try {
+ view.setBackgroundColor(Color.parseColor(backgroundColor));
+ } catch (IllegalArgumentException e) {}
+ }
+ }
+
+ protected View getLoadingScreen() {
+ // If we have an mLottieView or mImageView already, then do
+ // nothing because it will have already been made the content
+ // view or added to the layout.
+ if (mLottieView != null || mImageView != null) {
+ // we already have a splash screen
+ return mLottieView != null ? mLottieView : mImageView;
+ }
+
+ // first try to load the lottie one
+ try {
+ mLottieView = getLayoutInflater().inflate(
+ this.resourceManager.getIdentifier("lottie", "layout"),
+ mLayout,
+ false
+ );
+ try {
+ if (mLayout == null) {
+ setContentView(mLottieView);
+ } else if (PythonActivity.mLottieView.getParent() == null) {
+ mLayout.addView(mLottieView);
+ }
+ } catch (IllegalStateException e) {
+ // The loading screen can be attempted to be applied twice if app
+ // is tabbed in/out, quickly.
+ // (Gives error "The specified child already has a parent.
+ // You must call removeView() on the child's parent first.")
+ }
+ setBackgroundColor(mLottieView);
+ return mLottieView;
+ }
+ catch (NotFoundException e) {
+ Log.v("SDL", "couldn't find lottie layout or animation, trying static splash");
+ }
+
+ // no lottie asset, try to load the static image then
+ int presplashId = this.resourceManager.getIdentifier("presplash", "drawable");
+ InputStream is = this.getResources().openRawResource(presplashId);
+ Bitmap bitmap = null;
+ try {
+ bitmap = BitmapFactory.decodeStream(is);
+ } finally {
+ try {
+ is.close();
+ } catch (IOException e) {};
+ }
+
+ mImageView = new ImageView(this);
+ mImageView.setImageBitmap(bitmap);
+ setBackgroundColor(mImageView);
+
+ mImageView.setLayoutParams(new ViewGroup.LayoutParams(
+ ViewGroup.LayoutParams.FILL_PARENT,
+ ViewGroup.LayoutParams.FILL_PARENT));
+ mImageView.setScaleType(ImageView.ScaleType.FIT_CENTER);
+ return mImageView;
+ }
+
+ @Override
+ protected void onPause() {
+ if (this.mWakeLock != null && mWakeLock.isHeld()) {
+ this.mWakeLock.release();
+ }
+
+ Log.v(TAG, "onPause()");
+ try {
+ super.onPause();
+ } catch (UnsatisfiedLinkError e) {
+ // Catch pause while still in loading screen failing to
+ // call native function (since it's not yet loaded)
+ }
+ }
+
+ @Override
+ protected void onResume() {
+ if (this.mWakeLock != null) {
+ this.mWakeLock.acquire();
+ }
+ Log.v(TAG, "onResume()");
+ try {
+ super.onResume();
+ } catch (UnsatisfiedLinkError e) {
+ // Catch resume while still in loading screen failing to
+ // call native function (since it's not yet loaded)
+ }
+ considerLoadingScreenRemoval();
+ }
+
+ @Override
+ public void onWindowFocusChanged(boolean hasFocus) {
+ try {
+ super.onWindowFocusChanged(hasFocus);
+ } catch (UnsatisfiedLinkError e) {
+ // Catch window focus while still in loading screen failing to
+ // call native function (since it's not yet loaded)
+ }
+ considerLoadingScreenRemoval();
+ }
+
+ /**
+ * Used by android.permissions p4a module to register a call back after
+ * requesting runtime permissions
+ **/
+ public interface PermissionsCallback {
+ void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults);
+ }
+
+ private PermissionsCallback permissionCallback;
+ private boolean havePermissionsCallback = false;
+
+ public void addPermissionsCallback(PermissionsCallback callback) {
+ permissionCallback = callback;
+ havePermissionsCallback = true;
+ Log.v(TAG, "addPermissionsCallback(): Added callback for onRequestPermissionsResult");
+ }
+
+ @Override
+ public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
+ Log.v(TAG, "onRequestPermissionsResult()");
+ if (havePermissionsCallback) {
+ Log.v(TAG, "onRequestPermissionsResult passed to callback");
+ permissionCallback.onRequestPermissionsResult(requestCode, permissions, grantResults);
+ }
+ super.onRequestPermissionsResult(requestCode, permissions, grantResults);
+ }
+
+ /**
+ * Used by android.permissions p4a module to check a permission
+ **/
+ public boolean checkCurrentPermission(String permission) {
+ if (android.os.Build.VERSION.SDK_INT < 23)
+ return true;
+
+ try {
+ java.lang.reflect.Method methodCheckPermission =
+ Activity.class.getMethod("checkSelfPermission", String.class);
+ Object resultObj = methodCheckPermission.invoke(this, permission);
+ int result = Integer.parseInt(resultObj.toString());
+ if (result == PackageManager.PERMISSION_GRANTED)
+ return true;
+ } catch (IllegalAccessException | NoSuchMethodException |
+ InvocationTargetException e) {
+ }
+ return false;
+ }
+
+ /**
+ * Used by android.permissions p4a module to request runtime permissions
+ **/
+ public void requestPermissionsWithRequestCode(String[] permissions, int requestCode) {
+ if (android.os.Build.VERSION.SDK_INT < 23)
+ return;
+ try {
+ java.lang.reflect.Method methodRequestPermission =
+ Activity.class.getMethod("requestPermissions",
+ String[].class, int.class);
+ methodRequestPermission.invoke(this, permissions, requestCode);
+ } catch (IllegalAccessException | NoSuchMethodException |
+ InvocationTargetException e) {
+ }
+ }
+
+ public void requestPermissions(String[] permissions) {
+ requestPermissionsWithRequestCode(permissions, 1);
+ }
+
+ public static void changeKeyboard(int inputType) {
+ /*
+ if (SDLActivity.keyboardInputType != inputType){
+ SDLActivity.keyboardInputType = inputType;
+ InputMethodManager imm = (InputMethodManager) getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
+ imm.restartInput(mTextEdit);
+ }
+ */
+ }
+}
diff --git a/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLActivity.java.patch b/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLActivity.java.patch
new file mode 100644
index 0000000000..e1ad50cda5
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLActivity.java.patch
@@ -0,0 +1,49 @@
+--- a/src/main/java/org/libsdl/app/SDLActivity.java
++++ b/src/main/java/org/libsdl/app/SDLActivity.java
+@@ -259,6 +259,7 @@
+ String[] arguments = SDLActivity.mSingleton.getArguments();
+
+ Log.v("SDL", "Running main function " + function + " from library " + library);
++ SDLActivity.mSingleton.appConfirmedActive();
+ SDLActivity.nativeRunMain(library, function, arguments);
+ Log.v("SDL", "Finished main function");
+ }
+@@ -351,6 +352,15 @@
+ Log.v(TAG, "Model: " + Build.MODEL);
+ Log.v(TAG, "onCreate()");
+ super.onCreate(savedInstanceState);
++
++ SDL.initialize();
++ // So we can call stuff from static callbacks
++ mSingleton = this;
++ }
++
++ // We don't do this in onCreate because we unpack and load the app data on a thread
++ // and we can't run setup tasks until that thread completes.
++ protected void finishLoad() {
+
+
+ /* Control activity re-creation */
+@@ -1541,8 +1551,22 @@
+ return null;
+ }
+ return SDLActivity.mSurface.getNativeSurface();
++ }
++
++ /**
++ * Calls turnActive() on singleton to keep loading screen active
++ */
++ public static void triggerAppConfirmedActive() {
++ mSingleton.appConfirmedActive();
+ }
+
++ /**
++ * Trick needed for loading screen, overridden by PythonActivity
++ * to keep loading screen active
++ */
++ public void appConfirmedActive() {
++ }
++
+ // Input
+
+ /**
diff --git a/pythonforandroid/bootstraps/service_library/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/service_library/build/jni/application/src/bootstrap_name.h
index 01fd122890..95bd2ef3ae 100644
--- a/pythonforandroid/bootstraps/service_library/build/jni/application/src/bootstrap_name.h
+++ b/pythonforandroid/bootstraps/service_library/build/jni/application/src/bootstrap_name.h
@@ -1,6 +1,5 @@
#define BOOTSTRAP_NAME_LIBRARY
-#define BOOTSTRAP_USES_NO_SDL_HEADERS
const char bootstrap_name[] = "service_library";
diff --git a/pythonforandroid/bootstraps/service_only/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/service_only/build/jni/application/src/bootstrap_name.h
index b93a4ae6ce..9598d1abfe 100644
--- a/pythonforandroid/bootstraps/service_only/build/jni/application/src/bootstrap_name.h
+++ b/pythonforandroid/bootstraps/service_only/build/jni/application/src/bootstrap_name.h
@@ -1,6 +1,5 @@
#define BOOTSTRAP_NAME_SERVICEONLY
-#define BOOTSTRAP_USES_NO_SDL_HEADERS
const char bootstrap_name[] = "service_only";
diff --git a/pythonforandroid/bootstraps/webview/build/jni/application/src/bootstrap_name.h b/pythonforandroid/bootstraps/webview/build/jni/application/src/bootstrap_name.h
index 11c7905dfe..87b64ac724 100644
--- a/pythonforandroid/bootstraps/webview/build/jni/application/src/bootstrap_name.h
+++ b/pythonforandroid/bootstraps/webview/build/jni/application/src/bootstrap_name.h
@@ -1,6 +1,5 @@
#define BOOTSTRAP_NAME_WEBVIEW
-#define BOOTSTRAP_USES_NO_SDL_HEADERS
const char bootstrap_name[] = "webview";
diff --git a/pythonforandroid/build.py b/pythonforandroid/build.py
index 98e2d70b2b..8b1c723423 100644
--- a/pythonforandroid/build.py
+++ b/pythonforandroid/build.py
@@ -892,6 +892,10 @@ def copylibs_function(soname, objs_paths, extra_link_dirs=None, env=None):
'SDL2_ttf',
'SDL2_image',
'SDL2_mixer',
+ 'SDL3',
+ 'SDL3_ttf',
+ 'SDL3_image',
+ 'SDL3_mixer',
)
found_libs = []
sofiles = []
diff --git a/pythonforandroid/recipes/android/__init__.py b/pythonforandroid/recipes/android/__init__.py
index 608d9ee738..5174a69bfa 100644
--- a/pythonforandroid/recipes/android/__init__.py
+++ b/pythonforandroid/recipes/android/__init__.py
@@ -12,7 +12,7 @@ class AndroidRecipe(IncludedFilesBehaviour, CythonRecipe):
src_filename = 'src'
- depends = [('sdl2', 'genericndkbuild'), 'pyjnius']
+ depends = [('sdl3', 'sdl2', 'genericndkbuild'), 'pyjnius']
config_env = {}
@@ -34,8 +34,7 @@ def prebuild_arch(self, arch):
if isinstance(ctx_bootstrap, bytes):
ctx_bootstrap = ctx_bootstrap.decode('utf-8')
bootstrap = bootstrap_name = ctx_bootstrap
- is_sdl2 = (bootstrap_name == "sdl2")
- if bootstrap_name in ["sdl2", "webview", "service_only", "service_library", "qt"]:
+ if bootstrap_name in ["sdl2", "sdl3", "webview", "service_only", "service_library", "qt"]:
java_ns = u'org.kivy.android'
jni_ns = u'org/kivy/android'
else:
@@ -47,7 +46,8 @@ def prebuild_arch(self, arch):
config = {
'BOOTSTRAP': bootstrap,
- 'IS_SDL2': int(is_sdl2),
+ 'IS_SDL2': int(bootstrap_name == "sdl2"),
+ 'IS_SDL3': int(bootstrap_name == "sdl3"),
'PY2': 0,
'JAVA_NAMESPACE': java_ns,
'JNI_NAMESPACE': jni_ns,
@@ -73,11 +73,16 @@ def prebuild_arch(self, arch):
))
self.config_env[key] = str(value)
- if is_sdl2:
+ if bootstrap_name == "sdl2":
fh.write('JNIEnv *SDL_AndroidGetJNIEnv(void);\n')
fh.write(
'#define SDL_ANDROID_GetJNIEnv SDL_AndroidGetJNIEnv\n'
)
+ elif bootstrap_name == "sdl3":
+ fh.write('JNIEnv *SDL_GetAndroidJNIEnv(void);\n')
+ fh.write(
+ '#define SDL_ANDROID_GetJNIEnv SDL_GetAndroidJNIEnv\n'
+ )
else:
fh.write('JNIEnv *WebView_AndroidGetJNIEnv(void);\n')
fh.write(
diff --git a/pythonforandroid/recipes/android/src/android/_android.pyx b/pythonforandroid/recipes/android/src/android/_android.pyx
index 6708b846a8..1d6e65a161 100644
--- a/pythonforandroid/recipes/android/src/android/_android.pyx
+++ b/pythonforandroid/recipes/android/src/android/_android.pyx
@@ -194,7 +194,7 @@ TYPE_TEXT_VARIATION_POSTAL_ADDRESS = 112
TYPE_TEXT_VARIATION_URI = 16
TYPE_CLASS_PHONE = 3
-IF BOOTSTRAP == 'sdl2':
+IF BOOTSTRAP in ['sdl2', 'sdl3']:
def remove_presplash():
# Remove android presplash in SDL2 bootstrap.
mActivity.removeLoadingScreen()
diff --git a/pythonforandroid/recipes/android/src/setup.py b/pythonforandroid/recipes/android/src/setup.py
index bcd411f46b..0f5ceb1fd3 100755
--- a/pythonforandroid/recipes/android/src/setup.py
+++ b/pythonforandroid/recipes/android/src/setup.py
@@ -3,7 +3,8 @@
library_dirs = ['libs/' + os.environ['ARCH']]
lib_dict = {
- 'sdl2': ['SDL2', 'SDL2_image', 'SDL2_mixer', 'SDL2_ttf']
+ 'sdl2': ['SDL2', 'SDL2_image', 'SDL2_mixer', 'SDL2_ttf'],
+ 'sdl3': ['SDL3', 'SDL3_image', 'SDL3_mixer', 'SDL3_ttf'],
}
sdl_libs = lib_dict.get(os.environ['BOOTSTRAP'], ['main'])
diff --git a/pythonforandroid/recipes/genericndkbuild/__init__.py b/pythonforandroid/recipes/genericndkbuild/__init__.py
index 8b2a9c26a2..9e85aac5d6 100644
--- a/pythonforandroid/recipes/genericndkbuild/__init__.py
+++ b/pythonforandroid/recipes/genericndkbuild/__init__.py
@@ -10,7 +10,7 @@ class GenericNDKBuildRecipe(BootstrapNDKRecipe):
url = None
depends = ['python3']
- conflicts = ['sdl2']
+ conflicts = ['sdl2', 'sdl3']
def should_build(self, arch):
return True
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index b28e03a7a5..a5030abf95 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -26,8 +26,9 @@ class KivyRecipe(CythonRecipe):
url = 'https://github.com/kivy/kivy/archive/{version}.zip'
name = 'kivy'
- depends = ['sdl2', 'pyjnius', 'setuptools']
+ depends = [('sdl2', 'sdl3'), 'pyjnius', 'setuptools']
python_depends = ['certifi', 'chardet', 'idna', 'requests', 'urllib3', 'filetype']
+ hostpython_prerequisites = []
# sdl-gl-swapwindow-nogil.patch is needed to avoid a deadlock.
# See: https://github.com/kivy/kivy/pull/8025
@@ -53,7 +54,7 @@ def cythonize_build(self, env, build_dir='.'):
def cythonize_file(self, env, build_dir, filename):
# We can ignore a few files that aren't important to the
# android build, and may not work on Android anyway
- do_not_cythonize = ['window_x11.pyx', ]
+ do_not_cythonize = ['window_x11.pyx', 'camera_avfoundation.pyx', 'img_imageio.pyx', 'egl_angle_metal.pyx']
if basename(filename) in do_not_cythonize:
return
super().cythonize_file(env, build_dir, filename)
@@ -73,6 +74,21 @@ def get_recipe_env(self, arch):
*sdl2_mixer_recipe.get_include_dirs(arch),
join(self.ctx.bootstrap.build_dir, 'jni', 'SDL2_ttf'),
])
+ if "sdl3" in self.ctx.recipe_build_order:
+ sdl3_mixer_recipe = self.get_recipe("sdl3_mixer", self.ctx)
+ sdl3_image_recipe = self.get_recipe("sdl3_image", self.ctx)
+ sdl3_ttf_recipe = self.get_recipe("sdl3_ttf", self.ctx)
+ sdl3_recipe = self.get_recipe("sdl3", self.ctx)
+ env["USE_SDL3"] = "1"
+ env["KIVY_SPLIT_EXAMPLES"] = "1"
+ env["KIVY_SDL3_PATH"] = ":".join(
+ [
+ *sdl3_mixer_recipe.get_include_dirs(arch),
+ *sdl3_image_recipe.get_include_dirs(arch),
+ *sdl3_ttf_recipe.get_include_dirs(arch),
+ *sdl3_recipe.get_include_dirs(arch),
+ ]
+ )
return env
diff --git a/pythonforandroid/recipes/pyjnius/__init__.py b/pythonforandroid/recipes/pyjnius/__init__.py
index 0bcb74d392..00369df216 100644
--- a/pythonforandroid/recipes/pyjnius/__init__.py
+++ b/pythonforandroid/recipes/pyjnius/__init__.py
@@ -9,10 +9,13 @@ class PyjniusRecipe(CythonRecipe):
version = '1.6.1'
url = 'https://github.com/kivy/pyjnius/archive/{version}.zip'
name = 'pyjnius'
- depends = [('genericndkbuild', 'sdl2'), 'six']
+ depends = [('genericndkbuild', 'sdl2', 'sdl3'), 'six']
site_packages_name = 'jnius'
- patches = [('genericndkbuild_jnienv_getter.patch', will_build('genericndkbuild'))]
+ patches = [
+ ('genericndkbuild_jnienv_getter.patch', will_build('genericndkbuild')),
+ ('sdl3_jnienv_getter.patch', will_build('sdl3')),
+ ]
def get_recipe_env(self, arch):
env = super().get_recipe_env(arch)
diff --git a/pythonforandroid/recipes/pyjnius/sdl3_jnienv_getter.patch b/pythonforandroid/recipes/pyjnius/sdl3_jnienv_getter.patch
new file mode 100644
index 0000000000..d91da76fbb
--- /dev/null
+++ b/pythonforandroid/recipes/pyjnius/sdl3_jnienv_getter.patch
@@ -0,0 +1,24 @@
+diff -Naur pyjnius.orig/jnius/env.py pyjnius/jnius/env.py
+--- pyjnius.orig/jnius/env.py 2022-05-28 11:16:02.000000000 +0200
++++ pyjnius/jnius/env.py 2022-05-28 11:18:30.000000000 +0200
+@@ -268,7 +268,7 @@
+
+ class AndroidJavaLocation(UnixJavaLocation):
+ def get_libraries(self):
+- return ['SDL2', 'log']
++ return ['SDL3', 'log']
+
+ def get_include_dirs(self):
+ # When cross-compiling for Android, we should not use the include dirs
+diff -Naur pyjnius.orig/jnius/jnius_jvm_android.pxi pyjnius/jnius/jnius_jvm_android.pxi
+--- pyjnius.orig/jnius/jnius_jvm_android.pxi 2022-05-28 11:16:02.000000000 +0200
++++ pyjnius/jnius/jnius_jvm_android.pxi 2022-05-28 11:17:17.000000000 +0200
+@@ -1,6 +1,6 @@
+ # on android, rely on SDL to get the JNI env
+-cdef extern JNIEnv *SDL_AndroidGetJNIEnv()
++cdef extern JNIEnv *SDL_GetAndroidJNIEnv()
+
+
+ cdef JNIEnv *get_platform_jnienv() except NULL:
+- return <JNIEnv*>SDL_AndroidGetJNIEnv()
++ return <JNIEnv*>SDL_GetAndroidJNIEnv()
diff --git a/pythonforandroid/recipes/sdl2/__init__.py b/pythonforandroid/recipes/sdl2/__init__.py
index 8d5fbc2dc2..cd0185c717 100644
--- a/pythonforandroid/recipes/sdl2/__init__.py
+++ b/pythonforandroid/recipes/sdl2/__init__.py
@@ -10,6 +10,8 @@ class LibSDL2Recipe(BootstrapNDKRecipe):
url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL2-{version}.tar.gz"
md5sum = 'a344eb827a03045c9b399e99af4af13d'
+ conflicts = ['sdl3']
+
dir_name = 'SDL'
depends = ['sdl2_image', 'sdl2_mixer', 'sdl2_ttf']
diff --git a/pythonforandroid/recipes/sdl3/__init__.py b/pythonforandroid/recipes/sdl3/__init__.py
new file mode 100644
index 0000000000..6b9c8ee7c6
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3/__init__.py
@@ -0,0 +1,59 @@
+from os.path import exists, join
+
+from pythonforandroid.recipe import BootstrapNDKRecipe
+from pythonforandroid.toolchain import current_directory, shprint
+import sh
+
+
+class LibSDL3Recipe(BootstrapNDKRecipe):
+ version = "3.2.10"
+ url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL3-{version}.tar.gz"
+ md5sum = "70cda886bcf5a4fdac550db796d2efa2"
+
+ conflicts = ["sdl2"]
+
+ dir_name = "SDL"
+
+ depends = ["sdl3_image", "sdl3_mixer", "sdl3_ttf"]
+
+ def get_recipe_env(
+ self, arch=None, with_flags_in_cc=True, with_python=True
+ ):
+ env = super().get_recipe_env(
+ arch=arch,
+ with_flags_in_cc=with_flags_in_cc,
+ with_python=with_python,
+ )
+ env["APP_ALLOW_MISSING_DEPS"] = "true"
+ return env
+
+ def get_include_dirs(self, arch):
+ return [
+ join(self.ctx.bootstrap.build_dir, "jni", "SDL", "include"),
+ join(self.ctx.bootstrap.build_dir, "jni", "SDL", "include", "SDL3"),
+ ]
+
+ def should_build(self, arch):
+ libdir = join(self.get_build_dir(arch.arch), "../..", "libs", arch.arch)
+ libs = [
+ "libmain.so",
+ "libSDL3.so",
+ "libSDL3_image.so",
+ "libSDL3_mixer.so",
+ "libSDL3_ttf.so",
+ ]
+ return not all(exists(join(libdir, x)) for x in libs)
+
+ def build_arch(self, arch):
+ env = self.get_recipe_env(arch)
+
+ with current_directory(self.get_jni_dir()):
+ shprint(
+ sh.Command(join(self.ctx.ndk_dir, "ndk-build")),
+ "V=1",
+ "NDK_DEBUG=" + ("1" if self.ctx.build_as_debuggable else "0"),
+ _env=env,
+ )
+
+
+recipe = LibSDL3Recipe()
diff --git a/pythonforandroid/recipes/sdl3_image/__init__.py b/pythonforandroid/recipes/sdl3_image/__init__.py
new file mode 100644
index 0000000000..f6d705b168
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3_image/__init__.py
@@ -0,0 +1,41 @@
+import os
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.recipe import BootstrapNDKRecipe
+from pythonforandroid.util import current_directory
+
+
+class LibSDL3Image(BootstrapNDKRecipe):
+ version = "3.2.4"
+ url = "https://github.com/libsdl-org/SDL_image/releases/download/release-{version}/SDL3_image-{version}.tar.gz"
+ dir_name = "SDL3_image"
+
+ patches = ["enable-webp.patch"]
+
+ def get_include_dirs(self, arch):
+ return [
+ os.path.join(
+ self.ctx.bootstrap.build_dir, "jni", "SDL3_image", "include"
+ ),
+ os.path.join(
+ self.ctx.bootstrap.build_dir,
+ "jni",
+ "SDL3_image",
+ "include",
+ "SDL3_image",
+ ),
+ ]
+
+ def prebuild_arch(self, arch):
+ # We do not have a folder for each arch on BootstrapNDKRecipe, so we
+ # need to skip the external deps download if we already have done it.
+ external_deps_dir = os.path.join(
+ self.get_build_dir(arch.arch), "external"
+ )
+ if not os.path.exists(os.path.join(external_deps_dir, "libwebp")):
+ with current_directory(external_deps_dir):
+ shprint(sh.Command("./download.sh"))
+ super().prebuild_arch(arch)
+
+
+recipe = LibSDL3Image()
diff --git a/pythonforandroid/recipes/sdl3_image/enable-webp.patch b/pythonforandroid/recipes/sdl3_image/enable-webp.patch
new file mode 100644
index 0000000000..98d72f2017
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3_image/enable-webp.patch
@@ -0,0 +1,12 @@
+diff -Naur SDL2_image.orig/Android.mk SDL2_image/Android.mk
+--- SDL2_image.orig/Android.mk 2022-10-03 20:51:52.000000000 +0200
++++ SDL2_image/Android.mk 2022-10-03 20:52:48.000000000 +0200
+@@ -32,7 +32,7 @@
+
+ # Enable this if you want to support loading WebP images
+ # The library path should be a relative path to this directory.
+-SUPPORT_WEBP ?= false
++SUPPORT_WEBP := true
+ WEBP_LIBRARY_PATH := external/libwebp
+
+
diff --git a/pythonforandroid/recipes/sdl3_mixer/__init__.py b/pythonforandroid/recipes/sdl3_mixer/__init__.py
new file mode 100644
index 0000000000..c60c5bc157
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3_mixer/__init__.py
@@ -0,0 +1,45 @@
+import os
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.recipe import BootstrapNDKRecipe
+from pythonforandroid.util import current_directory
+
+
+class LibSDL3Mixer(BootstrapNDKRecipe):
+ version = "72a73339731a12c1002f9caca64f1ab924938102"
+ # url = "https://github.com/libsdl-org/SDL_ttf/releases/download/release-{version}/SDL3_ttf-{version}.tar.gz"
+ url = "https://github.com/libsdl-org/SDL_mixer/archive/{version}.tar.gz"
+ dir_name = "SDL3_mixer"
+
+ patches = ["disable-libgme.patch"]
+
+ def get_include_dirs(self, arch):
+ return [
+ os.path.join(
+ self.ctx.bootstrap.build_dir, "jni", "SDL3_mixer", "include"
+ ),
+ os.path.join(
+ self.ctx.bootstrap.build_dir,
+ "jni",
+ "SDL3_mixer",
+ "include",
+ "SDL3_mixer",
+ ),
+ ]
+
+ def prebuild_arch(self, arch):
+ # We do not have a folder for each arch on BootstrapNDKRecipe, so we
+ # need to skip the external deps download if we already have done it.
+ external_deps_dir = os.path.join(
+ self.get_build_dir(arch.arch), "external"
+ )
+
+ if not os.path.exists(
+ os.path.join(external_deps_dir, "libgme", "Android.mk")
+ ):
+ with current_directory(external_deps_dir):
+ shprint(sh.Command("./download.sh"))
+ super().prebuild_arch(arch)
+
+
+recipe = LibSDL3Mixer()
diff --git a/pythonforandroid/recipes/sdl3_mixer/disable-libgme.patch b/pythonforandroid/recipes/sdl3_mixer/disable-libgme.patch
new file mode 100644
index 0000000000..233808e7db
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3_mixer/disable-libgme.patch
@@ -0,0 +1,12 @@
+diff -Naur SDL3_mixer.orig/Android.mk SDL3_mixer/Android.mk
+--- SDL3_mixer.orig/Android.mk 2025-03-16 21:05:19
++++ SDL3_mixer/Android.mk 2025-03-16 21:06:33
+@@ -31,7 +31,7 @@
+ WAVPACK_LIBRARY_PATH := external/wavpack
+
+ # Enable this if you want to support loading music via libgme
+-SUPPORT_GME ?= true
++SUPPORT_GME ?= false
+ GME_LIBRARY_PATH := external/libgme
+
+ # Enable this if you want to support loading MOD music via XMP-lite
diff --git a/pythonforandroid/recipes/sdl3_ttf/__init__.py b/pythonforandroid/recipes/sdl3_ttf/__init__.py
new file mode 100644
index 0000000000..a0ebfac7a5
--- /dev/null
+++ b/pythonforandroid/recipes/sdl3_ttf/__init__.py
@@ -0,0 +1,39 @@
+import os
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.recipe import BootstrapNDKRecipe
+from pythonforandroid.util import current_directory
+
+
+class LibSDL3TTF(BootstrapNDKRecipe):
+ version = "3.2.2"
+ url = "https://github.com/libsdl-org/SDL_ttf/releases/download/release-{version}/SDL3_ttf-{version}.tar.gz"
+ dir_name = "SDL3_ttf"
+
+ def get_include_dirs(self, arch):
+ return [
+ os.path.join(
+ self.ctx.bootstrap.build_dir, "jni", "SDL3_ttf", "include"
+ ),
+ os.path.join(
+ self.ctx.bootstrap.build_dir,
+ "jni",
+ "SDL3_ttf",
+ "include",
+ "SDL3_ttf",
+ ),
+ ]
+
+ def prebuild_arch(self, arch):
+ # We do not have a folder for each arch on BootstrapNDKRecipe, so we
+ # need to skip the external deps download if we already have done it.
+ external_deps_dir = os.path.join(
+ self.get_build_dir(arch.arch), "external"
+ )
+ if not os.path.exists(os.path.join(external_deps_dir, "harfbuzz")):
+ with current_directory(external_deps_dir):
+ shprint(sh.Command("./download.sh"))
+ super().prebuild_arch(arch)
+
+
+recipe = LibSDL3TTF()
diff --git a/tests/test_bootstrap.py b/tests/test_bootstrap.py
index eea284b8c9..fc15bd45e6 100644
--- a/tests/test_bootstrap.py
+++ b/tests/test_bootstrap.py
@@ -144,9 +144,17 @@ def test_all_bootstraps(self):
"""A test which will initialize a bootstrap and will check if the
method :meth:`~pythonforandroid.bootstrap.Bootstrap.all_bootstraps `
returns the expected values, which should be: `empty", `service_only`,
- `webview`, `sdl2` and `qt`
+ `webview`, `sdl2`, `sdl3` and `qt`
"""
- expected_bootstraps = {"empty", "service_only", "service_library", "webview", "sdl2", "qt"}
+ expected_bootstraps = {
+ "empty",
+ "service_only",
+ "service_library",
+ "webview",
+ "sdl2",
+ "sdl3",
+ "qt",
+ }
set_of_bootstraps = Bootstrap.all_bootstraps()
self.assertEqual(
expected_bootstraps, expected_bootstraps & set_of_bootstraps
@@ -180,8 +188,9 @@ def test_expand_dependencies_with_pure_python_package(self):
expanded_result = expand_dependencies(
["python3", "kivy", "peewee"], self.ctx
)
- # we expect to one results for python3
- self.assertEqual(len(expanded_result), 1)
+ # we expect to 2 results for python3
+ # (python3, sdl2/sdl3 [one is blacklisted])
+ self.assertEqual(len(expanded_result), 2)
self.assertIsInstance(expanded_result, list)
for i in expanded_result:
self.assertIsInstance(i, list)
@@ -347,13 +356,13 @@ def bootstrap_name(self):
@mock.patch("pythonforandroid.bootstraps.qt.open", create=True)
@mock.patch("pythonforandroid.bootstraps.service_only.open", create=True)
@mock.patch("pythonforandroid.bootstraps.webview.open", create=True)
- @mock.patch("pythonforandroid.bootstraps.sdl2.open", create=True)
+ @mock.patch("pythonforandroid.bootstraps._sdl_common.open", create=True)
@mock.patch("pythonforandroid.distribution.open", create=True)
@mock.patch("pythonforandroid.bootstrap.Bootstrap.strip_libraries")
@mock.patch("pythonforandroid.util.exists")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.bootstrap.listdir")
- @mock.patch("pythonforandroid.bootstraps.sdl2.rmdir")
+ @mock.patch("pythonforandroid.bootstraps._sdl_common.rmdir")
@mock.patch("pythonforandroid.bootstraps.service_only.rmdir")
@mock.patch("pythonforandroid.bootstraps.webview.rmdir")
@mock.patch("pythonforandroid.bootstrap.sh.cp")
@@ -368,7 +377,7 @@ def test_assemble_distribution(
mock_ensure_dir,
mock_strip_libraries,
mock_open_dist_files,
- mock_open_sdl2_files,
+ mock_open_sdl_files,
mock_open_webview_files,
mock_open_service_only_files,
mock_open_qt_files
@@ -409,7 +418,8 @@ def test_assemble_distribution(
mock_open_dist_files.assert_called_once_with("dist_info.json", "w")
mock_open_bootstraps = {
- "sdl2": mock_open_sdl2_files,
+ "sdl2": mock_open_sdl_files,
+ "sdl3": mock_open_sdl_files,
"webview": mock_open_webview_files,
"service_only": mock_open_service_only_files,
"qt": mock_open_qt_files
@@ -419,6 +429,10 @@ def test_assemble_distribution(
mock.call("local.properties", "w"),
mock.call("blacklist.txt", "a"),
],
+ "sdl3": [
+ mock.call("local.properties", "w"),
+ mock.call("blacklist.txt", "a"),
+ ],
"webview": [mock.call("local.properties", "w")],
"service_only": [mock.call("local.properties", "w")],
"qt": [mock.call("local.properties", "w")]
@@ -432,7 +446,7 @@ def test_assemble_distribution(
mock.call().__enter__().write("sdk.dir=/opt/android/android-sdk"),
mock_open_bs.mock_calls,
)
- if self.bootstrap_name == "sdl2":
+ if self.bootstrap_name in ["sdl2", "sdl3"]:
self.assertIn(
mock.call()
.__enter__()
@@ -615,6 +629,18 @@ def bootstrap_name(self):
return "sdl2"
+class TestBootstrapSdl3(GenericBootstrapTest, unittest.TestCase):
+ """
+ An inherited class of `GenericBootstrapTest` and `unittest.TestCase` which
+ will be used to perform tests for
+ :class:`~pythonforandroid.bootstraps.sdl3.BootstrapSdl3`.
+ """
+
+ @property
+ def bootstrap_name(self):
+ return "sdl3"
+
+
class TestBootstrapServiceOnly(GenericBootstrapTest, unittest.TestCase):
"""
An inherited class of `GenericBootstrapTest` and `unittest.TestCase` which
diff --git a/tests/test_build.py b/tests/test_build.py
index 49f6311621..57db29b148 100644
--- a/tests/test_build.py
+++ b/tests/test_build.py
@@ -82,7 +82,7 @@ def test_android_manifest_xml(self):
"native_services": args.native_services
}
environment = jinja2.Environment(
- loader=jinja2.FileSystemLoader('pythonforandroid/bootstraps/sdl2/build/templates/')
+ loader=jinja2.FileSystemLoader('pythonforandroid/bootstraps/_sdl_common/build/templates/')
)
template = environment.get_template('AndroidManifest.tmpl.xml')
xml = template.render(**render_args)
diff --git a/tests/test_graph.py b/tests/test_graph.py
index f7647bcac7..1ac9c68090 100644
--- a/tests/test_graph.py
+++ b/tests/test_graph.py
@@ -101,9 +101,9 @@ def test_blacklist():
get_recipe_order_and_bootstrap(ctx, ["flask", "kivy"], wbootstrap)
assert "conflict" in e_info.value.message.lower()
- # We should no longer get a conflict blacklisting sdl2:
+ # We should no longer get a conflict blacklisting sdl2 and sdl3
get_recipe_order_and_bootstrap(
- ctx, ["flask", "kivy"], wbootstrap, blacklist=["sdl2"]
+ ctx, ["flask", "kivy"], wbootstrap, blacklist=["sdl2", "sdl3"]
)
From 53deb85d5f51a6ae5e15785411aa1923b7eafff6 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Fri, 18 Apr 2025 22:08:28 +0530
Subject: [PATCH 112/158] `kivy` and `pyjnius`: switch to `PyProjectRecipe`
(#3139)
---
pythonforandroid/recipes/kivy/__init__.py | 43 ++++++-------------
.../recipes/kivy/use_cython.patch | 11 +++++
pythonforandroid/recipes/pyjnius/__init__.py | 19 ++++++--
.../recipes/pyjnius/use_cython.patch | 13 ++++++
4 files changed, 53 insertions(+), 33 deletions(-)
create mode 100644 pythonforandroid/recipes/kivy/use_cython.patch
create mode 100644 pythonforandroid/recipes/pyjnius/use_cython.patch
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index a5030abf95..f80024155f 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -1,10 +1,9 @@
-import glob
-from os.path import basename, exists, join
+from os.path import join
import sys
import packaging.version
import sh
-from pythonforandroid.recipe import CythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
from pythonforandroid.toolchain import current_directory, shprint
@@ -21,7 +20,7 @@ def is_kivy_affected_by_deadlock_issue(recipe=None, arch=None):
) < packaging.version.Version("2.2.0.dev0")
-class KivyRecipe(CythonRecipe):
+class KivyRecipe(PyProjectRecipe):
version = '2.3.1'
url = 'https://github.com/kivy/kivy/archive/{version}.zip'
name = 'kivy'
@@ -33,34 +32,20 @@ class KivyRecipe(CythonRecipe):
# sdl-gl-swapwindow-nogil.patch is needed to avoid a deadlock.
# See: https://github.com/kivy/kivy/pull/8025
# WARNING: Remove this patch when a new Kivy version is released.
- patches = [("sdl-gl-swapwindow-nogil.patch", is_kivy_affected_by_deadlock_issue)]
+ patches = [("sdl-gl-swapwindow-nogil.patch", is_kivy_affected_by_deadlock_issue), "use_cython.patch"]
- def cythonize_build(self, env, build_dir='.'):
- super().cythonize_build(env, build_dir=build_dir)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
- if not exists(join(build_dir, 'kivy', 'include')):
- return
+ # Taken from CythonRecipe
+ env['LDFLAGS'] = env['LDFLAGS'] + ' -L{} '.format(
+ self.ctx.get_libs_dir(arch.arch) +
+ ' -L{} '.format(self.ctx.libs_dir) +
+ ' -L{}'.format(join(self.ctx.bootstrap.build_dir, 'obj', 'local',
+ arch.arch)))
+ env['LDSHARED'] = env['CC'] + ' -shared'
+ env['LIBLINK'] = 'NOTNONE'
- # If kivy is new enough to use the include dir, copy it
- # manually to the right location as we bypass this stage of
- # the build
- with current_directory(build_dir):
- build_libs_dirs = glob.glob(join('build', 'lib.*'))
-
- for dirn in build_libs_dirs:
- shprint(sh.cp, '-r', join('kivy', 'include'),
- join(dirn, 'kivy'))
-
- def cythonize_file(self, env, build_dir, filename):
- # We can ignore a few files that aren't important to the
- # android build, and may not work on Android anyway
- do_not_cythonize = ['window_x11.pyx', 'camera_avfoundation.pyx', 'img_imageio.pyx', 'egl_angle_metal.pyx']
- if basename(filename) in do_not_cythonize:
- return
- super().cythonize_file(env, build_dir, filename)
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
# NDKPLATFORM is our switch for detecting Android platform, so can't be None
env['NDKPLATFORM'] = "NOTNONE"
if 'sdl2' in self.ctx.recipe_build_order:
diff --git a/pythonforandroid/recipes/kivy/use_cython.patch b/pythonforandroid/recipes/kivy/use_cython.patch
new file mode 100644
index 0000000000..2a0d2074ba
--- /dev/null
+++ b/pythonforandroid/recipes/kivy/use_cython.patch
@@ -0,0 +1,11 @@
+--- kivy-master/setup.py 2025-02-25 03:08:18.000000000 +0530
++++ kivy-master.mod/setup.py 2025-03-01 13:10:24.227808612 +0530
+@@ -249,7 +249,7 @@
+ # This determines whether Cython specific functionality may be used.
+ can_use_cython = True
+
+-if platform in ('ios', 'android'):
++if platform in ('ios'):
+ # NEVER use or declare cython on these platforms
+ print('Not using cython on %s' % platform)
+ can_use_cython = False
diff --git a/pythonforandroid/recipes/pyjnius/__init__.py b/pythonforandroid/recipes/pyjnius/__init__.py
index 00369df216..cd80b46342 100644
--- a/pythonforandroid/recipes/pyjnius/__init__.py
+++ b/pythonforandroid/recipes/pyjnius/__init__.py
@@ -1,11 +1,11 @@
-from pythonforandroid.recipe import CythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
from pythonforandroid.toolchain import shprint, current_directory, info
from pythonforandroid.patching import will_build
import sh
from os.path import join
-class PyjniusRecipe(CythonRecipe):
+class PyjniusRecipe(PyProjectRecipe):
version = '1.6.1'
url = 'https://github.com/kivy/pyjnius/archive/{version}.zip'
name = 'pyjnius'
@@ -13,12 +13,23 @@ class PyjniusRecipe(CythonRecipe):
site_packages_name = 'jnius'
patches = [
+ "use_cython.patch",
('genericndkbuild_jnienv_getter.patch', will_build('genericndkbuild')),
('sdl3_jnienv_getter.patch', will_build('sdl3')),
]
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+
+ # Taken from CythonRecipe
+ env['LDFLAGS'] = env['LDFLAGS'] + ' -L{} '.format(
+ self.ctx.get_libs_dir(arch.arch) +
+ ' -L{} '.format(self.ctx.libs_dir) +
+ ' -L{}'.format(join(self.ctx.bootstrap.build_dir, 'obj', 'local',
+ arch.arch)))
+ env['LDSHARED'] = env['CC'] + ' -shared'
+ env['LIBLINK'] = 'NOTNONE'
+
# NDKPLATFORM is our switch for detecting Android platform, so can't be None
env['NDKPLATFORM'] = "NOTNONE"
return env
diff --git a/pythonforandroid/recipes/pyjnius/use_cython.patch b/pythonforandroid/recipes/pyjnius/use_cython.patch
new file mode 100644
index 0000000000..59265e99a7
--- /dev/null
+++ b/pythonforandroid/recipes/pyjnius/use_cython.patch
@@ -0,0 +1,13 @@
+--- pyjnius-1.6.1/setup.py 2023-11-05 21:07:43.000000000 +0530
++++ pyjnius-1.6.1.mod/setup.py 2025-03-01 14:47:11.964847337 +0530
+@@ -59,10 +59,6 @@
+ if NDKPLATFORM is not None and getenv('LIBLINK'):
+ PLATFORM = 'android'
+
+-# detect platform
+-if PLATFORM == 'android':
+- PYX_FILES = [fn[:-3] + 'c' for fn in PYX_FILES]
+-
+ JAVA=get_java_setup(PLATFORM)
+
+ assert JAVA.is_jdk(), "You need a JDK, we only found a JRE. Try setting JAVA_HOME"
From f491c6eb20007f9d3897c5c83cfb625c9fee07cf Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Fri, 18 Apr 2025 22:11:04 +0530
Subject: [PATCH 113/158] `primp`: update to `0.14.0` (#3138)
---
.../recipes/{pyreqwest_impersonate => primp}/__init__.py | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
rename pythonforandroid/recipes/{pyreqwest_impersonate => primp}/__init__.py (81%)
diff --git a/pythonforandroid/recipes/pyreqwest_impersonate/__init__.py b/pythonforandroid/recipes/primp/__init__.py
similarity index 81%
rename from pythonforandroid/recipes/pyreqwest_impersonate/__init__.py
rename to pythonforandroid/recipes/primp/__init__.py
index 7e8d5db9ae..b932eb3e61 100644
--- a/pythonforandroid/recipes/pyreqwest_impersonate/__init__.py
+++ b/pythonforandroid/recipes/primp/__init__.py
@@ -2,9 +2,9 @@
from pythonforandroid.recipe import RustCompiledComponentsRecipe
-class Pyreqwest_impersonateRecipe(RustCompiledComponentsRecipe):
- version = "v0.4.5"
- url = "https://github.com/deedy5/pyreqwest_impersonate/archive/refs/tags/{version}.tar.gz"
+class PrimpRecipe(RustCompiledComponentsRecipe):
+ version = "v0.14.0"
+ url = "https://github.com/deedy5/primp/archive/refs/tags/{version}.tar.gz"
def get_recipe_env_post(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
@@ -30,4 +30,4 @@ def build_arch(self, arch):
prebuild_(arch)
-recipe = Pyreqwest_impersonateRecipe()
+recipe = PrimpRecipe()
From d9e943b9934d06733d9e03e14b72c2f5e58bde82 Mon Sep 17 00:00:00 2001
From: William Scaff <williamscaff.usp@gmail.com>
Date: Mon, 5 May 2025 18:59:32 -0300
Subject: [PATCH 114/158] Fix Cmake compatibility issue
The jpeg recipe doesn't compile anymore because CMake dropped compatibility with versions below 3.5.
"Calls to cmake_minimum_required() or cmake_policy() that set the policy version to an older value now issue an error."
Source: https://cmake.org/cmake/help/latest/release/4.0.html
---
pythonforandroid/recipes/jpeg/__init__.py | 3 +++
1 file changed, 3 insertions(+)
diff --git a/pythonforandroid/recipes/jpeg/__init__.py b/pythonforandroid/recipes/jpeg/__init__.py
index 436dc129ba..33a9ba44da 100644
--- a/pythonforandroid/recipes/jpeg/__init__.py
+++ b/pythonforandroid/recipes/jpeg/__init__.py
@@ -48,6 +48,9 @@ def build_arch(self, arch):
# Force disable shared, with the static ones is enough
'-DENABLE_SHARED=0',
'-DENABLE_STATIC=1',
+
+ # Fix cmake compatibility issue
+ '-DCMAKE_POLICY_VERSION_MINIMUM=3.5',
_env=env)
shprint(sh.make, _env=env)
From bd32caf401aa499b267e462c5fa1e5e57fd08271 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 11 May 2025 20:39:31 +0530
Subject: [PATCH 115/158] `pyjnius`: pin cython version
---
pythonforandroid/recipes/pyjnius/__init__.py | 1 +
.../recipes/pyjnius/cython_version_pin.patch | 9 +++++++++
2 files changed, 10 insertions(+)
create mode 100644 pythonforandroid/recipes/pyjnius/cython_version_pin.patch
diff --git a/pythonforandroid/recipes/pyjnius/__init__.py b/pythonforandroid/recipes/pyjnius/__init__.py
index cd80b46342..c6b6746fa1 100644
--- a/pythonforandroid/recipes/pyjnius/__init__.py
+++ b/pythonforandroid/recipes/pyjnius/__init__.py
@@ -14,6 +14,7 @@ class PyjniusRecipe(PyProjectRecipe):
patches = [
"use_cython.patch",
+ "cython_version_pin.patch",
('genericndkbuild_jnienv_getter.patch', will_build('genericndkbuild')),
('sdl3_jnienv_getter.patch', will_build('sdl3')),
]
diff --git a/pythonforandroid/recipes/pyjnius/cython_version_pin.patch b/pythonforandroid/recipes/pyjnius/cython_version_pin.patch
new file mode 100644
index 0000000000..3d5cea2350
--- /dev/null
+++ b/pythonforandroid/recipes/pyjnius/cython_version_pin.patch
@@ -0,0 +1,9 @@
+--- pyjnius-1.6.1/pyproject.toml 2023-11-05 21:07:43.000000000 +0530
++++ pyjnius-1.6.1.mod/pyproject.toml 2025-05-11 20:31:14.699072764 +0530
+@@ -2,5 +2,5 @@
+ requires = [
+ "setuptools>=58.0.0",
+ "wheel",
+- "Cython"
++ "Cython==3.0.0"
+ ]
From 5a844e7248ea195f40e1d01c21db3614f7ce89f1 Mon Sep 17 00:00:00 2001
From: clayote <zacharyspector@gmail.com>
Date: Tue, 13 May 2025 04:48:57 +1200
Subject: [PATCH 116/158] Fix typo in comment
---
pythonforandroid/recipes/sdl2_image/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl2_image/__init__.py b/pythonforandroid/recipes/sdl2_image/__init__.py
index 8cea604e59..fe2e258e16 100644
--- a/pythonforandroid/recipes/sdl2_image/__init__.py
+++ b/pythonforandroid/recipes/sdl2_image/__init__.py
@@ -25,7 +25,7 @@ def prebuild_arch(self, arch):
with open(os.path.join(build_dir, ".gitmodules"), "r") as file:
for section in file.read().split('[submodule "')[1:]:
line_split = section.split(" = ")
- # Parse .gitmoulde section
+ # Parse .gitmodule section
clone_path, url, branch = (
os.path.join(build_dir, line_split[1].split("\n")[0].strip()),
line_split[2].split("\n")[0].strip(),
From 2d5ef7f54af54b46e9913cdce74a4bb917a32260 Mon Sep 17 00:00:00 2001
From: clayote <public@zacharyspector.com>
Date: Tue, 13 May 2025 04:50:34 +1200
Subject: [PATCH 117/158] Remove redundant `file.close()`
The end of a `with` block already does that.
---
pythonforandroid/recipes/sdl2_image/__init__.py | 1 -
1 file changed, 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl2_image/__init__.py b/pythonforandroid/recipes/sdl2_image/__init__.py
index fe2e258e16..340e688315 100644
--- a/pythonforandroid/recipes/sdl2_image/__init__.py
+++ b/pythonforandroid/recipes/sdl2_image/__init__.py
@@ -38,7 +38,6 @@ def prebuild_arch(self, arch):
"--depth", "1", "-b",
branch, clone_path, "--recursive"
)
- file.close()
super().prebuild_arch(arch)
From d2911a7b25bf95f258d194466a56245782f067ee Mon Sep 17 00:00:00 2001
From: clayote <public@zacharyspector.com>
Date: Tue, 13 May 2025 05:01:20 +1200
Subject: [PATCH 118/158] Fix bad empty-directory check
---
pythonforandroid/recipes/sdl2_image/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl2_image/__init__.py b/pythonforandroid/recipes/sdl2_image/__init__.py
index 340e688315..39411a740f 100644
--- a/pythonforandroid/recipes/sdl2_image/__init__.py
+++ b/pythonforandroid/recipes/sdl2_image/__init__.py
@@ -32,7 +32,7 @@ def prebuild_arch(self, arch):
line_split[-1].strip()
)
# Clone if needed
- if not os.path.exists(clone_path) or os.listdir(clone_path) == 0:
+ if not os.path.exists(clone_path) or not os.listdir(clone_path):
shprint(
sh.git, "clone", url,
"--depth", "1", "-b",
From 2473ddeabb286b826924c49e62ea6c42a9acfd7a Mon Sep 17 00:00:00 2001
From: Kenechukwu Akubue <kengoon19@gmail.com>
Date: Fri, 16 May 2025 00:36:17 +0100
Subject: [PATCH 119/158] Patch SDLSurface to allow touch to be intercepted by
python application
---
.../build/src/patches/SDLSurface.java.patch | 25 ++++++++++++++++++
.../build/src/patches/SDLSurface.java.patch | 26 +++++++++++++++++++
2 files changed, 51 insertions(+)
create mode 100644 pythonforandroid/bootstraps/sdl2/build/src/patches/SDLSurface.java.patch
create mode 100644 pythonforandroid/bootstraps/sdl3/build/src/patches/SDLSurface.java.patch
diff --git a/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLSurface.java.patch b/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLSurface.java.patch
new file mode 100644
index 0000000000..591e977176
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl2/build/src/patches/SDLSurface.java.patch
@@ -0,0 +1,25 @@
+--- a/src/main/java/org/libsdl/app/SDLSurface.java
++++ b/src/main/java/org/libsdl/app/SDLSurface.java
+@@ -193,9 +193,22 @@
+ return SDLActivity.handleKeyEvent(v, keyCode, event, null);
+ }
+
++ public interface OnInterceptTouchListener {
++ boolean onTouch(MotionEvent ev);
++ }
++
++ private OnInterceptTouchListener mOnInterceptTouchListener = null;
++
++ public void setInterceptTouchListener(OnInterceptTouchListener listener) {
++ this.mOnInterceptTouchListener = listener;
++ }
++
+ // Touch events
+ @Override
+ public boolean onTouch(View v, MotionEvent event) {
++ if (mOnInterceptTouchListener != null)
++ if (mOnInterceptTouchListener.onTouch(event))
++ return false;
+ /* Ref: http://developer.android.com/training/gestures/multi.html */
+ int touchDevId = event.getDeviceId();
+ final int pointerCount = event.getPointerCount();
diff --git a/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLSurface.java.patch b/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLSurface.java.patch
new file mode 100644
index 0000000000..4253ca7d33
--- /dev/null
+++ b/pythonforandroid/bootstraps/sdl3/build/src/patches/SDLSurface.java.patch
@@ -0,0 +1,26 @@
+--- a/src/main/java/org/libsdl/app/SDLSurface.java
++++ b/src/main/java/org/libsdl/app/SDLSurface.java
+@@ -232,9 +232,23 @@
+ }
+ }
+
++ public interface OnInterceptTouchListener {
++ boolean onTouch(MotionEvent ev);
++ }
++
++ private OnInterceptTouchListener mOnInterceptTouchListener = null;
++
++ public void setInterceptTouchListener(OnInterceptTouchListener listener) {
++ this.mOnInterceptTouchListener = listener;
++ }
++
+ // Touch events
+ @Override
+ public boolean onTouch(View v, MotionEvent event) {
++ // Allow touch to be intercepted by python application
++ if (mOnInterceptTouchListener != null)
++ if (mOnInterceptTouchListener.onTouch(event))
++ return false;
+ /* Ref: http://developer.android.com/training/gestures/multi.html */
+ int touchDevId = event.getDeviceId();
+ final int pointerCount = event.getPointerCount();
From 64ce0f8a2a0258e8451e08b9c91c6141db5446b2 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Mathias=20Lindstr=C3=B6m?= <kuzeyron@gmail.com>
Date: Fri, 11 Jul 2025 12:32:20 +0300
Subject: [PATCH 120/158] [Display-Cutout] More tools for the developer + minor
changes (#3170)
* More tools for the developer + minor changes
---
.../android/src/android/display_cutout.py | 41 ++++++++++++++-----
1 file changed, 30 insertions(+), 11 deletions(-)
diff --git a/pythonforandroid/recipes/android/src/android/display_cutout.py b/pythonforandroid/recipes/android/src/android/display_cutout.py
index dbe5d8a137..09f89672da 100644
--- a/pythonforandroid/recipes/android/src/android/display_cutout.py
+++ b/pythonforandroid/recipes/android/src/android/display_cutout.py
@@ -4,7 +4,8 @@
from android import mActivity
__all__ = ('get_cutout_pos', 'get_cutout_size', 'get_width_of_bar',
- 'get_height_of_bar', 'get_size_of_bar')
+ 'get_height_of_bar', 'get_size_of_bar', 'get_width_of_bar',
+ 'get_cutout_mode')
def _core_cutout():
@@ -15,20 +16,20 @@ def _core_cutout():
def get_cutout_pos():
- """ Get position of the display-cutout.
- Returns integer for each positions (xy)
+ """Get position of the display-cutout.
+ Returns integer for each positions (xy)
"""
try:
cutout = _core_cutout()
- return int(cutout.left), Window.height - int(cutout.height())
+ return int(cutout.left), int(Window.height - cutout.height())
except Exception:
# Doesn't have a camera builtin with the display
return 0, 0
def get_cutout_size():
- """ Get the size (xy) of the front camera.
- Returns size with float values
+ """Get the size (xy) of the front camera.
+ Returns size with float values
"""
try:
cutout = _core_cutout()
@@ -39,8 +40,8 @@ def get_cutout_size():
def get_height_of_bar(bar_target=None):
- """ Get the height of either statusbar or navigationbar
- bar_target = status or navigation and defaults to status
+ """Get the height of either statusbar or navigationbar
+ bar_target = status or navigation and defaults to status
"""
bar_target = bar_target or 'status'
@@ -61,12 +62,30 @@ def get_height_of_bar(bar_target=None):
def get_width_of_bar(bar_target=None):
- " Get the width of the bar "
+ """Get the width of the bar"""
return Window.width
def get_size_of_bar(bar_target=None):
- """ Get the size of either statusbar or navigationbar
- bar_target = status or navigation and defaults to status
+ """Get the size of either statusbar or navigationbar
+ bar_target = status or navigation and defaults to status
"""
return get_width_of_bar(), get_height_of_bar(bar_target)
+
+
+def get_heights_of_both_bars():
+ """Return heights of both bars"""
+ return get_height_of_bar('status'), get_height_of_bar('navigation')
+
+
+def get_cutout_mode():
+ """Return mode for cutout supported applications"""
+ LayoutParams = autoclass('android.view.WindowManager$LayoutParams')
+ window = mActivity.getWindow()
+ layout_params = window.getAttributes()
+ cutout_mode = layout_params.layoutInDisplayCutoutMode
+ cutout_modes = {LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS: 'always',
+ LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT: 'default',
+ LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES: 'shortEdges'}
+
+ return cutout_modes.get(cutout_mode, 'never')
From 9e9469588b90c3087d3840c23adfa3332d5a4de8 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Mathias=20Lindstr=C3=B6m?= <kuzeyron@gmail.com>
Date: Sat, 12 Jul 2025 13:08:39 +0300
Subject: [PATCH 121/158] [Display-Cutout] Added missing features due to
limitations
---
.../android/src/android/display_cutout.py | 26 ++++++++++++-------
1 file changed, 17 insertions(+), 9 deletions(-)
diff --git a/pythonforandroid/recipes/android/src/android/display_cutout.py b/pythonforandroid/recipes/android/src/android/display_cutout.py
index 09f89672da..a52868502d 100644
--- a/pythonforandroid/recipes/android/src/android/display_cutout.py
+++ b/pythonforandroid/recipes/android/src/android/display_cutout.py
@@ -80,12 +80,20 @@ def get_heights_of_both_bars():
def get_cutout_mode():
"""Return mode for cutout supported applications"""
- LayoutParams = autoclass('android.view.WindowManager$LayoutParams')
- window = mActivity.getWindow()
- layout_params = window.getAttributes()
- cutout_mode = layout_params.layoutInDisplayCutoutMode
- cutout_modes = {LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS: 'always',
- LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT: 'default',
- LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES: 'shortEdges'}
-
- return cutout_modes.get(cutout_mode, 'never')
+ BuildVersion = autoclass('android.os.Build$VERSION')
+ cutout_modes = {}
+
+ if BuildVersion.SDK_INT >= 28:
+ LayoutParams = autoclass('android.view.WindowManager$LayoutParams')
+ window = mActivity.getWindow()
+ layout_params = window.getAttributes()
+ cutout_mode = layout_params.layoutInDisplayCutoutMode
+ cutout_modes.update({LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT: 'default',
+ LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES: 'shortEdges'})
+
+ if BuildVersion.SDK_INT >= 30:
+ cutout_modes[LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS] = 'always'
+
+ return cutout_modes.get(cutout_mode, 'never')
+
+ return None
From a8f2ca1c5b1bb6696b47fdf2c052285e116e0ebe Mon Sep 17 00:00:00 2001
From: Filipe Marchesini <filipe.marchesini@gmail.com>
Date: Tue, 22 Jul 2025 18:42:40 -0300
Subject: [PATCH 122/158] Update Android Gradle Plugin to `8.11.0` and Gradle
Wrapper to `8.14.3` + update default API target to `35` in tests (#3172)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* 🔧 build: upgrade Gradle wrapper to `8.14.3`
* `distributionUrl` now points to `gradle‑8.14.3‑all.zip`
* 🔧 build: bump AGP to `8.11.0`, replace `jcenter()` with `mavenCentral()`
* Update `android-api` to 35 in `testapps/on_device_unit_tests/setup.py`
* Update target android.api to 35 in buildozer.spec
* Update `ANDROID_API_LEVEL` to 35 in `android.mk`
* Set `APP_PLATFORM` in `Application.mk` to use `NDK_API`
* Add `Application.mk` with `APP_PLATFORM` for service_library bootstrap
Hopefully [this](https://github.com/kivy/python-for-android/actions/runs/16309986846/job/46063689527?pr=3172) will pass
* Define `APP_ABI` variable in `Application.mk` for arch targeting
---
ci/makefiles/android.mk | 2 +-
.../common/build/gradle/wrapper/gradle-wrapper.properties | 2 +-
.../bootstraps/common/build/templates/build.tmpl.gradle | 6 +++---
.../bootstraps/service_library/build/jni/Application.mk | 2 ++
.../bootstraps/webview/build/jni/Application.mk | 1 +
testapps/on_device_unit_tests/buildozer.spec | 2 +-
testapps/on_device_unit_tests/setup.py | 6 +++---
7 files changed, 12 insertions(+), 9 deletions(-)
create mode 100644 pythonforandroid/bootstraps/service_library/build/jni/Application.mk
diff --git a/ci/makefiles/android.mk b/ci/makefiles/android.mk
index 2041a6ce76..f04ebcdc54 100644
--- a/ci/makefiles/android.mk
+++ b/ci/makefiles/android.mk
@@ -6,7 +6,7 @@ ANDROID_NDK_VERSION_LEGACY ?= 21e
ANDROID_SDK_TOOLS_VERSION ?= 6514223
ANDROID_SDK_BUILD_TOOLS_VERSION ?= 29.0.3
ANDROID_HOME ?= $(HOME)/.android
-ANDROID_API_LEVEL ?= 27
+ANDROID_API_LEVEL ?= 35
# per OS dictionary-like
UNAME_S := $(shell uname -s)
diff --git a/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties b/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
index 8f174bc31b..4a2223651a 100644
--- a/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
+++ b/pythonforandroid/bootstraps/common/build/gradle/wrapper/gradle-wrapper.properties
@@ -3,4 +3,4 @@ distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
-distributionUrl=https\://services.gradle.org/distributions/gradle-8.0.2-all.zip
+distributionUrl=https\://services.gradle.org/distributions/gradle-8.14.3-all.zip
diff --git a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
index 750a435d99..370b3957f9 100644
--- a/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
+++ b/pythonforandroid/bootstraps/common/build/templates/build.tmpl.gradle
@@ -2,17 +2,17 @@
buildscript {
repositories {
google()
- jcenter()
+ mavenCentral()
}
dependencies {
- classpath 'com.android.tools.build:gradle:8.1.1'
+ classpath 'com.android.tools.build:gradle:8.11.0'
}
}
allprojects {
repositories {
google()
- jcenter()
+ mavenCentral()
{%- for repo in args.gradle_repositories %}
{{repo}}
{%- endfor %}
diff --git a/pythonforandroid/bootstraps/service_library/build/jni/Application.mk b/pythonforandroid/bootstraps/service_library/build/jni/Application.mk
new file mode 100644
index 0000000000..f6893f30e4
--- /dev/null
+++ b/pythonforandroid/bootstraps/service_library/build/jni/Application.mk
@@ -0,0 +1,2 @@
+APP_PLATFORM := $(NDK_API)
+APP_ABI := $(ARCH)
diff --git a/pythonforandroid/bootstraps/webview/build/jni/Application.mk b/pythonforandroid/bootstraps/webview/build/jni/Application.mk
index e79e378f94..15598537ca 100644
--- a/pythonforandroid/bootstraps/webview/build/jni/Application.mk
+++ b/pythonforandroid/bootstraps/webview/build/jni/Application.mk
@@ -5,3 +5,4 @@
# APP_ABI := armeabi armeabi-v7a x86
APP_ABI := $(ARCH)
+APP_PLATFORM := $(NDK_API)
diff --git a/testapps/on_device_unit_tests/buildozer.spec b/testapps/on_device_unit_tests/buildozer.spec
index b372d5faa5..17fca683f1 100644
--- a/testapps/on_device_unit_tests/buildozer.spec
+++ b/testapps/on_device_unit_tests/buildozer.spec
@@ -88,7 +88,7 @@ fullscreen = 0
#android.permissions = INTERNET
# (int) Target Android API, should be as high as possible.
-#android.api = 27
+android.api = 35
# (int) Minimum API your APK will support.
#android.minapi = 21
diff --git a/testapps/on_device_unit_tests/setup.py b/testapps/on_device_unit_tests/setup.py
index a63ca8bcb7..4d2019390c 100644
--- a/testapps/on_device_unit_tests/setup.py
+++ b/testapps/on_device_unit_tests/setup.py
@@ -42,7 +42,7 @@
'requirements':
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
- 'android-api': 27,
+ 'android-api': 35,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
@@ -56,7 +56,7 @@
'requirements':
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
- 'android-api': 27,
+ 'android-api': 35,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
@@ -68,7 +68,7 @@
'aar':
{
'requirements' : 'python3',
- 'android-api': 27,
+ 'android-api': 35,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'arm64-v8a',
From 2ebea90d9d7e965ef0754466abd185a41ab298aa Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Fri, 25 Jul 2025 00:43:06 +0530
Subject: [PATCH 123/158] Update: `numpy`, `pandas`, `sdl2` to newer versions
which support `ndk28c` (#3164)
---
ci/makefiles/android.mk | 4 ++--
doc/source/quickstart.rst | 4 ++--
doc/source/testing_pull_requests.rst | 4 ++--
pythonforandroid/recipe.py | 8 ++++++++
pythonforandroid/recipes/numpy/__init__.py | 13 +++----------
pythonforandroid/recipes/pandas/__init__.py | 8 ++++----
pythonforandroid/recipes/python3/__init__.py | 11 ++++++++++-
pythonforandroid/recipes/sdl2/__init__.py | 4 ++--
pythonforandroid/recommendations.py | 2 +-
pythonforandroid/util.py | 17 ++++++++++++-----
testapps/on_device_unit_tests/setup.py | 6 +++---
.../setup_testapp_python3_sqlite_openssl.py | 2 +-
testapps/setup_vispy.py | 2 +-
testapps/testlauncherreboot_setup/sdl2.py | 2 +-
tests/recipes/test_pandas.py | 2 +-
tests/test_recipe.py | 2 ++
16 files changed, 55 insertions(+), 36 deletions(-)
diff --git a/ci/makefiles/android.mk b/ci/makefiles/android.mk
index f04ebcdc54..c7196b0e24 100644
--- a/ci/makefiles/android.mk
+++ b/ci/makefiles/android.mk
@@ -1,12 +1,12 @@
# Downloads and installs the Android SDK depending on supplied platform: darwin or linux
# Those android NDK/SDK variables can be override when running the file
-ANDROID_NDK_VERSION ?= 25b
+ANDROID_NDK_VERSION ?= 28c
ANDROID_NDK_VERSION_LEGACY ?= 21e
ANDROID_SDK_TOOLS_VERSION ?= 6514223
ANDROID_SDK_BUILD_TOOLS_VERSION ?= 29.0.3
ANDROID_HOME ?= $(HOME)/.android
-ANDROID_API_LEVEL ?= 35
+ANDROID_API_LEVEL ?= 36
# per OS dictionary-like
UNAME_S := $(shell uname -s)
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index 61c33d6f16..f9aea86996 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -119,7 +119,7 @@ named ``tools``, and you will need to run extra commands to install
the SDK packages needed.
For Android NDK, note that modern releases will only work on a 64-bit
-operating system. **The minimal, and recommended, NDK version to use is r25b:**
+operating system. **The minimal, and recommended, NDK version to use is r28c:**
- `Go to ndk downloads page <https://developer.android.com/ndk/downloads/>`_
- Windows users should create a virtual machine with an GNU Linux os
@@ -154,7 +154,7 @@ variables necessary for building on android::
# Adjust the paths!
export ANDROIDSDK="$HOME/Documents/android-sdk-27"
export ANDROIDNDK="$HOME/Documents/android-ndk-r23b"
- export ANDROIDAPI="27" # Target API version of your application
+ export ANDROIDAPI="36" # Target API version of your application
export NDKAPI="21" # Minimum supported API version of your application
export ANDROIDNDKVER="r10e" # Version of the NDK you installed
diff --git a/doc/source/testing_pull_requests.rst b/doc/source/testing_pull_requests.rst
index f77748e336..8708b4c003 100644
--- a/doc/source/testing_pull_requests.rst
+++ b/doc/source/testing_pull_requests.rst
@@ -118,7 +118,7 @@ Using python-for-android commands directly from the pull request files
--requirements=sdl2,pyjnius,kivy,python3,pycryptodome \
--ndk-dir=/media/DEVEL/Android/android-ndk-r20 \
--sdk-dir=/media/DEVEL/Android/android-sdk-linux \
- --android-api=27 \
+ --android-api=36 \
--arch=arm64-v8a \
--permission=VIBRATE \
--debug
@@ -175,7 +175,7 @@ Installing python-for-android using the github's branch of the pull request
python3 setup.py apk \
--ndk-dir=/media/DEVEL/Android/android-ndk-r20 \
--sdk-dir=/media/DEVEL/Android/android-sdk-linux \
- --android-api=27 \
+ --android-api=36 \
--arch=arm64-v8a \
--debug
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 44469aef2c..0ebe005d14 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -155,6 +155,11 @@ class Recipe(metaclass=RecipeMeta):
starting from NDK r18 the `gnustl_shared` lib has been deprecated.
'''
+ min_ndk_api_support = 20
+ '''
+ Minimum ndk api recipe will support.
+ '''
+
def get_stl_library(self, arch):
return join(
arch.ndk_lib_dir,
@@ -375,6 +380,9 @@ def get_recipe_dir(self):
# Public Recipe API to be subclassed if needed
def download_if_necessary(self):
+ if self.ctx.ndk_api < self.min_ndk_api_support:
+ error(f"In order to build '{self.name}', you must set minimum ndk api (minapi) to `{self.min_ndk_api_support}`.\n")
+ exit(1)
info_main('Downloading {}'.format(self.name))
user_dir = environ.get('P4A_{}_DIR'.format(self.name.lower()))
if user_dir is not None:
diff --git a/pythonforandroid/recipes/numpy/__init__.py b/pythonforandroid/recipes/numpy/__init__.py
index fb34c0c9f7..f85e44d4f5 100644
--- a/pythonforandroid/recipes/numpy/__init__.py
+++ b/pythonforandroid/recipes/numpy/__init__.py
@@ -1,5 +1,4 @@
from pythonforandroid.recipe import Recipe, MesonRecipe
-from pythonforandroid.logger import error
from os.path import join
import shutil
@@ -7,11 +6,12 @@
class NumpyRecipe(MesonRecipe):
- version = 'v1.26.5'
+ version = 'v2.3.0'
url = 'git+https://github.com/numpy/numpy'
- hostpython_prerequisites = ["Cython>=3.0.6"] # meson does not detects venv's cython
+ hostpython_prerequisites = ["Cython>=3.0.6", "numpy"] # meson does not detects venv's cython
extra_build_args = ['-Csetup-args=-Dblas=none', '-Csetup-args=-Dlapack=none']
need_stl_shared = True
+ min_ndk_api_support = 24
def get_recipe_meson_options(self, arch):
options = super().get_recipe_meson_options(arch)
@@ -36,13 +36,6 @@ def get_recipe_env(self, arch, **kwargs):
"python3", self.ctx).get_build_dir(arch.arch), "android-build", "python")
return env
- def download_if_necessary(self):
- # NumPy requires complex math functions which were added in api 24
- if self.ctx.ndk_api < 24:
- error(NUMPY_NDK_MESSAGE)
- exit(1)
- super().download_if_necessary()
-
def build_arch(self, arch):
super().build_arch(arch)
self.restore_hostpython_prerequisites(["cython"])
diff --git a/pythonforandroid/recipes/pandas/__init__.py b/pythonforandroid/recipes/pandas/__init__.py
index 3f56adef6c..6986e9efbe 100644
--- a/pythonforandroid/recipes/pandas/__init__.py
+++ b/pythonforandroid/recipes/pandas/__init__.py
@@ -3,10 +3,10 @@
class PandasRecipe(MesonRecipe):
- version = 'v2.2.1'
- url = 'git+https://github.com/pandas-dev/pandas' # noqa
+ version = 'v2.3.0'
+ url = 'git+https://github.com/pandas-dev/pandas'
depends = ['numpy', 'libbz2', 'liblzma']
- hostpython_prerequisites = ["Cython~=3.0.5"] # meson does not detects venv's cython
+ hostpython_prerequisites = ["Cython<4.0.0a0", "versioneer", "numpy"] # meson does not detects venv's cython
patches = ['fix_numpy_includes.patch']
python_depends = ['python-dateutil', 'pytz']
need_stl_shared = True
@@ -17,7 +17,7 @@ def get_recipe_env(self, arch, **kwargs):
# because we need some includes generated at numpy's compile time
env['NUMPY_INCLUDES'] = join(
- self.ctx.get_python_install_dir(arch.arch), "numpy/core/include",
+ self.ctx.get_python_install_dir(arch.arch), "numpy/_core/include",
)
env["PYTHON_INCLUDE_DIR"] = self.ctx.python_recipe.include_root(arch)
diff --git a/pythonforandroid/recipes/python3/__init__.py b/pythonforandroid/recipes/python3/__init__.py
index 2334db6add..7cd3928d76 100644
--- a/pythonforandroid/recipes/python3/__init__.py
+++ b/pythonforandroid/recipes/python3/__init__.py
@@ -146,6 +146,14 @@ class Python3Recipe(TargetPythonRecipe):
'''The directories from site packages dir that we don't want to be included
in our python bundle.'''
+ site_packages_excluded_dir_exceptions = [
+ # 'numpy' is excluded here because importing with `import numpy as np`
+ # can fail if the `tests` directory inside the numpy package is excluded.
+ 'numpy',
+ ]
+ '''Directories from `site_packages_dir_blacklist` will not be excluded
+ if the full path contains any of these exceptions.'''
+
site_packages_filen_blacklist = [
'*.py'
]
@@ -419,7 +427,8 @@ def create_python_bundle(self, dirn, arch):
with current_directory(self.ctx.get_python_install_dir(arch.arch)):
filens = list(walk_valid_filens(
'.', self.site_packages_dir_blacklist,
- self.site_packages_filen_blacklist))
+ self.site_packages_filen_blacklist,
+ excluded_dir_exceptions=self.site_packages_excluded_dir_exceptions))
info("Copy {} files into the site-packages".format(len(filens)))
for filen in filens:
info(" - copy {}".format(filen))
diff --git a/pythonforandroid/recipes/sdl2/__init__.py b/pythonforandroid/recipes/sdl2/__init__.py
index cd0185c717..d1a5fdc8b3 100644
--- a/pythonforandroid/recipes/sdl2/__init__.py
+++ b/pythonforandroid/recipes/sdl2/__init__.py
@@ -6,9 +6,9 @@
class LibSDL2Recipe(BootstrapNDKRecipe):
- version = "2.28.5"
+ version = "2.30.11"
url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL2-{version}.tar.gz"
- md5sum = 'a344eb827a03045c9b399e99af4af13d'
+ md5sum = 'bea190b480f6df249db29eb3bacfe41e'
conflicts = ['sdl3']
diff --git a/pythonforandroid/recommendations.py b/pythonforandroid/recommendations.py
index 269a57fcf8..397dcd9ca4 100644
--- a/pythonforandroid/recommendations.py
+++ b/pythonforandroid/recommendations.py
@@ -13,7 +13,7 @@
MAX_NDK_VERSION = 25
# DO NOT CHANGE LINE FORMAT: buildozer parses the existence of a RECOMMENDED_NDK_VERSION
-RECOMMENDED_NDK_VERSION = "25b"
+RECOMMENDED_NDK_VERSION = "28c"
NDK_DOWNLOAD_URL = "https://developer.android.com/ndk/downloads/"
diff --git a/pythonforandroid/util.py b/pythonforandroid/util.py
index 13f38e6c0f..3cdcaaae76 100644
--- a/pythonforandroid/util.py
+++ b/pythonforandroid/util.py
@@ -48,7 +48,7 @@ def temp_directory():
temp_dir, Err_Fore.RESET)))
-def walk_valid_filens(base_dir, invalid_dir_names, invalid_file_patterns):
+def walk_valid_filens(base_dir, invalid_dir_names, invalid_file_patterns, excluded_dir_exceptions=None):
"""Recursively walks all the files and directories in ``dirn``,
ignoring directories that match any pattern in ``invalid_dirns``
and files that patch any pattern in ``invalid_filens``.
@@ -60,15 +60,22 @@ def walk_valid_filens(base_dir, invalid_dir_names, invalid_file_patterns):
File and directory paths are evaluated as full paths relative to ``dirn``.
+ If ``excluded_dir_exceptions`` is given, any directory path that contains
+ any of those strings will *not* exclude subdirectories matching
+ ``invalid_dir_names``.
"""
+ excluded_dir_exceptions = [] if excluded_dir_exceptions is None else excluded_dir_exceptions
+
for dirn, subdirs, filens in walk(base_dir):
+ allow_invalid_dirs = any(ex in dirn for ex in excluded_dir_exceptions)
# Remove invalid subdirs so that they will not be walked
- for i in reversed(range(len(subdirs))):
- subdir = subdirs[i]
- if subdir in invalid_dir_names:
- subdirs.pop(i)
+ if not allow_invalid_dirs:
+ for i in reversed(range(len(subdirs))):
+ subdir = subdirs[i]
+ if subdir in invalid_dir_names:
+ subdirs.pop(i)
for filen in filens:
for pattern in invalid_file_patterns:
diff --git a/testapps/on_device_unit_tests/setup.py b/testapps/on_device_unit_tests/setup.py
index 4d2019390c..ef07e29320 100644
--- a/testapps/on_device_unit_tests/setup.py
+++ b/testapps/on_device_unit_tests/setup.py
@@ -42,7 +42,7 @@
'requirements':
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
- 'android-api': 35,
+ 'android-api': 36,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
@@ -56,7 +56,7 @@
'requirements':
'sqlite3,libffi,openssl,pyjnius,kivy,python3,requests,urllib3,'
'chardet,idna',
- 'android-api': 35,
+ 'android-api': 36,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'armeabi-v7a',
@@ -68,7 +68,7 @@
'aar':
{
'requirements' : 'python3',
- 'android-api': 35,
+ 'android-api': 36,
'ndk-api': 24,
'dist-name': 'bdist_unit_tests_app',
'arch': 'arm64-v8a',
diff --git a/testapps/setup_testapp_python3_sqlite_openssl.py b/testapps/setup_testapp_python3_sqlite_openssl.py
index c6360679f1..49c6a83d27 100644
--- a/testapps/setup_testapp_python3_sqlite_openssl.py
+++ b/testapps/setup_testapp_python3_sqlite_openssl.py
@@ -1,7 +1,7 @@
from setuptools import setup, find_packages
options = {'apk': {'requirements': 'requests,peewee,sdl2,pyjnius,kivy,python3',
- 'android-api': 27,
+ 'android-api': 36,
'ndk-api': 21,
'bootstrap': 'sdl2',
'dist-name': 'bdisttest_python3_sqlite_openssl_googlendk',
diff --git a/testapps/setup_vispy.py b/testapps/setup_vispy.py
index f59d057b0b..81f4b5d7db 100644
--- a/testapps/setup_vispy.py
+++ b/testapps/setup_vispy.py
@@ -3,7 +3,7 @@
options = {'apk': {'debug': None,
'requirements': 'python3,vispy',
'blacklist-requirements': 'openssl,sqlite3',
- 'android-api': 27,
+ 'android-api': 33,
'ndk-api': 21,
'bootstrap': 'empty',
'ndk-dir': '/home/asandy/android/android-ndk-r17c',
diff --git a/testapps/testlauncherreboot_setup/sdl2.py b/testapps/testlauncherreboot_setup/sdl2.py
index 8ea0d43c4c..a53ef66a78 100644
--- a/testapps/testlauncherreboot_setup/sdl2.py
+++ b/testapps/testlauncherreboot_setup/sdl2.py
@@ -54,7 +54,7 @@
# 'cymunk,lxml,pil,openssl,pyopenssl,'
# 'twisted,audiostream,ffmpeg,numpy'
- 'android-api': 27,
+ 'android-api': 36,
'ndk-api': 21,
'dist-name': 'bdisttest_python3launcher_sdl2_googlendk',
'name': 'TestLauncherPy3-sdl2',
diff --git a/tests/recipes/test_pandas.py b/tests/recipes/test_pandas.py
index b8366863fe..9a028d49b2 100644
--- a/tests/recipes/test_pandas.py
+++ b/tests/recipes/test_pandas.py
@@ -35,7 +35,7 @@ def test_get_recipe_env(
self.ctx.recipe_build_order
)
numpy_includes = join(
- self.ctx.get_python_install_dir(self.arch.arch), "numpy/core/include",
+ self.ctx.get_python_install_dir(self.arch.arch), "numpy/_core/include",
)
env = self.recipe.get_recipe_env(self.arch)
self.assertIn(numpy_includes, env["NUMPY_INCLUDES"])
diff --git a/tests/test_recipe.py b/tests/test_recipe.py
index b02a874e84..e2e0e9d826 100644
--- a/tests/test_recipe.py
+++ b/tests/test_recipe.py
@@ -93,6 +93,8 @@ def test_download_if_necessary(self):
"""
# download should happen as the environment variable is not set
recipe = DummyRecipe()
+ recipe.ctx = Context()
+ recipe.ctx._ndk_api = 36
with mock.patch.object(Recipe, 'download') as m_download:
recipe.download_if_necessary()
assert m_download.call_args_list == [mock.call()]
From 331e4e4336e50550f0bf6a055c3b1d77cf801bb8 Mon Sep 17 00:00:00 2001
From: Mirko Galimberti <me@mirkogalimberti.com>
Date: Thu, 31 Jul 2025 22:01:31 +0200
Subject: [PATCH 124/158] Require `c++` stl for Kivy when `sdl3` is used (Kivy
3.0.0) (#3181)
---
pythonforandroid/recipes/kivy/__init__.py | 7 +++++++
1 file changed, 7 insertions(+)
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index f80024155f..2311da6cc5 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -34,6 +34,13 @@ class KivyRecipe(PyProjectRecipe):
# WARNING: Remove this patch when a new Kivy version is released.
patches = [("sdl-gl-swapwindow-nogil.patch", is_kivy_affected_by_deadlock_issue), "use_cython.patch"]
+ @property
+ def need_stl_shared(self):
+ if "sdl3" in self.ctx.recipe_build_order:
+ return True
+ else:
+ return False
+
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
From 75d2c9aac2e69e08d3d374d3d9369a684dfbef61 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Fri, 1 Aug 2025 03:55:52 -0300
Subject: [PATCH 125/158] Bump SDL3 recipe version to the latest stable release
(`3.2.18`) (#3182)
---
pythonforandroid/recipes/sdl3/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl3/__init__.py b/pythonforandroid/recipes/sdl3/__init__.py
index 6b9c8ee7c6..ea2655df70 100644
--- a/pythonforandroid/recipes/sdl3/__init__.py
+++ b/pythonforandroid/recipes/sdl3/__init__.py
@@ -6,7 +6,7 @@
class LibSDL3Recipe(BootstrapNDKRecipe):
- version = "3.2.10"
+ version = "3.2.18"
url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL3-{version}.tar.gz"
md5sum = "70cda886bcf5a4fdac550db796d2efa2"
From 5330267b5cdf8609c921f1ae4d36143d23450d1b Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Fri, 1 Aug 2025 12:43:23 -0300
Subject: [PATCH 126/158] Fix `md5sum` for SDL3 recipe version `3.2.18` (#3183)
* fix md5sum
* use lowercase
---
pythonforandroid/recipes/sdl3/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/sdl3/__init__.py b/pythonforandroid/recipes/sdl3/__init__.py
index ea2655df70..139509c3e8 100644
--- a/pythonforandroid/recipes/sdl3/__init__.py
+++ b/pythonforandroid/recipes/sdl3/__init__.py
@@ -8,7 +8,7 @@
class LibSDL3Recipe(BootstrapNDKRecipe):
version = "3.2.18"
url = "https://github.com/libsdl-org/SDL/releases/download/release-{version}/SDL3-{version}.tar.gz"
- md5sum = "70cda886bcf5a4fdac550db796d2efa2"
+ md5sum = "c7808ef624b74e2ac69cf531e78e0c6e"
conflicts = ["sdl2"]
From 51287006e02dfb8e6a010eabcc3045d3f09b42cc Mon Sep 17 00:00:00 2001
From: Kartavya Shukla <87070473+Novfensec@users.noreply.github.com>
Date: Sun, 14 Sep 2025 17:09:59 +0530
Subject: [PATCH 127/158] Update
---
pythonforandroid/recipes/sympy/__init__.py | 16 +-
.../recipes/sympy/fix_android_detection.patch | 47 ----
.../recipes/sympy/fix_pretty_print.patch | 223 ------------------
.../recipes/sympy/fix_timeutils.patch | 13 -
4 files changed, 7 insertions(+), 292 deletions(-)
delete mode 100644 pythonforandroid/recipes/sympy/fix_android_detection.patch
delete mode 100644 pythonforandroid/recipes/sympy/fix_pretty_print.patch
delete mode 100644 pythonforandroid/recipes/sympy/fix_timeutils.patch
diff --git a/pythonforandroid/recipes/sympy/__init__.py b/pythonforandroid/recipes/sympy/__init__.py
index 8684a95e06..5420bf0379 100644
--- a/pythonforandroid/recipes/sympy/__init__.py
+++ b/pythonforandroid/recipes/sympy/__init__.py
@@ -1,16 +1,14 @@
+from pythonforandroid.recipe import PyProjectRecipe
-from pythonforandroid.recipe import PythonRecipe
+class SympyRecipe(PyProjectRecipe):
-class SympyRecipe(PythonRecipe):
- version = '1.1.1'
- url = 'https://github.com/sympy/sympy/releases/download/sympy-{version}/sympy-{version}.tar.gz'
-
+ version = '1.14.0'
+ url = 'https://github.com/sympy/sympy/archive/refs/tags/{version}.zip'
depends = ['mpmath']
-
- call_hostpython_via_targetpython = True
-
- patches = ['fix_timeutils.patch', 'fix_pretty_print.patch']
+ hostpython_prerequisites = ['setuptools', 'build']
+ call_hostpython_via_targetpython = False
+ install_in_hostpython = True
recipe = SympyRecipe()
diff --git a/pythonforandroid/recipes/sympy/fix_android_detection.patch b/pythonforandroid/recipes/sympy/fix_android_detection.patch
deleted file mode 100644
index 964c3db66f..0000000000
--- a/pythonforandroid/recipes/sympy/fix_android_detection.patch
+++ /dev/null
@@ -1,47 +0,0 @@
-diff --git a/pip/download.py b/pip/download.py
-index 54d3131..1aab70f 100644
---- a/pip/download.py
-+++ b/pip/download.py
-@@ -89,23 +89,25 @@ def user_agent():
- # Complete Guess
- data["implementation"]["version"] = platform.python_version()
-
-- if sys.platform.startswith("linux"):
-- from pip._vendor import distro
-- distro_infos = dict(filter(
-- lambda x: x[1],
-- zip(["name", "version", "id"], distro.linux_distribution()),
-- ))
-- libc = dict(filter(
-- lambda x: x[1],
-- zip(["lib", "version"], libc_ver()),
-- ))
-- if libc:
-- distro_infos["libc"] = libc
-- if distro_infos:
-- data["distro"] = distro_infos
--
-- if sys.platform.startswith("darwin") and platform.mac_ver()[0]:
-- data["distro"] = {"name": "macOS", "version": platform.mac_ver()[0]}
-+ # if sys.platform.startswith("linux"):
-+ # from pip._vendor import distro
-+ # distro_infos = dict(filter(
-+ # lambda x: x[1],
-+ # zip(["name", "version", "id"], distro.linux_distribution()),
-+ # ))
-+ # libc = dict(filter(
-+ # lambda x: x[1],
-+ # zip(["lib", "version"], libc_ver()),
-+ # ))
-+ # if libc:
-+ # distro_infos["libc"] = libc
-+ # if distro_infos:
-+ # data["distro"] = distro_infos
-+
-+ # if sys.platform.startswith("darwin") and platform.mac_ver()[0]:
-+ # data["distro"] = {"name": "macOS", "version": platform.mac_ver()[0]}
-+
-+ data['distro'] = {'name': 'Android'}
-
- if platform.system():
- data.setdefault("system", {})["name"] = platform.system()
diff --git a/pythonforandroid/recipes/sympy/fix_pretty_print.patch b/pythonforandroid/recipes/sympy/fix_pretty_print.patch
deleted file mode 100644
index f94cb2245c..0000000000
--- a/pythonforandroid/recipes/sympy/fix_pretty_print.patch
+++ /dev/null
@@ -1,223 +0,0 @@
-diff --git a/sympy/printing/pretty/pretty.py b/sympy/printing/pretty/pretty.py
-index 604e97c..ddd3eb2 100644
---- a/sympy/printing/pretty/pretty.py
-+++ b/sympy/printing/pretty/pretty.py
-@@ -166,14 +166,14 @@ class PrettyPrinter(Printer):
- arg = e.args[0]
- pform = self._print(arg)
- if isinstance(arg, Equivalent):
-- return self._print_Equivalent(arg, altchar=u"\N{NOT IDENTICAL TO}")
-+ return self._print_Equivalent(arg, altchar=u"NOT IDENTICAL TO")
- if isinstance(arg, Implies):
-- return self._print_Implies(arg, altchar=u"\N{RIGHTWARDS ARROW WITH STROKE}")
-+ return self._print_Implies(arg, altchar=u"RIGHTWARDS ARROW WITH STROKE")
-
- if arg.is_Boolean and not arg.is_Not:
- pform = prettyForm(*pform.parens())
-
-- return prettyForm(*pform.left(u"\N{NOT SIGN}"))
-+ return prettyForm(*pform.left(u"NOT SIGN"))
- else:
- return self._print_Function(e)
-
-@@ -200,43 +200,43 @@ class PrettyPrinter(Printer):
-
- def _print_And(self, e):
- if self._use_unicode:
-- return self.__print_Boolean(e, u"\N{LOGICAL AND}")
-+ return self.__print_Boolean(e, u"LOGICAL AND")
- else:
- return self._print_Function(e, sort=True)
-
- def _print_Or(self, e):
- if self._use_unicode:
-- return self.__print_Boolean(e, u"\N{LOGICAL OR}")
-+ return self.__print_Boolean(e, u"LOGICAL OR")
- else:
- return self._print_Function(e, sort=True)
-
- def _print_Xor(self, e):
- if self._use_unicode:
-- return self.__print_Boolean(e, u"\N{XOR}")
-+ return self.__print_Boolean(e, u"XOR")
- else:
- return self._print_Function(e, sort=True)
-
- def _print_Nand(self, e):
- if self._use_unicode:
-- return self.__print_Boolean(e, u"\N{NAND}")
-+ return self.__print_Boolean(e, u"NAND")
- else:
- return self._print_Function(e, sort=True)
-
- def _print_Nor(self, e):
- if self._use_unicode:
-- return self.__print_Boolean(e, u"\N{NOR}")
-+ return self.__print_Boolean(e, u"NOR")
- else:
- return self._print_Function(e, sort=True)
-
- def _print_Implies(self, e, altchar=None):
- if self._use_unicode:
-- return self.__print_Boolean(e, altchar or u"\N{RIGHTWARDS ARROW}", sort=False)
-+ return self.__print_Boolean(e, altchar or u"RIGHTWARDS ARROW", sort=False)
- else:
- return self._print_Function(e)
-
- def _print_Equivalent(self, e, altchar=None):
- if self._use_unicode:
-- return self.__print_Boolean(e, altchar or u"\N{IDENTICAL TO}")
-+ return self.__print_Boolean(e, altchar or u"IDENTICAL TO")
- else:
- return self._print_Function(e, sort=True)
-
-@@ -425,7 +425,7 @@ class PrettyPrinter(Printer):
- if self._use_unicode:
- # use unicode corners
- horizontal_chr = xobj('-', 1)
-- corner_chr = u'\N{BOX DRAWINGS LIGHT DOWN AND HORIZONTAL}'
-+ corner_chr = u'BOX DRAWINGS LIGHT DOWN AND HORIZONTAL'
-
- func_height = pretty_func.height()
-
-@@ -580,7 +580,7 @@ class PrettyPrinter(Printer):
-
- LimArg = self._print(z)
- if self._use_unicode:
-- LimArg = prettyForm(*LimArg.right(u'\N{BOX DRAWINGS LIGHT HORIZONTAL}\N{RIGHTWARDS ARROW}'))
-+ LimArg = prettyForm(*LimArg.right(u'BOX DRAWINGS LIGHT HORIZONTALRIGHTWARDS ARROW'))
- else:
- LimArg = prettyForm(*LimArg.right('->'))
- LimArg = prettyForm(*LimArg.right(self._print(z0)))
-@@ -589,7 +589,7 @@ class PrettyPrinter(Printer):
- dir = ""
- else:
- if self._use_unicode:
-- dir = u'\N{SUPERSCRIPT PLUS SIGN}' if str(dir) == "+" else u'\N{SUPERSCRIPT MINUS}'
-+ dir = u'SUPERSCRIPT PLUS SIGN' if str(dir) == "+" else u'SUPERSCRIPT MINUS'
-
- LimArg = prettyForm(*LimArg.right(self._print(dir)))
-
-@@ -740,7 +740,7 @@ class PrettyPrinter(Printer):
- def _print_Adjoint(self, expr):
- pform = self._print(expr.arg)
- if self._use_unicode:
-- dag = prettyForm(u'\N{DAGGER}')
-+ dag = prettyForm(u'DAGGER')
- else:
- dag = prettyForm('+')
- from sympy.matrices import MatrixSymbol
-@@ -850,8 +850,8 @@ class PrettyPrinter(Printer):
- if '\n' in partstr:
- tempstr = partstr
- tempstr = tempstr.replace(vectstrs[i], '')
-- tempstr = tempstr.replace(u'\N{RIGHT PARENTHESIS UPPER HOOK}',
-- u'\N{RIGHT PARENTHESIS UPPER HOOK}'
-+ tempstr = tempstr.replace(u'RIGHT PARENTHESIS UPPER HOOK',
-+ u'RIGHT PARENTHESIS UPPER HOOK'
- + ' ' + vectstrs[i])
- o1[i] = tempstr
- o1 = [x.split('\n') for x in o1]
-@@ -1153,7 +1153,7 @@ class PrettyPrinter(Printer):
- def _print_Lambda(self, e):
- vars, expr = e.args
- if self._use_unicode:
-- arrow = u" \N{RIGHTWARDS ARROW FROM BAR} "
-+ arrow = u" RIGHTWARDS ARROW FROM BAR "
- else:
- arrow = " -> "
- if len(vars) == 1:
-@@ -1173,7 +1173,7 @@ class PrettyPrinter(Printer):
- elif len(expr.variables):
- pform = prettyForm(*pform.right(self._print(expr.variables[0])))
- if self._use_unicode:
-- pform = prettyForm(*pform.right(u" \N{RIGHTWARDS ARROW} "))
-+ pform = prettyForm(*pform.right(u" RIGHTWARDS ARROW "))
- else:
- pform = prettyForm(*pform.right(" -> "))
- if len(expr.point) > 1:
-@@ -1462,7 +1462,7 @@ class PrettyPrinter(Printer):
- and expt is S.Half and bpretty.height() == 1
- and (bpretty.width() == 1
- or (base.is_Integer and base.is_nonnegative))):
-- return prettyForm(*bpretty.left(u'\N{SQUARE ROOT}'))
-+ return prettyForm(*bpretty.left(u'SQUARE ROOT'))
-
- # Construct root sign, start with the \/ shape
- _zZ = xobj('/', 1)
-@@ -1558,7 +1558,7 @@ class PrettyPrinter(Printer):
- from sympy import Pow
- return self._print(Pow(p.sets[0], len(p.sets), evaluate=False))
- else:
-- prod_char = u"\N{MULTIPLICATION SIGN}" if self._use_unicode else 'x'
-+ prod_char = u"MULTIPLICATION SIGN" if self._use_unicode else 'x'
- return self._print_seq(p.sets, None, None, ' %s ' % prod_char,
- parenthesize=lambda set: set.is_Union or
- set.is_Intersection or set.is_ProductSet)
-@@ -1570,7 +1570,7 @@ class PrettyPrinter(Printer):
- def _print_Range(self, s):
-
- if self._use_unicode:
-- dots = u"\N{HORIZONTAL ELLIPSIS}"
-+ dots = u"HORIZONTAL ELLIPSIS"
- else:
- dots = '...'
-
-@@ -1641,7 +1641,7 @@ class PrettyPrinter(Printer):
-
- def _print_ImageSet(self, ts):
- if self._use_unicode:
-- inn = u"\N{SMALL ELEMENT OF}"
-+ inn = u"SMALL ELEMENT OF"
- else:
- inn = 'in'
- variables = self._print_seq(ts.lamda.variables)
-@@ -1653,10 +1653,10 @@ class PrettyPrinter(Printer):
-
- def _print_ConditionSet(self, ts):
- if self._use_unicode:
-- inn = u"\N{SMALL ELEMENT OF}"
-+ inn = u"SMALL ELEMENT OF"
- # using _and because and is a keyword and it is bad practice to
- # overwrite them
-- _and = u"\N{LOGICAL AND}"
-+ _and = u"LOGICAL AND"
- else:
- inn = 'in'
- _and = 'and'
-@@ -1677,7 +1677,7 @@ class PrettyPrinter(Printer):
-
- def _print_ComplexRegion(self, ts):
- if self._use_unicode:
-- inn = u"\N{SMALL ELEMENT OF}"
-+ inn = u"SMALL ELEMENT OF"
- else:
- inn = 'in'
- variables = self._print_seq(ts.variables)
-@@ -1690,7 +1690,7 @@ class PrettyPrinter(Printer):
- def _print_Contains(self, e):
- var, set = e.args
- if self._use_unicode:
-- el = u" \N{ELEMENT OF} "
-+ el = u" ELEMENT OF "
- return prettyForm(*stringPict.next(self._print(var),
- el, self._print(set)), binding=8)
- else:
-@@ -1698,7 +1698,7 @@ class PrettyPrinter(Printer):
-
- def _print_FourierSeries(self, s):
- if self._use_unicode:
-- dots = u"\N{HORIZONTAL ELLIPSIS}"
-+ dots = u"HORIZONTAL ELLIPSIS"
- else:
- dots = '...'
- return self._print_Add(s.truncate()) + self._print(dots)
-@@ -1708,7 +1708,7 @@ class PrettyPrinter(Printer):
-
- def _print_SeqFormula(self, s):
- if self._use_unicode:
-- dots = u"\N{HORIZONTAL ELLIPSIS}"
-+ dots = u"HORIZONTAL ELLIPSIS"
- else:
- dots = '...'
-
diff --git a/pythonforandroid/recipes/sympy/fix_timeutils.patch b/pythonforandroid/recipes/sympy/fix_timeutils.patch
deleted file mode 100644
index c8424eaa2c..0000000000
--- a/pythonforandroid/recipes/sympy/fix_timeutils.patch
+++ /dev/null
@@ -1,13 +0,0 @@
-diff --git a/sympy/utilities/timeutils.py b/sympy/utilities/timeutils.py
-index 3770d85..c53594e 100644
---- a/sympy/utilities/timeutils.py
-+++ b/sympy/utilities/timeutils.py
-@@ -8,7 +8,7 @@ import math
- from sympy.core.compatibility import range
-
- _scales = [1e0, 1e3, 1e6, 1e9]
--_units = [u's', u'ms', u'\N{GREEK SMALL LETTER MU}s', u'ns']
-+_units = [u's', u'ms', u'mus', u'ns']
-
-
- def timed(func, setup="pass", limit=None):
From ed3295f31d946becfb3ba8768a9810a3d0a84a77 Mon Sep 17 00:00:00 2001
From: Kartavya Shukla <87070473+Novfensec@users.noreply.github.com>
Date: Sun, 14 Sep 2025 17:25:11 +0530
Subject: [PATCH 128/158] Delete pythonforandroid/recipes/sympy directory
---
pythonforandroid/recipes/sympy/__init__.py | 14 --------------
1 file changed, 14 deletions(-)
delete mode 100644 pythonforandroid/recipes/sympy/__init__.py
diff --git a/pythonforandroid/recipes/sympy/__init__.py b/pythonforandroid/recipes/sympy/__init__.py
deleted file mode 100644
index 5420bf0379..0000000000
--- a/pythonforandroid/recipes/sympy/__init__.py
+++ /dev/null
@@ -1,14 +0,0 @@
-from pythonforandroid.recipe import PyProjectRecipe
-
-
-class SympyRecipe(PyProjectRecipe):
-
- version = '1.14.0'
- url = 'https://github.com/sympy/sympy/archive/refs/tags/{version}.zip'
- depends = ['mpmath']
- hostpython_prerequisites = ['setuptools', 'build']
- call_hostpython_via_targetpython = False
- install_in_hostpython = True
-
-
-recipe = SympyRecipe()
From 0f13b85c60fd9a9b92276ab9645fd2e984d2d410 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Sep 2025 17:40:57 +0530
Subject: [PATCH 129/158] `flask`: update to `3.1.1`
---
pythonforandroid/recipes/flask/__init__.py | 15 ++++-----------
1 file changed, 4 insertions(+), 11 deletions(-)
diff --git a/pythonforandroid/recipes/flask/__init__.py b/pythonforandroid/recipes/flask/__init__.py
index b2729420da..4b05e5ff84 100644
--- a/pythonforandroid/recipes/flask/__init__.py
+++ b/pythonforandroid/recipes/flask/__init__.py
@@ -1,17 +1,10 @@
+from pythonforandroid.recipe import PyProjectRecipe
-from pythonforandroid.recipe import PythonRecipe
-
-class FlaskRecipe(PythonRecipe):
- version = '2.0.3'
+class FlaskRecipe(PyProjectRecipe):
+ version = '3.1.1'
url = 'https://github.com/pallets/flask/archive/{version}.zip'
-
- depends = ['setuptools']
-
- python_depends = ['jinja2', 'werkzeug', 'markupsafe', 'itsdangerous', 'click']
-
- call_hostpython_via_targetpython = False
- install_in_hostpython = False
+ python_depends = ['jinja2', 'werkzeug', 'markupsafe', 'itsdangerous', 'click', 'blinker']
recipe = FlaskRecipe()
From 684c68149b3acc3197ea95627c1403e782d32cc0 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Sep 2025 18:01:05 +0530
Subject: [PATCH 130/158] `kivy`: depend on android
---
pythonforandroid/recipes/kivy/__init__.py | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index 2311da6cc5..d9ca543f36 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -25,9 +25,9 @@ class KivyRecipe(PyProjectRecipe):
url = 'https://github.com/kivy/kivy/archive/{version}.zip'
name = 'kivy'
- depends = [('sdl2', 'sdl3'), 'pyjnius', 'setuptools']
+ depends = [('sdl2', 'sdl3'), 'pyjnius', 'setuptools', 'android']
python_depends = ['certifi', 'chardet', 'idna', 'requests', 'urllib3', 'filetype']
- hostpython_prerequisites = []
+ hostpython_prerequisites = ["cython>=0.29.1,<=3.0.12"]
# sdl-gl-swapwindow-nogil.patch is needed to avoid a deadlock.
# See: https://github.com/kivy/kivy/pull/8025
From d045912b416519f114636d15ec9f31d21cb7031c Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Sep 2025 17:46:32 +0530
Subject: [PATCH 131/158] `android`: shift to PPR
---
pythonforandroid/recipes/android/__init__.py | 13 +++++---
pythonforandroid/recipes/android/src/setup.py | 30 ++++++++++++-------
2 files changed, 29 insertions(+), 14 deletions(-)
diff --git a/pythonforandroid/recipes/android/__init__.py b/pythonforandroid/recipes/android/__init__.py
index 5174a69bfa..c6c15ec04a 100644
--- a/pythonforandroid/recipes/android/__init__.py
+++ b/pythonforandroid/recipes/android/__init__.py
@@ -1,11 +1,11 @@
-from pythonforandroid.recipe import CythonRecipe, IncludedFilesBehaviour
+from pythonforandroid.recipe import PyProjectRecipe, IncludedFilesBehaviour
from pythonforandroid.util import current_directory
from pythonforandroid import logger
from os.path import join
-class AndroidRecipe(IncludedFilesBehaviour, CythonRecipe):
+class AndroidRecipe(IncludedFilesBehaviour, PyProjectRecipe):
# name = 'android'
version = None
url = None
@@ -13,11 +13,12 @@ class AndroidRecipe(IncludedFilesBehaviour, CythonRecipe):
src_filename = 'src'
depends = [('sdl3', 'sdl2', 'genericndkbuild'), 'pyjnius']
+ hostpython_prerequisites = ["Cython>=0.29,<3.1"]
config_env = {}
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
env.update(self.config_env)
return env
@@ -49,6 +50,10 @@ def prebuild_arch(self, arch):
'IS_SDL2': int(bootstrap_name == "sdl2"),
'IS_SDL3': int(bootstrap_name == "sdl3"),
'PY2': 0,
+ 'ANDROID_LIBS_DIR': "{}:{}".format(
+ self.ctx.get_libs_dir(arch.arch),
+ join(self.ctx.bootstrap.build_dir, 'obj', 'local', arch.arch)
+ ),
'JAVA_NAMESPACE': java_ns,
'JNI_NAMESPACE': jni_ns,
'ACTIVITY_CLASS_NAME': self.ctx.activity_class_name,
diff --git a/pythonforandroid/recipes/android/src/setup.py b/pythonforandroid/recipes/android/src/setup.py
index 0f5ceb1fd3..8bf4512e05 100755
--- a/pythonforandroid/recipes/android/src/setup.py
+++ b/pythonforandroid/recipes/android/src/setup.py
@@ -1,25 +1,35 @@
from distutils.core import setup, Extension
+from Cython.Build import cythonize
import os
-library_dirs = ['libs/' + os.environ['ARCH']]
+library_dirs = os.environ['ANDROID_LIBS_DIR'].split(":")
lib_dict = {
'sdl2': ['SDL2', 'SDL2_image', 'SDL2_mixer', 'SDL2_ttf'],
'sdl3': ['SDL3', 'SDL3_image', 'SDL3_mixer', 'SDL3_ttf'],
}
sdl_libs = lib_dict.get(os.environ['BOOTSTRAP'], ['main'])
-modules = [Extension('android._android',
- ['android/_android.c', 'android/_android_jni.c'],
- libraries=sdl_libs + ['log'],
- library_dirs=library_dirs),
- Extension('android._android_billing',
- ['android/_android_billing.c', 'android/_android_billing_jni.c'],
- libraries=['log'],
- library_dirs=library_dirs)]
+modules = [
+ Extension('android._android',
+ ['android/_android.pyx', 'android/_android_jni.c'],
+ libraries=sdl_libs + ['log'],
+ library_dirs=library_dirs),
+ Extension('android._android_billing',
+ ['android/_android_billing.pyx', 'android/_android_billing_jni.c'],
+ libraries=['log'],
+ library_dirs=library_dirs),
+ Extension('android._android_sound',
+ ['android/_android_sound.pyx', 'android/_android_sound_jni.c'],
+ libraries=['log'],
+ library_dirs=library_dirs,
+ extra_compile_args=['-include', 'stdlib.h'])
+]
+
+cythonized_modules = cythonize(modules, compiler_directives={'language_level': "3"})
setup(name='android',
version='1.0',
packages=['android'],
package_dir={'android': 'android'},
- ext_modules=modules
+ ext_modules=cythonized_modules
)
From 9774cec2fd4b939322a6c978938c99c91a0d2fbf Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Sep 2025 18:11:45 +0530
Subject: [PATCH 132/158] `pyjnius`: update to 1.7.0
---
pythonforandroid/recipes/pyjnius/__init__.py | 5 ++---
.../recipes/pyjnius/cython_version_pin.patch | 9 ---------
2 files changed, 2 insertions(+), 12 deletions(-)
delete mode 100644 pythonforandroid/recipes/pyjnius/cython_version_pin.patch
diff --git a/pythonforandroid/recipes/pyjnius/__init__.py b/pythonforandroid/recipes/pyjnius/__init__.py
index c6b6746fa1..86d8803f18 100644
--- a/pythonforandroid/recipes/pyjnius/__init__.py
+++ b/pythonforandroid/recipes/pyjnius/__init__.py
@@ -6,15 +6,14 @@
class PyjniusRecipe(PyProjectRecipe):
- version = '1.6.1'
+ version = '1.7.0'
url = 'https://github.com/kivy/pyjnius/archive/{version}.zip'
name = 'pyjnius'
depends = [('genericndkbuild', 'sdl2', 'sdl3'), 'six']
site_packages_name = 'jnius'
-
+ hostpython_prerequisites = ["Cython<3.2"]
patches = [
"use_cython.patch",
- "cython_version_pin.patch",
('genericndkbuild_jnienv_getter.patch', will_build('genericndkbuild')),
('sdl3_jnienv_getter.patch', will_build('sdl3')),
]
diff --git a/pythonforandroid/recipes/pyjnius/cython_version_pin.patch b/pythonforandroid/recipes/pyjnius/cython_version_pin.patch
deleted file mode 100644
index 3d5cea2350..0000000000
--- a/pythonforandroid/recipes/pyjnius/cython_version_pin.patch
+++ /dev/null
@@ -1,9 +0,0 @@
---- pyjnius-1.6.1/pyproject.toml 2023-11-05 21:07:43.000000000 +0530
-+++ pyjnius-1.6.1.mod/pyproject.toml 2025-05-11 20:31:14.699072764 +0530
-@@ -2,5 +2,5 @@
- requires = [
- "setuptools>=58.0.0",
- "wheel",
-- "Cython"
-+ "Cython==3.0.0"
- ]
From b998bf47f7cf898d91b02dc91fab874ebb7b7b4a Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 14 Sep 2025 18:17:08 +0530
Subject: [PATCH 133/158] `pillow`: update to `11.3.0`
---
pythonforandroid/recipes/Pillow/__init__.py | 5 +-
.../recipes/Pillow/setup.py.patch | 64 ++++++-------------
2 files changed, 22 insertions(+), 47 deletions(-)
diff --git a/pythonforandroid/recipes/Pillow/__init__.py b/pythonforandroid/recipes/Pillow/__init__.py
index ffac810a2b..447803263f 100644
--- a/pythonforandroid/recipes/Pillow/__init__.py
+++ b/pythonforandroid/recipes/Pillow/__init__.py
@@ -23,11 +23,12 @@ class PillowRecipe(PyProjectRecipe):
- libwebp: library to encode and decode images in WebP format.
"""
- version = '10.3.0'
+ version = '11.3.0'
url = 'https://github.com/python-pillow/Pillow/archive/{version}.tar.gz'
site_packages_name = 'PIL'
patches = ["setup.py.patch"]
- depends = ['png', 'jpeg', 'freetype', 'setuptools']
+ depends = ['png', 'jpeg', 'freetype']
+ hostpython_prerequisites = ["setuptools>=77"]
opt_depends = ['libwebp']
def get_recipe_env(self, arch, **kwargs):
diff --git a/pythonforandroid/recipes/Pillow/setup.py.patch b/pythonforandroid/recipes/Pillow/setup.py.patch
index 2970e23670..aae9d1eb17 100644
--- a/pythonforandroid/recipes/Pillow/setup.py.patch
+++ b/pythonforandroid/recipes/Pillow/setup.py.patch
@@ -1,76 +1,50 @@
---- Pillow/setup.py 2024-05-24 19:35:08.270160608 +0530
-+++ Pillow.mod/setup.py 2024-05-24 22:07:52.741495666 +0530
-@@ -39,6 +39,7 @@
- LCMS_ROOT = None
- TIFF_ROOT = None
- ZLIB_ROOT = None
-+WEBP_ROOT = None
- FUZZING_BUILD = "LIB_FUZZING_ENGINE" in os.environ
+diff '--color=auto' -uNr Pillow-11.3.0/setup.py Pillow-11.3.0.mod/setup.py
+--- Pillow-11.3.0/setup.py 2025-07-01 13:11:24.000000000 +0530
++++ Pillow-11.3.0.mod/setup.py 2025-09-17 01:50:35.498105827 +0530
+@@ -156,6 +156,7 @@
- if sys.platform == "win32" and sys.version_info >= (3, 13):
-@@ -150,6 +151,7 @@
-
- def _find_library_dirs_ldconfig():
+ def _find_library_dirs_ldconfig() -> list[str]:
+ return []
# Based on ctypes.util from Python 2
ldconfig = "ldconfig" if shutil.which("ldconfig") else "/sbin/ldconfig"
-@@ -460,15 +462,16 @@
- "HARFBUZZ_ROOT": "harfbuzz",
- "FRIBIDI_ROOT": "fribidi",
- "LCMS_ROOT": "lcms2",
-+ "WEBP_ROOT": "libwebp",
- "IMAGEQUANT_ROOT": "libimagequant",
- }.items():
- root = globals()[root_name]
+@@ -514,12 +515,10 @@
if root is None and root_name in os.environ:
-- prefix = os.environ[root_name]
-- root = (os.path.join(prefix, "lib"), os.path.join(prefix, "include"))
+ root_prefix = os.environ[root_name]
+- root = (
+- os.path.join(root_prefix, "lib"),
+- os.path.join(root_prefix, "include"),
+- )
+ root = tuple(os.environ[root_name].split(":"))
if root is None and pkg_config:
+ continue
- if isinstance(lib_name, tuple):
- for lib_name2 in lib_name:
- _dbg(f"Looking for `{lib_name2}` using pkg-config.")
-@@ -495,14 +498,6 @@
- for include_dir in include_root:
- _add_directory(include_dirs, include_dir)
-
-- # respect CFLAGS/CPPFLAGS/LDFLAGS
-- for k in ("CFLAGS", "CPPFLAGS", "LDFLAGS"):
-- if k in os.environ:
-- for match in re.finditer(r"-I([^\s]+)", os.environ[k]):
-- _add_directory(include_dirs, match.group(1))
-- for match in re.finditer(r"-L([^\s]+)", os.environ[k]):
-- _add_directory(library_dirs, match.group(1))
--
- # include, rpath, if set as environment variables:
- for k in ("C_INCLUDE_PATH", "CPATH", "INCLUDE"):
- if k in os.environ:
-@@ -514,13 +509,10 @@
+ if isinstance(lib_name, str):
+ _dbg("Looking for `%s` using pkg-config.", lib_name)
+ root = pkg_config(lib_name)
+@@ -565,13 +564,11 @@
for d in os.environ[k].split(os.path.pathsep):
_add_directory(library_dirs, d)
- _add_directory(library_dirs, os.path.join(sys.prefix, "lib"))
- _add_directory(include_dirs, os.path.join(sys.prefix, "include"))
--
+
#
# add platform directories
- if self.disable_platform_guessing:
-+ if True:
++ if True: # self.disable_platform_guessing:
pass
elif sys.platform == "cygwin":
-@@ -614,7 +606,7 @@
+@@ -674,7 +671,7 @@
# FIXME: check /opt/stuff directories here?
# standard locations
- if not self.disable_platform_guessing:
-+ if False: #not self.disable_platform_guessing:
++ if False: # not self.disable_platform_guessing:
_add_directory(library_dirs, "/usr/local/lib")
_add_directory(include_dirs, "/usr/local/include")
From 5be7531c85f7e5c486a08a3041d786f210ffb036 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 20 Sep 2025 21:08:07 +0530
Subject: [PATCH 134/158] fix plat-name
---
pythonforandroid/recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 0ebe005d14..44aab013a6 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1222,7 +1222,7 @@ def get_recipe_env(self, arch, **kwargs):
build_opts = join(build_dir, "build-opts.cfg")
with open(build_opts, "w") as file:
- file.write("[bdist_wheel]\nplat-name={}".format(
+ file.write("[bdist_wheel]\nplat_name={}".format(
self.get_wheel_platform_tag(arch)
))
file.close()
From 48b6b4a70322d0aeb290f9d3bd091a103d1399fc Mon Sep 17 00:00:00 2001
From: Gonzalo Atienza <gonatienza@gmail.com>
Date: Sun, 14 Sep 2025 21:06:37 -0400
Subject: [PATCH 135/158] pynacl-recipe
---
pythonforandroid/recipes/pynacl/__init__.py | 31 +++++++++++++--------
1 file changed, 19 insertions(+), 12 deletions(-)
diff --git a/pythonforandroid/recipes/pynacl/__init__.py b/pythonforandroid/recipes/pynacl/__init__.py
index 0ab9352eeb..6c5e50762d 100644
--- a/pythonforandroid/recipes/pynacl/__init__.py
+++ b/pythonforandroid/recipes/pynacl/__init__.py
@@ -1,27 +1,34 @@
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
import os
-class PyNaCLRecipe(CompiledComponentsPythonRecipe):
+class PyNaCLRecipe(PyProjectRecipe):
name = 'pynacl'
version = '1.3.0'
- url = 'https://pypi.python.org/packages/source/P/PyNaCl/PyNaCl-{version}.tar.gz'
+ url = 'https://github.com/pyca/pynacl/archive/refs/tags/{version}.tar.gz'
depends = ['hostpython3', 'six', 'setuptools', 'cffi', 'libsodium']
call_hostpython_via_targetpython = False
+ hostpython_prerequisites = ["cffi>=2.0.0"]
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
env['SODIUM_INSTALL'] = 'system'
libsodium_build_dir = self.get_recipe(
- 'libsodium', self.ctx).get_build_dir(arch.arch)
- env['CFLAGS'] += ' -I{}'.format(os.path.join(libsodium_build_dir,
- 'src/libsodium/include'))
- env['LDFLAGS'] += ' -L{}'.format(
- self.ctx.get_libs_dir(arch.arch) +
- '-L{}'.format(self.ctx.libs_dir)) + ' -L{}'.format(
- libsodium_build_dir)
+ 'libsodium', self.ctx
+ ).get_build_dir(arch.arch)
+
+ env['CFLAGS'] += ' -I{}'.format(
+ os.path.join(libsodium_build_dir, 'src/libsodium/include')
+ )
+
+ for ldflag in [
+ self.ctx.get_libs_dir(arch.arch),
+ self.ctx.libs_dir,
+ libsodium_build_dir
+ ]:
+ env['LDFLAGS'] += ' -L{}'.format(ldflag)
return env
From bb02aeff6128ba80270d066d89968285126d4b18 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 20 Sep 2025 21:15:13 +0530
Subject: [PATCH 136/158] `opencv`: fix build
---
pythonforandroid/recipes/opencv/__init__.py | 16 +++++++--
.../recipes/opencv/patches/p4a_build.patch | 33 +++++++------------
2 files changed, 25 insertions(+), 24 deletions(-)
diff --git a/pythonforandroid/recipes/opencv/__init__.py b/pythonforandroid/recipes/opencv/__init__.py
index 8f7e2b8c07..ce78879cd7 100644
--- a/pythonforandroid/recipes/opencv/__init__.py
+++ b/pythonforandroid/recipes/opencv/__init__.py
@@ -15,7 +15,7 @@ class OpenCVRecipe(NDKRecipe):
build of most of the libraries of the opencv's package, so we can
process images, videos, objects, photos...
'''
- version = '4.5.1'
+ version = '4.12.0'
url = 'https://github.com/opencv/opencv/archive/{version}.zip'
depends = ['numpy']
patches = ['patches/p4a_build.patch']
@@ -68,8 +68,9 @@ def build_arch(self, arch):
python_link_version = self.ctx.python_recipe.link_version
python_library = join(python_link_root,
'libpython{}.so'.format(python_link_version))
- python_include_numpy = join(python_site_packages,
- 'numpy', 'core', 'include')
+ python_include_numpy = join(
+ self.ctx.get_python_install_dir(arch.arch), "numpy/_core/include",
+ )
shprint(sh.cmake,
'-DP4A=ON',
@@ -136,6 +137,15 @@ def build_arch(self, arch):
self.get_build_dir(arch.arch),
_env=env)
+
+ # patch link.txt for unsupported flag
+ link_txt = 'modules/python3/CMakeFiles/opencv_python3.dir/link.txt'
+ with open(link_txt, 'r+') as f:
+ content = f.read().replace('-version', ' ')
+ f.seek(0)
+ f.write(content)
+ f.truncate()
+
shprint(sh.make, '-j' + str(cpu_count()), 'opencv_python' + python_major)
# Install python bindings (cv2.so)
shprint(sh.cmake, '-DCOMPONENT=python', '-P', './cmake_install.cmake')
diff --git a/pythonforandroid/recipes/opencv/patches/p4a_build.patch b/pythonforandroid/recipes/opencv/patches/p4a_build.patch
index fd60c01d38..10b81f776b 100644
--- a/pythonforandroid/recipes/opencv/patches/p4a_build.patch
+++ b/pythonforandroid/recipes/opencv/patches/p4a_build.patch
@@ -1,31 +1,22 @@
-This patch allow that the opencv's build command correctly detects our version
-of python, so we can successfully build the python bindings (cv2.so)
---- opencv-4.0.1/cmake/OpenCVDetectPython.cmake.orig 2018-12-22 08:03:30.000000000 +0100
-+++ opencv-4.0.1/cmake/OpenCVDetectPython.cmake 2019-01-31 11:33:10.896502978 +0100
-@@ -175,7 +175,7 @@ if(NOT ${found})
+diff '--color=auto' -uNr opencv-4.12.0/cmake/OpenCVDetectPython.cmake opencv-4.12.0.mod/cmake/OpenCVDetectPython.cmake
+--- opencv-4.12.0/cmake/OpenCVDetectPython.cmake 2025-07-02 13:24:13.000000000 +0530
++++ opencv-4.12.0.mod/cmake/OpenCVDetectPython.cmake 2025-09-20 22:22:14.961944470 +0530
+@@ -175,7 +175,7 @@
endif()
endif()
-
-- if(NOT ANDROID AND NOT IOS)
-+ if(P4A OR NOT ANDROID AND NOT IOS)
+
+- if(NOT ANDROID AND NOT IOS AND NOT XROS)
++ if(P4A OR NOT ANDROID AND NOT IOS AND NOT XROS)
if(CMAKE_HOST_UNIX)
- execute_process(COMMAND ${_executable} -c "from distutils.sysconfig import *; print(get_python_lib())"
+ execute_process(COMMAND ${_executable} -c "from sysconfig import *; print(get_path('purelib'))"
RESULT_VARIABLE _cvpy_process
-@@ -244,7 +244,7 @@ if(NOT ${found})
- OUTPUT_STRIP_TRAILING_WHITESPACE)
- endif()
- endif()
-- endif(NOT ANDROID AND NOT IOS)
-+ endif(P4A OR NOT ANDROID AND NOT IOS)
- endif()
-
- # Export return values
---- opencv-4.0.1/modules/python/CMakeLists.txt.orig 2018-12-22 08:03:30.000000000 +0100
-+++ opencv-4.0.1/modules/python/CMakeLists.txt 2019-01-31 11:47:17.100494908 +0100
+diff '--color=auto' -uNr opencv-4.12.0/modules/python/CMakeLists.txt opencv-4.12.0.mod/modules/python/CMakeLists.txt
+--- opencv-4.12.0/modules/python/CMakeLists.txt 2025-07-02 13:24:13.000000000 +0530
++++ opencv-4.12.0.mod/modules/python/CMakeLists.txt 2025-09-20 22:23:15.124356524 +0530
@@ -3,7 +3,7 @@
# ----------------------------------------------------------------------------
if(DEFINED OPENCV_INITIAL_PASS) # OpenCV build
-
+
-if(ANDROID OR APPLE_FRAMEWORK OR WINRT)
+if(ANDROID AND NOT P4A OR APPLE_FRAMEWORK OR WINRT)
ocv_module_disable_(python2)
From 9d68069566083850fe7ff55888a76d6744b8e437 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Sun, 5 Oct 2025 14:00:32 +0000
Subject: [PATCH 137/158] Bump pypa/gh-action-pypi-publish in
/.github/workflows
Bumps [pypa/gh-action-pypi-publish](https://github.com/pypa/gh-action-pypi-publish) from 1.4.2 to 1.13.0.
- [Release notes](https://github.com/pypa/gh-action-pypi-publish/releases)
- [Commits](https://github.com/pypa/gh-action-pypi-publish/compare/v1.4.2...v1.13.0)
---
updated-dependencies:
- dependency-name: pypa/gh-action-pypi-publish
dependency-version: 1.13.0
dependency-type: direct:production
...
Signed-off-by: dependabot[bot] <support@github.com>
---
.github/workflows/pypi-release.yml | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/.github/workflows/pypi-release.yml b/.github/workflows/pypi-release.yml
index 487a903016..7f06df83c2 100644
--- a/.github/workflows/pypi-release.yml
+++ b/.github/workflows/pypi-release.yml
@@ -19,7 +19,7 @@ jobs:
twine check dist/*
- name: Publish package
if: github.event_name == 'push' && startsWith(github.event.ref, 'refs/tags')
- uses: pypa/gh-action-pypi-publish@v1.4.2
+ uses: pypa/gh-action-pypi-publish@v1.13.0
with:
user: __token__
password: ${{ secrets.pypi_password }}
From 40bbdacf487aaecaf776e5b84a4d40e26ccf7136 Mon Sep 17 00:00:00 2001
From: Christian Clauss <cclauss@me.com>
Date: Tue, 7 Oct 2025 21:12:33 +0200
Subject: [PATCH 138/158] GitHub Actions: Add Python 3.12, 3.13, 3.14, and
3.14t to the testing (#3076)
* GitHub Actions: Add Python 3.12 and 3.13 to the testing
* fail-fast: false
* Re-add the EOL Python 3.8
* Add Python 3.14 release candidate 3
* Add free threaded Python 3.14t to workflow matrix
* macos-13 --> macos-15-intel
Updated macOS version references in the workflow to reflect changes in the GitHub Actions environment.
---
.github/workflows/push.yml | 42 ++++++++++++++++++++------------------
1 file changed, 22 insertions(+), 20 deletions(-)
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 99f63de53a..2d64086bb0 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -19,9 +19,9 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
- name: Set up Python 3.x
- uses: actions/setup-python@v5
+ uses: actions/setup-python@v6
with:
python-version: 3.x
- name: Run flake8
@@ -35,16 +35,18 @@ jobs:
needs: flake8
runs-on: ${{ matrix.os }}
strategy:
+ fail-fast: false
matrix:
- python-version: ['3.8', '3.9', '3.10', '3.11']
- os: [ubuntu-latest, macOs-latest]
+ python-version: ['3.8', '3.9', '3.10', '3.11', '3.12', '3.13', '3.14', '3.14t']
+ os: [ubuntu-latest, macos-latest]
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
- name: Set up Python ${{ matrix.python-version }}
- uses: actions/setup-python@v5
+ uses: actions/setup-python@v6
with:
python-version: ${{ matrix.python-version }}
+ allow-prereleases: true
- name: Tox tests
run: |
python -m pip install --upgrade pip
@@ -114,9 +116,9 @@ jobs:
continue-on-error: true
strategy:
matrix:
- # macos-latest (ATM macos-14) runs on Apple Silicon,
- # macos-13 runs on Intel
- runs_on: ['macos-latest', 'macos-13']
+ # macos-latest (ATM macos-15) runs on Apple Silicon,
+ # macos-15-intel runs on Intel
+ runs_on: ['macos-latest', 'macos-15-intel']
bootstrap:
- name: sdl2
target: testapps-with-numpy
@@ -129,14 +131,14 @@ jobs:
ANDROID_NDK_HOME: ${HOME}/.android/android-ndk
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
- name: Set up Python 3.x
- uses: actions/setup-python@v5
+ uses: actions/setup-python@v6
with:
python-version: 3.x
- name: Install python-for-android
run: |
- python3 -m pip install -e .
+ python3 -m pip install --editable .
- name: Install prerequisites via pythonforandroid/prerequisites.py (Experimental)
run: |
python3 pythonforandroid/prerequisites.py
@@ -174,7 +176,7 @@ jobs:
REBUILD_UPDATED_RECIPES_EXTRA_ARGS: --arch=${{ matrix.android_arch }}
steps:
- name: Checkout python-for-android (all-history)
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
with:
fetch-depth: 0
# helps with GitHub runner getting out of space
@@ -201,9 +203,9 @@ jobs:
strategy:
matrix:
android_arch: ["arm64-v8a", "armeabi-v7a", "x86_64", "x86"]
- # macos-latest (ATM macos-14) runs on Apple Silicon,
- # macos-13 runs on Intel
- runs_on: ['macos-latest', 'macos-13']
+ # macos-latest (ATM macos-15) runs on Apple Silicon,
+ # macos-15-intel runs on Intel
+ runs_on: ['macos-latest', 'macos-15-intel']
env:
ANDROID_HOME: ${HOME}/.android
ANDROID_SDK_ROOT: ${HOME}/.android/android-sdk
@@ -212,16 +214,16 @@ jobs:
REBUILD_UPDATED_RECIPES_EXTRA_ARGS: --arch=${{ matrix.android_arch }}
steps:
- name: Checkout python-for-android (all-history)
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
with:
fetch-depth: 0
- name: Set up Python 3.x
- uses: actions/setup-python@v5
+ uses: actions/setup-python@v6
with:
python-version: 3.x
- name: Install python-for-android
run: |
- python3 -m pip install -e .
+ python3 -m pip install --editable .
- name: Install prerequisites via pythonforandroid/prerequisites.py (Experimental)
run: |
python3 pythonforandroid/prerequisites.py
@@ -244,7 +246,7 @@ jobs:
documentation:
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v4
+ - uses: actions/checkout@v5
- name: Requirements
run: |
python -m pip install --upgrade pip
From ee326409f3141ba17b2223156ac835400060a419 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sun, 5 Oct 2025 00:30:58 +0200
Subject: [PATCH 139/158] :construction_worker: Runtime test the app from the
CI
Leverage `reactivecircus/android-emulator-runner` to run the testapps
artifact on an emulator and check the ADB logs.
We're currently checking for the following pattern from the logcat output:
- "I python: Initialized python"
- "I python: Ran 14 tests in"
- "I python: OK"
Note that we're keeping things simple for the first iteration and we only
run the `ubuntu-latest-sdl2-artifacts` artifact.
In the future we may want to test the different bootstraps, build host
and even updated recipes dynamically.
Also bump actions/checkout to v5.
---
.github/workflows/push.yml | 22 +++++++++++++-
ci/run_emulator_tests.sh | 60 ++++++++++++++++++++++++++++++++++++++
2 files changed, 81 insertions(+), 1 deletion(-)
create mode 100755 ci/run_emulator_tests.sh
diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml
index 2d64086bb0..66865e3855 100644
--- a/.github/workflows/push.yml
+++ b/.github/workflows/push.yml
@@ -80,7 +80,7 @@ jobs:
target: testapps-qt
steps:
- name: Checkout python-for-android
- uses: actions/checkout@v4
+ uses: actions/checkout@v5
- name: Build python-for-android docker image
run: |
docker build --tag=kivy/python-for-android .
@@ -164,6 +164,26 @@ jobs:
name: ${{ matrix.runs_on }}-${{ matrix.bootstrap.name }}-artifacts
path: dist
+ test_on_emulator:
+ name: Run App on Emulator
+ needs: ubuntu_build
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@v5
+ - name: Download Artifacts
+ uses: actions/download-artifact@v5
+ with:
+ name: ubuntu-latest-sdl2-artifacts
+ path: dist/
+
+ - name: Setup and start Android Emulator
+ uses: reactivecircus/android-emulator-runner@v2
+ with:
+ api-level: 30
+ arch: x86_64
+ script: ci/run_emulator_tests.sh
+
ubuntu_rebuild_updated_recipes:
name: Test updated recipes for arch ${{ matrix.android_arch }} [ ubuntu-latest ]
needs: [flake8]
diff --git a/ci/run_emulator_tests.sh b/ci/run_emulator_tests.sh
new file mode 100755
index 0000000000..9eaaf3d99c
--- /dev/null
+++ b/ci/run_emulator_tests.sh
@@ -0,0 +1,60 @@
+#!/bin/bash
+set -euxo pipefail
+
+# Find the built APK file
+APK_FILE=$(find dist -name "*.apk" -print -quit)
+
+if [ -z "$APK_FILE" ]; then
+ echo "Error: No APK file found in dist/"
+ exit 1
+fi
+
+echo "Installing $APK_FILE..."
+adb install "$APK_FILE"
+
+# Extract package and activity names
+AAPT2_PATH=$(find ${ANDROID_HOME}/build-tools/ -name aapt2 | sort -r | head -n 1)
+APP_PACKAGE=$(${AAPT2_PATH} dump badging "${APK_FILE}" | awk -F"'" '/package: name=/{print $2}')
+APP_ACTIVITY=$(${AAPT2_PATH} dump badging "${APK_FILE}" | awk -F"'" '/launchable-activity/ {print $2}')
+
+echo "Launching $APP_PACKAGE/$APP_ACTIVITY..."
+adb shell am start -n "$APP_PACKAGE/$APP_ACTIVITY" -a android.intent.action.MAIN -c android.intent.category.LAUNCHER
+
+# Poll for test completion with timeout
+MAX_WAIT=300
+POLL_INTERVAL=5
+elapsed=0
+
+echo "Waiting for tests to complete (max ${MAX_WAIT}s)..."
+
+while [ $elapsed -lt $MAX_WAIT ]; do
+ # Dump current logs
+ adb logcat -d -s python:I *:S > app_logs.txt
+
+ # Check if all success patterns are present
+ if grep --extended-regexp --quiet "I python[ ]+: Initialized python" app_logs.txt && \
+ grep --extended-regexp --quiet "I python[ ]+: Ran 14 tests in" app_logs.txt && \
+ grep --extended-regexp --quiet "I python[ ]+: OK" app_logs.txt; then
+ echo "✅ SUCCESS: App launched and all unit tests passed in ${elapsed}s."
+ exit 0
+ fi
+
+ # Check for early failure indicators
+ if grep --extended-regexp --quiet "I python[ ]+: FAILED" app_logs.txt; then
+ echo "❌ FAILURE: Tests failed after ${elapsed}s."
+ echo "--- Full Logs ---"
+ cat app_logs.txt
+ echo "-----------------"
+ exit 1
+ fi
+
+ sleep $POLL_INTERVAL
+ elapsed=$((elapsed + POLL_INTERVAL))
+ echo "Still waiting... (${elapsed}s elapsed)"
+done
+
+echo "❌ TIMEOUT: Tests did not complete within ${MAX_WAIT}s."
+echo "--- Full Logs ---"
+cat app_logs.txt
+echo "-----------------"
+exit 1
From 6b66944a2f51e0c848c7ac51e04a771324067ecc Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Wed, 8 Oct 2025 22:39:43 +0530
Subject: [PATCH 140/158] `python`: add `3.14` support (#3180)
---
pythonforandroid/archs.py | 5 +-
.../common/build/jni/application/src/start.c | 82 ++++++++---
.../java/org/kivy/android/PythonUtil.java | 39 +++--
pythonforandroid/recipe.py | 114 ++++++++++-----
.../recipes/hostpython3/__init__.py | 42 ++++--
.../patches/pyconfig_detection.patch | 13 --
pythonforandroid/recipes/numpy/__init__.py | 5 +-
pythonforandroid/recipes/openssl/__init__.py | 38 ++---
.../recipes/openssl/disable-sover.patch | 11 --
pythonforandroid/recipes/python3/__init__.py | 135 ++++++++++--------
.../python3/patches/3.14_armv7l_fix.patch | 12 ++
.../recipes/setuptools/__init__.py | 9 +-
tests/recipes/test_openssl.py | 15 +-
tests/recipes/test_python3.py | 11 +-
tests/recipes/test_reportlab.py | 1 +
15 files changed, 302 insertions(+), 230 deletions(-)
delete mode 100644 pythonforandroid/recipes/hostpython3/patches/pyconfig_detection.patch
delete mode 100644 pythonforandroid/recipes/openssl/disable-sover.patch
create mode 100644 pythonforandroid/recipes/python3/patches/3.14_armv7l_fix.patch
diff --git a/pythonforandroid/archs.py b/pythonforandroid/archs.py
index 4137768096..c2c1f91fc7 100644
--- a/pythonforandroid/archs.py
+++ b/pythonforandroid/archs.py
@@ -132,10 +132,7 @@ def get_env(self, with_flags_in_cc=True):
env['CPPFLAGS'] = ' '.join(self.common_cppflags).format(
ctx=self.ctx,
command_prefix=self.command_prefix,
- python_includes=join(
- self.ctx.get_python_install_dir(self.arch),
- 'include/python{}'.format(self.ctx.python_recipe.version[0:3]),
- ),
+ python_includes=join(self.ctx.python_recipe.get_build_dir(self.arch), 'Include')
)
# LDFLAGS: Link the extra global link paths first before anything else
diff --git a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
index ef910cab3d..1862208943 100644
--- a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
+++ b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
@@ -31,6 +31,7 @@
#define ENTRYPOINT_MAXLEN 128
#define LOG(n, x) __android_log_write(ANDROID_LOG_INFO, (n), (x))
#define LOGP(x) LOG("python", (x))
+#define P4A_MIN_VER 11
static PyObject *androidembed_log(PyObject *self, PyObject *args) {
char *logstr = NULL;
@@ -154,11 +155,6 @@ int main(int argc, char *argv[]) {
Py_NoSiteFlag=1;
#endif
-#if PY_MAJOR_VERSION < 3
- Py_SetProgramName("android_python");
-#else
- Py_SetProgramName(L"android_python");
-#endif
#if PY_MAJOR_VERSION >= 3
/* our logging module for android
@@ -174,40 +170,80 @@ int main(int argc, char *argv[]) {
char python_bundle_dir[256];
snprintf(python_bundle_dir, 256,
"%s/_python_bundle", getenv("ANDROID_UNPACK"));
- if (dir_exists(python_bundle_dir)) {
- LOGP("_python_bundle dir exists");
- snprintf(paths, 256,
- "%s/stdlib.zip:%s/modules",
- python_bundle_dir, python_bundle_dir);
- LOGP("calculated paths to be...");
- LOGP(paths);
+ #if PY_MAJOR_VERSION >= 3
- #if PY_MAJOR_VERSION >= 3
- wchar_t *wchar_paths = Py_DecodeLocale(paths, NULL);
- Py_SetPath(wchar_paths);
+ #if PY_MINOR_VERSION >= P4A_MIN_VER
+ PyConfig config;
+ PyConfig_InitPythonConfig(&config);
+ config.program_name = L"android_python";
+ #else
+ Py_SetProgramName(L"android_python");
#endif
- LOGP("set wchar paths...");
+ #else
+ Py_SetProgramName("android_python");
+ #endif
+
+ if (dir_exists(python_bundle_dir)) {
+ LOGP("_python_bundle dir exists");
+
+ #if PY_MAJOR_VERSION >= 3
+ #if PY_MINOR_VERSION >= P4A_MIN_VER
+
+ wchar_t wchar_zip_path[256];
+ wchar_t wchar_modules_path[256];
+ swprintf(wchar_zip_path, 256, L"%s/stdlib.zip", python_bundle_dir);
+ swprintf(wchar_modules_path, 256, L"%s/modules", python_bundle_dir);
+
+ config.module_search_paths_set = 1;
+ PyWideStringList_Append(&config.module_search_paths, wchar_zip_path);
+ PyWideStringList_Append(&config.module_search_paths, wchar_modules_path);
+ #else
+ char paths[512];
+ snprintf(paths, 512, "%s/stdlib.zip:%s/modules", python_bundle_dir, python_bundle_dir);
+ wchar_t *wchar_paths = Py_DecodeLocale(paths, NULL);
+ Py_SetPath(wchar_paths);
+ #endif
+
+ #endif
+
+ LOGP("set wchar paths...");
} else {
LOGP("_python_bundle does not exist...this not looks good, all python"
" recipes should have this folder, should we expect a crash soon?");
}
- Py_Initialize();
+#if PY_MAJOR_VERSION >= 3 && PY_MINOR_VERSION >= P4A_MIN_VER
+ PyStatus status = Py_InitializeFromConfig(&config);
+ if (PyStatus_Exception(status)) {
+ LOGP("Python initialization failed:");
+ LOGP(status.err_msg);
+ }
+#else
+ Py_Initialize();
+ LOGP("Python initialized using legacy Py_Initialize().");
+#endif
+
LOGP("Initialized python");
- /* ensure threads will work.
- */
- LOGP("AND: Init threads");
- PyEval_InitThreads();
+ /* < 3.9 requires explicit GIL initialization
+ * 3.9+ PyEval_InitThreads() is deprecated and unnecessary
+ */
+ #if PY_VERSION_HEX < 0x03090000
+ LOGP("Initializing threads (required for Python < 3.9)");
+ PyEval_InitThreads();
+ #endif
#if PY_MAJOR_VERSION < 3
initandroidembed();
#endif
- PyRun_SimpleString("import androidembed\nandroidembed.log('testing python "
- "print redirection')");
+ PyRun_SimpleString(
+ "import androidembed\n"
+ "androidembed.log('testing python print redirection')"
+
+ );
/* inject our bootstrap code to redirect python stdin/stdout
* replace sys.path with our path
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
index 83d11639bb..065f43c3bd 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
@@ -39,27 +39,22 @@ protected static void addLibraryIfExists(ArrayList<String> libsList, String patt
}
protected static ArrayList<String> getLibraries(File libsDir) {
- ArrayList<String> libsList = new ArrayList<String>();
- addLibraryIfExists(libsList, "sqlite3", libsDir);
- addLibraryIfExists(libsList, "ffi", libsDir);
- addLibraryIfExists(libsList, "png16", libsDir);
- addLibraryIfExists(libsList, "ssl.*", libsDir);
- addLibraryIfExists(libsList, "crypto.*", libsDir);
- addLibraryIfExists(libsList, "SDL2", libsDir);
- addLibraryIfExists(libsList, "SDL2_image", libsDir);
- addLibraryIfExists(libsList, "SDL2_mixer", libsDir);
- addLibraryIfExists(libsList, "SDL2_ttf", libsDir);
- addLibraryIfExists(libsList, "SDL3", libsDir);
- addLibraryIfExists(libsList, "SDL3_image", libsDir);
- addLibraryIfExists(libsList, "SDL3_mixer", libsDir);
- addLibraryIfExists(libsList, "SDL3_ttf", libsDir);
- libsList.add("python3.5m");
- libsList.add("python3.6m");
- libsList.add("python3.7m");
- libsList.add("python3.8");
- libsList.add("python3.9");
- libsList.add("python3.10");
- libsList.add("python3.11");
+ ArrayList<String> libsList = new ArrayList<>();
+
+ String[] libNames = {
+ "sqlite3", "ffi", "png16", "ssl.*", "crypto.*",
+ "SDL2", "SDL2_image", "SDL2_mixer", "SDL2_ttf",
+ "SDL3", "SDL3_image", "SDL3_mixer", "SDL3_ttf"
+ };
+
+ for (String name : libNames) {
+ addLibraryIfExists(libsList, name, libsDir);
+ }
+
+ for (int v = 5; v <= 14; v++) {
+ libsList.add("python3." + v + (v <= 7 ? "m" : ""));
+ }
+
libsList.add("main");
return libsList;
}
@@ -79,7 +74,7 @@ public static void loadLibraries(File filesDir, File libsDir) {
// load, and it has failed, give a more
// general error
Log.v(TAG, "Library loading error: " + e.getMessage());
- if (lib.startsWith("python3.11") && !foundPython) {
+ if (lib.startsWith("python3.14") && !foundPython) {
throw new RuntimeException("Could not load any libpythonXXX.so");
} else if (lib.startsWith("python")) {
continue;
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 44aab013a6..8808ffc0d2 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -12,6 +12,7 @@
from urllib.request import urlretrieve
from os import listdir, unlink, environ, curdir, walk
from sys import stdout
+from multiprocessing import cpu_count
import time
try:
from urlparse import urlparse
@@ -517,7 +518,7 @@ def unpack(self, arch):
for entry in listdir(extraction_filename):
# Previously we filtered out the .git folder, but during the build process for some recipes
# (e.g. when version is parsed by `setuptools_scm`) that may be needed.
- shprint(sh.cp, '-Rv',
+ shprint(sh.cp, '-R',
join(extraction_filename, entry),
directory_name)
else:
@@ -830,6 +831,8 @@ def build_arch(self, arch, *extra_args):
shprint(
sh.Command(join(self.ctx.ndk_dir, "ndk-build")),
'V=1',
+ "-j",
+ str(cpu_count()),
'NDK_DEBUG=' + ("1" if self.ctx.build_as_debuggable else "0"),
'APP_PLATFORM=android-' + str(self.ctx.ndk_api),
'APP_ABI=' + arch.arch,
@@ -878,6 +881,8 @@ class PythonRecipe(Recipe):
hostpython_prerequisites = []
'''List of hostpython packages required to build a recipe'''
+ _host_recipe = None
+
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
if 'python3' not in self.depends:
@@ -890,6 +895,10 @@ def __init__(self, *args, **kwargs):
depends = list(set(depends))
self.depends = depends
+ def prebuild_arch(self, arch):
+ self._host_recipe = Recipe.get_recipe("hostpython3", self.ctx)
+ return super().prebuild_arch(arch)
+
def clean_build(self, arch=None):
super().clean_build(arch=arch)
name = self.folder_name
@@ -907,8 +916,7 @@ def clean_build(self, arch=None):
def real_hostpython_location(self):
host_name = 'host{}'.format(self.ctx.python_recipe.name)
if host_name == 'hostpython3':
- python_recipe = Recipe.get_recipe(host_name, self.ctx)
- return python_recipe.python_exe
+ return self._host_recipe.python_exe
else:
python_recipe = self.ctx.python_recipe
return 'python{}'.format(python_recipe.version)
@@ -927,14 +935,50 @@ def folder_name(self):
name = self.name
return name
+ def patch_shebang(self, _file, original_bin):
+ _file_des = open(_file, "r")
+
+ try:
+ data = _file_des.readlines()
+ except UnicodeDecodeError:
+ return
+
+ if "#!" in (line := data[0]):
+ if line.split("#!")[-1].strip() == original_bin:
+ return
+
+ info(f"Fixing shebang for '{_file}'")
+ data.pop(0)
+ data.insert(0, "#!" + original_bin + "\n")
+ _file_des.close()
+ _file_des = open(_file, "w")
+ _file_des.write("".join(data))
+ _file_des.close()
+
+ def patch_shebangs(self, path, original_bin):
+ if not isdir(path):
+ warning(f"Shebang patch skipped: '{path}' does not exist.")
+ return
+ # set correct shebang
+ for file in listdir(path):
+ _file = join(path, file)
+ self.patch_shebang(_file, original_bin)
+
def get_recipe_env(self, arch=None, with_flags_in_cc=True):
+ if self._host_recipe is None:
+ self._host_recipe = Recipe.get_recipe("hostpython3", self.ctx)
+
env = super().get_recipe_env(arch, with_flags_in_cc)
- env['PYTHONNOUSERSITE'] = '1'
# Set the LANG, this isn't usually important but is a better default
# as it occasionally matters how Python e.g. reads files
env['LANG'] = "en_GB.UTF-8"
+
# Binaries made by packages installed by pip
- env["PATH"] = join(self.hostpython_site_dir, "bin") + ":" + env["PATH"]
+ self.patch_shebangs(self._host_recipe.local_bin, self.real_hostpython_location)
+ env["PATH"] = self._host_recipe.local_bin + ":" + self._host_recipe.site_bin + ":" + env["PATH"]
+
+ host_env = self.get_hostrecipe_env()
+ env['PYTHONPATH'] = host_env["PYTHONPATH"]
if not self.call_hostpython_via_targetpython:
env['CFLAGS'] += ' -I{}'.format(
@@ -945,18 +989,6 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
self.ctx.python_recipe.link_version,
)
- hppath = []
- hppath.append(join(dirname(self.hostpython_location), 'Lib'))
- hppath.append(join(hppath[0], 'site-packages'))
- builddir = join(dirname(self.hostpython_location), 'build')
- if exists(builddir):
- hppath += [join(builddir, d) for d in listdir(builddir)
- if isdir(join(builddir, d))]
- if len(hppath) > 0:
- if 'PYTHONPATH' in env:
- env['PYTHONPATH'] = ':'.join(hppath + [env['PYTHONPATH']])
- else:
- env['PYTHONPATH'] = ':'.join(hppath)
return env
def should_build(self, arch):
@@ -988,18 +1020,25 @@ def install_python_package(self, arch, name=None, env=None, is_dir=True):
hostpython = sh.Command(self.hostpython_location)
hpenv = env.copy()
with current_directory(self.get_build_dir(arch.arch)):
- shprint(hostpython, 'setup.py', 'install', '-O2',
- '--root={}'.format(self.ctx.get_python_install_dir(arch.arch)),
- '--install-lib=.',
- _env=hpenv, *self.setup_extra_args)
- # If asked, also install in the hostpython build dir
- if self.install_in_hostpython:
- self.install_hostpython_package(arch)
+ if isfile("setup.py"):
+ shprint(hostpython, 'setup.py', 'install', '-O2',
+ '--root={}'.format(self.ctx.get_python_install_dir(arch.arch)),
+ '--install-lib=.',
+ _env=hpenv, *self.setup_extra_args)
- def get_hostrecipe_env(self, arch):
+ # If asked, also install in the hostpython build dir
+ if self.install_in_hostpython:
+ self.install_hostpython_package(arch)
+ else:
+ warning("`PythonRecipe.install_python_package` called without `setup.py` file!")
+
+ def get_hostrecipe_env(self, arch=None):
env = environ.copy()
- env['PYTHONPATH'] = self.hostpython_site_dir
+ _python_path = self._host_recipe.get_path_to_python()
+ libdir = glob.glob(join(_python_path, "build", "lib*"))
+ env['PYTHONPATH'] = self._host_recipe.site_dir + ":" + join(
+ _python_path, "Modules") + ":" + (libdir[0] if libdir else "")
return env
@property
@@ -1010,8 +1049,8 @@ def install_hostpython_package(self, arch):
env = self.get_hostrecipe_env(arch)
real_hostpython = sh.Command(self.real_hostpython_location)
shprint(real_hostpython, 'setup.py', 'install', '-O2',
- '--root={}'.format(dirname(self.real_hostpython_location)),
'--install-lib=Lib/site-packages',
+ '--root={}'.format(self._host_recipe.site_root),
_env=env, *self.setup_extra_args)
@property
@@ -1029,7 +1068,7 @@ def install_hostpython_prerequisites(self, packages=None, force_upgrade=True):
pip_options = [
"install",
*packages,
- "--target", self.hostpython_site_dir, "--python-version",
+ "--target", self._host_recipe.site_dir, "--python-version",
self.ctx.python_recipe.version,
# Don't use sources, instead wheels
"--only-binary=:all:",
@@ -1037,7 +1076,9 @@ def install_hostpython_prerequisites(self, packages=None, force_upgrade=True):
if force_upgrade:
pip_options.append("--upgrade")
# Use system's pip
- shprint(sh.pip, *pip_options)
+ pip_env = self.get_hostrecipe_env()
+ pip_env["HOME"] = "/tmp"
+ shprint(sh.Command(self.real_hostpython_location), "-m", "pip", *pip_options, _env=pip_env)
def restore_hostpython_prerequisites(self, packages):
_packages = []
@@ -1231,7 +1272,7 @@ def get_recipe_env(self, arch, **kwargs):
return env
def get_wheel_platform_tag(self, arch):
- return "android_" + {
+ return f"android_{self.ctx.ndk_api}_" + {
"armeabi-v7a": "arm",
"arm64-v8a": "aarch64",
"x86_64": "x86_64",
@@ -1265,10 +1306,17 @@ def install_wheel(self, arch, built_wheels):
wf.close()
def build_arch(self, arch):
+
+ build_dir = self.get_build_dir(arch.arch)
+ if not (isfile(join(build_dir, "pyproject.toml")) or isfile(join(build_dir, "setup.py"))):
+ warning("Skipping build because it does not appear to be a Python project.")
+ return
+
self.install_hostpython_prerequisites(
- packages=["build[virtualenv]", "pip"] + self.hostpython_prerequisites
+ packages=["build[virtualenv]", "pip", "setuptools"] + self.hostpython_prerequisites
)
- build_dir = self.get_build_dir(arch.arch)
+ self.patch_shebangs(self._host_recipe.site_bin, self.real_hostpython_location)
+
env = self.get_recipe_env(arch, with_flags_in_cc=True)
# make build dir separately
sub_build_dir = join(build_dir, "p4a_android_build")
@@ -1423,7 +1471,7 @@ def get_recipe_env(self, arch, **kwargs):
env["PYO3_CROSS_LIB_DIR"] = realpath(glob.glob(join(
realpython_dir, "android-build", "build",
- "lib.linux-*-{}/".format(self.python_major_minor_version),
+ "lib.*{}/".format(self.python_major_minor_version),
))[0])
info_main("Ensuring rust build toolchain")
diff --git a/pythonforandroid/recipes/hostpython3/__init__.py b/pythonforandroid/recipes/hostpython3/__init__.py
index 9ba4580019..094660fada 100644
--- a/pythonforandroid/recipes/hostpython3/__init__.py
+++ b/pythonforandroid/recipes/hostpython3/__init__.py
@@ -5,6 +5,7 @@
from pathlib import Path
from os.path import join
+from packaging.version import Version
from pythonforandroid.logger import shprint
from pythonforandroid.recipe import Recipe
from pythonforandroid.util import (
@@ -35,18 +36,15 @@ class HostPython3Recipe(Recipe):
:class:`~pythonforandroid.python.HostPythonRecipe`
'''
- version = '3.11.5'
- name = 'hostpython3'
+ version = '3.11.13'
- build_subdir = 'native-build'
- '''Specify the sub build directory for the hostpython3 recipe. Defaults
- to ``native-build``.'''
-
- url = 'https://www.python.org/ftp/python/{version}/Python-{version}.tgz'
+ url = 'https://github.com/python/cpython/archive/refs/tags/v{version}.tar.gz'
'''The default url to download our host python recipe. This url will
change depending on the python version set in attribute :attr:`version`.'''
- patches = ['patches/pyconfig_detection.patch']
+ build_subdir = 'native-build'
+ '''Specify the sub build directory for the hostpython3 recipe. Defaults
+ to ``native-build``.'''
@property
def _exe_name(self):
@@ -95,6 +93,26 @@ def get_build_dir(self, arch=None):
def get_path_to_python(self):
return join(self.get_build_dir(), self.build_subdir)
+ @property
+ def site_root(self):
+ return join(self.get_path_to_python(), "root")
+
+ @property
+ def site_bin(self):
+ return join(self.site_root, self.site_dir, "bin")
+
+ @property
+ def local_bin(self):
+ return join(self.site_root, "usr/local/bin/")
+
+ @property
+ def site_dir(self):
+ p_version = Version(self.version)
+ return join(
+ self.site_root,
+ f"usr/local/lib/python{p_version.major}.{p_version.minor}/site-packages/"
+ )
+
def build_arch(self, arch):
env = self.get_recipe_env(arch)
@@ -105,9 +123,11 @@ def build_arch(self, arch):
ensure_dir(build_dir)
# Configure the build
+ build_configured = False
with current_directory(build_dir):
if not Path('config.status').exists():
shprint(sh.Command(join(recipe_build_dir, 'configure')), _env=env)
+ build_configured = True
with current_directory(recipe_build_dir):
# Create the Setup file. This copying from Setup.dist is
@@ -138,7 +158,13 @@ def build_arch(self, arch):
shprint(sh.cp, exe, self.python_exe)
break
+ ensure_dir(self.site_root)
self.ctx.hostpython = self.python_exe
+ if build_configured:
+ shprint(
+ sh.Command(self.python_exe), "-m", "ensurepip", "--root", self.site_root, "-U",
+ _env={"HOME": "/tmp"}
+ )
recipe = HostPython3Recipe()
diff --git a/pythonforandroid/recipes/hostpython3/patches/pyconfig_detection.patch b/pythonforandroid/recipes/hostpython3/patches/pyconfig_detection.patch
deleted file mode 100644
index 7f78b664e1..0000000000
--- a/pythonforandroid/recipes/hostpython3/patches/pyconfig_detection.patch
+++ /dev/null
@@ -1,13 +0,0 @@
-diff -Nru Python-3.8.2/Lib/site.py Python-3.8.2-new/Lib/site.py
---- Python-3.8.2/Lib/site.py 2020-04-28 12:48:38.000000000 -0700
-+++ Python-3.8.2-new/Lib/site.py 2020-04-28 12:52:46.000000000 -0700
-@@ -487,7 +487,8 @@
- if key == 'include-system-site-packages':
- system_site = value.lower()
- elif key == 'home':
-- sys._home = value
-+ # this is breaking pyconfig.h path detection with venv
-+ print('Ignoring "sys._home = value" override', file=sys.stderr)
-
- sys.prefix = sys.exec_prefix = site_prefix
-
diff --git a/pythonforandroid/recipes/numpy/__init__.py b/pythonforandroid/recipes/numpy/__init__.py
index f85e44d4f5..140ff849d8 100644
--- a/pythonforandroid/recipes/numpy/__init__.py
+++ b/pythonforandroid/recipes/numpy/__init__.py
@@ -40,10 +40,9 @@ def build_arch(self, arch):
super().build_arch(arch)
self.restore_hostpython_prerequisites(["cython"])
- def get_hostrecipe_env(self, arch):
- env = super().get_hostrecipe_env(arch)
+ def get_hostrecipe_env(self, arch=None):
+ env = super().get_hostrecipe_env(arch=arch)
env['RANLIB'] = shutil.which('ranlib')
- env["LDFLAGS"] += " -lm"
return env
diff --git a/pythonforandroid/recipes/openssl/__init__.py b/pythonforandroid/recipes/openssl/__init__.py
index 766c10e361..9a9a8c8a0f 100644
--- a/pythonforandroid/recipes/openssl/__init__.py
+++ b/pythonforandroid/recipes/openssl/__init__.py
@@ -1,4 +1,5 @@
from os.path import join
+from multiprocessing import cpu_count
from pythonforandroid.recipe import Recipe
from pythonforandroid.util import current_directory
@@ -44,35 +45,23 @@ class OpenSSLRecipe(Recipe):
'''
- version = '1.1'
- '''the major minor version used to link our recipes'''
-
- url_version = '1.1.1w'
- '''the version used to download our libraries'''
-
- url = 'https://www.openssl.org/source/openssl-{url_version}.tar.gz'
+ version = '3.3.1'
+ url = 'https://www.openssl.org/source/openssl-{version}.tar.gz'
built_libraries = {
- 'libcrypto{version}.so'.format(version=version): '.',
- 'libssl{version}.so'.format(version=version): '.',
+ 'libcrypto.so': '.',
+ 'libssl.so': '.',
}
- @property
- def versioned_url(self):
- if self.url is None:
- return None
- return self.url.format(url_version=self.url_version)
-
def get_build_dir(self, arch):
return join(
- self.get_build_container_dir(arch), self.name + self.version
+ self.get_build_container_dir(arch), self.name + self.version[0]
)
def include_flags(self, arch):
'''Returns a string with the include folders'''
openssl_includes = join(self.get_build_dir(arch.arch), 'include')
return (' -I' + openssl_includes +
- ' -I' + join(openssl_includes, 'internal') +
' -I' + join(openssl_includes, 'openssl'))
def link_dirs_flags(self, arch):
@@ -85,7 +74,7 @@ def link_libs_flags(self):
'''Returns a string with the appropriate `-l<lib>` flags to link with
the openssl libs. This string is usually added to the environment
variable `LIBS`'''
- return ' -lcrypto{version} -lssl{version}'.format(version=self.version)
+ return ' -lcrypto -lssl'
def link_flags(self, arch):
'''Returns a string with the flags to link with the openssl libraries
@@ -94,10 +83,12 @@ def link_flags(self, arch):
def get_recipe_env(self, arch=None):
env = super().get_recipe_env(arch)
- env['OPENSSL_VERSION'] = self.version
- env['MAKE'] = 'make' # This removes the '-j5', which isn't safe
+ env['OPENSSL_VERSION'] = self.version[0]
env['CC'] = 'clang'
- env['ANDROID_NDK_HOME'] = self.ctx.ndk_dir
+ env['ANDROID_NDK_ROOT'] = self.ctx.ndk_dir
+ env["PATH"] = f"{self.ctx.ndk.llvm_bin_dir}:{env['PATH']}"
+ env["CFLAGS"] += " -Wno-macro-redefined"
+ env["MAKE"] = "make"
return env
def select_build_arch(self, arch):
@@ -125,13 +116,12 @@ def build_arch(self, arch):
'shared',
'no-dso',
'no-asm',
+ 'no-tests',
buildarch,
'-D__ANDROID_API__={}'.format(self.ctx.ndk_api),
]
shprint(perl, 'Configure', *config_args, _env=env)
- self.apply_patch('disable-sover.patch', arch.arch)
-
- shprint(sh.make, 'build_libs', _env=env)
+ shprint(sh.make, '-j', str(cpu_count()), _env=env)
recipe = OpenSSLRecipe()
diff --git a/pythonforandroid/recipes/openssl/disable-sover.patch b/pythonforandroid/recipes/openssl/disable-sover.patch
deleted file mode 100644
index d944483cda..0000000000
--- a/pythonforandroid/recipes/openssl/disable-sover.patch
+++ /dev/null
@@ -1,11 +0,0 @@
---- openssl/Makefile.orig 2018-10-20 22:49:40.418310423 +0200
-+++ openssl/Makefile 2018-10-20 22:50:23.347322403 +0200
-@@ -19,7 +19,7 @@
- SHLIB_MAJOR=1
- SHLIB_MINOR=1
- SHLIB_TARGET=linux-shared
--SHLIB_EXT=.so.$(SHLIB_VERSION_NUMBER)
-+SHLIB_EXT=$(SHLIB_VERSION_NUMBER).so
- SHLIB_EXT_SIMPLE=.so
- SHLIB_EXT_IMPORT=
-
diff --git a/pythonforandroid/recipes/python3/__init__.py b/pythonforandroid/recipes/python3/__init__.py
index 7cd3928d76..81aee7c66e 100644
--- a/pythonforandroid/recipes/python3/__init__.py
+++ b/pythonforandroid/recipes/python3/__init__.py
@@ -3,12 +3,11 @@
import subprocess
from os import environ, utime
-from os.path import dirname, exists, join
-from pathlib import Path
+from os.path import dirname, exists, join, isfile
import shutil
-from pythonforandroid.logger import info, warning, shprint
-from pythonforandroid.patching import version_starts_with
+from packaging.version import Version
+from pythonforandroid.logger import info, shprint, warning
from pythonforandroid.recipe import Recipe, TargetPythonRecipe
from pythonforandroid.util import (
current_directory,
@@ -55,34 +54,37 @@ class Python3Recipe(TargetPythonRecipe):
:class:`~pythonforandroid.python.GuestPythonRecipe`
'''
- version = '3.11.5'
- url = 'https://www.python.org/ftp/python/{version}/Python-{version}.tgz'
+ version = '3.11.13'
+ _p_version = Version(version)
+ url = 'https://github.com/python/cpython/archive/refs/tags/v{version}.tar.gz'
name = 'python3'
patches = [
'patches/pyconfig_detection.patch',
'patches/reproducible-buildinfo.diff',
-
- # Python 3.7.1
- ('patches/py3.7.1_fix-ctypes-util-find-library.patch', version_starts_with("3.7")),
- ('patches/py3.7.1_fix-zlib-version.patch', version_starts_with("3.7")),
-
- # Python 3.8.1 & 3.9.X
- ('patches/py3.8.1.patch', version_starts_with("3.8")),
- ('patches/py3.8.1.patch', version_starts_with("3.9")),
- ('patches/py3.8.1.patch', version_starts_with("3.10")),
- ('patches/cpython-311-ctypes-find-library.patch', version_starts_with("3.11")),
]
- if shutil.which('lld') is not None:
+ if _p_version.major == 3 and _p_version.minor == 7:
patches += [
- ("patches/py3.7.1_fix_cortex_a8.patch", version_starts_with("3.7")),
- ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.8")),
- ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.9")),
- ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.10")),
- ("patches/py3.8.1_fix_cortex_a8.patch", version_starts_with("3.11")),
+ 'patches/py3.7.1_fix-ctypes-util-find-library.patch',
+ 'patches/py3.7.1_fix-zlib-version.patch',
]
+ if 8 <= _p_version.minor <= 10:
+ patches.append('patches/py3.8.1.patch')
+
+ if _p_version.minor >= 11:
+ patches.append('patches/cpython-311-ctypes-find-library.patch')
+
+ if _p_version.minor >= 14:
+ patches.append('patches/3.14_armv7l_fix.patch')
+
+ if shutil.which('lld') is not None:
+ if _p_version.minor == 7:
+ patches.append("patches/py3.7.1_fix_cortex_a8.patch")
+ elif _p_version.minor >= 8:
+ patches.append("patches/py3.8.1_fix_cortex_a8.patch")
+
depends = ['hostpython3', 'sqlite3', 'openssl', 'libffi']
# those optional depends allow us to build python compression modules:
# - _bz2.so
@@ -90,23 +92,33 @@ class Python3Recipe(TargetPythonRecipe):
opt_depends = ['libbz2', 'liblzma']
'''The optional libraries which we would like to get our python linked'''
- configure_args = (
+ configure_args = [
'--host={android_host}',
'--build={android_build}',
'--enable-shared',
'--enable-ipv6',
- 'ac_cv_file__dev_ptmx=yes',
- 'ac_cv_file__dev_ptc=no',
+ '--enable-loadable-sqlite-extensions',
+ '--without-static-libpython',
+ '--without-readline',
'--without-ensurepip',
- 'ac_cv_little_endian_double=yes',
- 'ac_cv_header_sys_eventfd_h=no',
+
+ # Android prefix
'--prefix={prefix}',
'--exec-prefix={exec_prefix}',
- '--enable-loadable-sqlite-extensions'
- )
+ '--enable-loadable-sqlite-extensions',
+
+ # Special cross compile args
+ 'ac_cv_file__dev_ptmx=yes',
+ 'ac_cv_file__dev_ptc=no',
+ 'ac_cv_header_sys_eventfd_h=no',
+ 'ac_cv_little_endian_double=yes',
+ 'ac_cv_header_bzlib_h=no',
+ ]
- if version_starts_with("3.11"):
- configure_args += ('--with-build-python={python_host_bin}',)
+ if _p_version.minor >= 11:
+ configure_args.extend([
+ '--with-build-python={python_host_bin}',
+ ])
'''The configure arguments needed to build the python recipe. Those are
used in method :meth:`build_arch` (if not overwritten like python3's
@@ -168,6 +180,9 @@ class Python3Recipe(TargetPythonRecipe):
longer used and has been removed in favour of extension .pyc
'''
+ disable_gil = False
+ '''python3.13 experimental free-threading build'''
+
def __init__(self, *args, **kwargs):
self._ctx = None
super().__init__(*args, **kwargs)
@@ -199,7 +214,7 @@ def link_root(self, arch_name):
return join(self.get_build_dir(arch_name), 'android-build')
def should_build(self, arch):
- return not Path(self.link_root(arch.arch), self._libpython).is_file()
+ return not isfile(join(self.link_root(arch.arch), self._libpython))
def prebuild_arch(self, arch):
super().prebuild_arch(arch)
@@ -244,30 +259,26 @@ def add_flags(include_flags, link_dirs, link_libs):
env['LDFLAGS'] = env.get('LDFLAGS', '') + link_dirs
env['LIBS'] = env.get('LIBS', '') + link_libs
- if 'sqlite3' in self.ctx.recipe_build_order:
- info('Activating flags for sqlite3')
- recipe = Recipe.get_recipe('sqlite3', self.ctx)
- add_flags(' -I' + recipe.get_build_dir(arch.arch),
- ' -L' + recipe.get_lib_dir(arch), ' -lsqlite3')
-
- if 'libffi' in self.ctx.recipe_build_order:
- info('Activating flags for libffi')
- recipe = Recipe.get_recipe('libffi', self.ctx)
- # In order to force the correct linkage for our libffi library, we
- # set the following variable to point where is our libffi.pc file,
- # because the python build system uses pkg-config to configure it.
- env['PKG_CONFIG_PATH'] = recipe.get_build_dir(arch.arch)
- add_flags(' -I' + ' -I'.join(recipe.get_include_dirs(arch)),
- ' -L' + join(recipe.get_build_dir(arch.arch), '.libs'),
- ' -lffi')
-
- if 'openssl' in self.ctx.recipe_build_order:
- info('Activating flags for openssl')
- recipe = Recipe.get_recipe('openssl', self.ctx)
- self.configure_args += \
- ('--with-openssl=' + recipe.get_build_dir(arch.arch),)
- add_flags(recipe.include_flags(arch),
- recipe.link_dirs_flags(arch), recipe.link_libs_flags())
+ info('Activating flags for sqlite3')
+ recipe = Recipe.get_recipe('sqlite3', self.ctx)
+ add_flags(' -I' + recipe.get_build_dir(arch.arch),
+ ' -L' + recipe.get_lib_dir(arch), ' -lsqlite3')
+
+ info('Activating flags for libffi')
+ recipe = Recipe.get_recipe('libffi', self.ctx)
+ # In order to force the correct linkage for our libffi library, we
+ # set the following variable to point where is our libffi.pc file,
+ # because the python build system uses pkg-config to configure it.
+ env['PKG_CONFIG_LIBDIR'] = recipe.get_build_dir(arch.arch)
+ add_flags(' -I' + ' -I'.join(recipe.get_include_dirs(arch)),
+ ' -L' + join(recipe.get_build_dir(arch.arch), '.libs'),
+ ' -lffi')
+
+ info('Activating flags for openssl')
+ recipe = Recipe.get_recipe('openssl', self.ctx)
+ self.configure_args.append('--with-openssl=' + recipe.get_build_dir(arch.arch))
+ add_flags(recipe.include_flags(arch),
+ recipe.link_dirs_flags(arch), recipe.link_libs_flags())
for library_name in {'libbz2', 'liblzma'}:
if library_name in self.ctx.recipe_build_order:
@@ -303,6 +314,9 @@ def add_flags(include_flags, link_dirs, link_libs):
env['ZLIB_VERSION'] = line.replace('#define ZLIB_VERSION ', '')
add_flags(' -I' + zlib_includes, ' -L' + zlib_lib_path, ' -lz')
+ if self._p_version.minor >= 13 and self.disable_gil:
+ self.configure_args.append("--disable-gil")
+
return env
def build_arch(self, arch):
@@ -344,9 +358,6 @@ def build_arch(self, arch):
exec_prefix=sys_exec_prefix)).split(' '),
_env=env)
- # Python build does not seem to play well with make -j option from Python 3.11 and onwards
- # Before losing some time, please check issue
- # https://github.com/python/cpython/issues/101295 , as the root cause looks similar
shprint(
sh.make,
'all',
@@ -382,11 +393,13 @@ def create_python_bundle(self, dirn, arch):
self.get_build_dir(arch.arch),
'android-build',
'build',
- 'lib.linux{}-{}-{}'.format(
+ 'lib.{}{}-{}-{}'.format(
+ # android is now supported platform
+ "android" if self._p_version.minor >= 13 else "linux",
'2' if self.version[0] == '2' else '',
arch.command_prefix.split('-')[0],
self.major_minor_version_string
- ))
+ ))
# Compile to *.pyc the python modules
self.compile_python_files(modules_build_dir)
diff --git a/pythonforandroid/recipes/python3/patches/3.14_armv7l_fix.patch b/pythonforandroid/recipes/python3/patches/3.14_armv7l_fix.patch
new file mode 100644
index 0000000000..7565489a28
--- /dev/null
+++ b/pythonforandroid/recipes/python3/patches/3.14_armv7l_fix.patch
@@ -0,0 +1,12 @@
+diff '--color=auto' -uNr cpython-3.14.0/Lib/sysconfig/__init__.py cpython-3.14.0.mod/Lib/sysconfig/__init__.py
+--- cpython-3.14.0/Lib/sysconfig/__init__.py 2025-10-07 21:45:41.236149298 +0530
++++ cpython-3.14.0.mod/Lib/sysconfig/__init__.py 2025-10-07 21:45:54.650245131 +0530
+@@ -702,7 +702,7 @@
+ "x86_64": "x86_64",
+ "i686": "x86",
+ "aarch64": "arm64_v8a",
+- "armv7l": "armeabi_v7a",
++ "arm": "armeabi_v7a",
+ }[machine]
+ elif osname == "linux":
+ # At least on Linux/Intel, 'machine' is the processor --
diff --git a/pythonforandroid/recipes/setuptools/__init__.py b/pythonforandroid/recipes/setuptools/__init__.py
index 02b205023d..0c77b9fde1 100644
--- a/pythonforandroid/recipes/setuptools/__init__.py
+++ b/pythonforandroid/recipes/setuptools/__init__.py
@@ -1,11 +1,8 @@
-from pythonforandroid.recipe import PythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class SetuptoolsRecipe(PythonRecipe):
- version = '69.2.0'
- url = 'https://pypi.python.org/packages/source/s/setuptools/setuptools-{version}.tar.gz'
- call_hostpython_via_targetpython = False
- install_in_hostpython = True
+class SetuptoolsRecipe(PyProjectRecipe):
+ hostpython_prerequisites = ['setuptools']
recipe = SetuptoolsRecipe()
diff --git a/tests/recipes/test_openssl.py b/tests/recipes/test_openssl.py
index f7ed362f68..e0910f59f3 100644
--- a/tests/recipes/test_openssl.py
+++ b/tests/recipes/test_openssl.py
@@ -11,7 +11,6 @@ class TestOpensslRecipe(BaseTestForMakeRecipe, unittest.TestCase):
recipe_name = "openssl"
sh_command_calls = ["perl"]
- @mock.patch("pythonforandroid.recipes.openssl.sh.patch")
@mock.patch("pythonforandroid.util.chdir")
@mock.patch("pythonforandroid.build.ensure_dir")
@mock.patch("shutil.which")
@@ -20,31 +19,21 @@ def test_build_arch(
mock_shutil_which,
mock_ensure_dir,
mock_current_directory,
- mock_sh_patch,
):
# We overwrite the base test method because we need to mock a little
# more with this recipe.
super().test_build_arch()
- # make sure that the mocked methods are actually called
- mock_sh_patch.assert_called()
-
- def test_versioned_url(self):
- self.assertEqual(
- self.recipe.url.format(url_version=self.recipe.url_version),
- self.recipe.versioned_url,
- )
def test_include_flags(self):
inc = self.recipe.include_flags(self.arch)
build_dir = self.recipe.get_build_dir(self.arch)
- for i in {"include/internal", "include/openssl"}:
+ for i in {"include", "include/openssl"}:
self.assertIn(f"-I{build_dir}/{i}", inc)
def test_link_flags(self):
build_dir = self.recipe.get_build_dir(self.arch)
- openssl_version = self.recipe.version
self.assertEqual(
- f" -L{build_dir} -lcrypto{openssl_version} -lssl{openssl_version}",
+ f" -L{build_dir} -lcrypto -lssl",
self.recipe.link_flags(self.arch),
)
diff --git a/tests/recipes/test_python3.py b/tests/recipes/test_python3.py
index f1d652b6c1..01d58f7d27 100644
--- a/tests/recipes/test_python3.py
+++ b/tests/recipes/test_python3.py
@@ -26,14 +26,6 @@ def test_property__libpython(self):
f'libpython{self.recipe.link_version}.so'
)
- @mock.patch('pythonforandroid.recipes.python3.Path.is_file')
- def test_should_build(self, mock_is_file):
- # in case that python lib exists, we shouldn't trigger the build
- self.assertFalse(self.recipe.should_build(self.arch))
- # in case that python lib doesn't exist, we should trigger the build
- mock_is_file.return_value = False
- self.assertTrue(self.recipe.should_build(self.arch))
-
def test_include_root(self):
expected_include_dir = join(
self.recipe.get_build_dir(self.arch.arch), 'Include',
@@ -186,7 +178,8 @@ def test_create_python_bundle(
recipe_build_dir,
'android-build',
'build',
- 'lib.linux{}-{}-{}'.format(
+ 'lib.{}{}-{}-{}'.format(
+ 'android' if self.recipe.version[2] >= "3" else 'linux',
'2' if self.recipe.version[0] == '2' else '',
self.arch.command_prefix.split('-')[0],
self.recipe.major_minor_version_string
diff --git a/tests/recipes/test_reportlab.py b/tests/recipes/test_reportlab.py
index 6129a6a963..cde17fd532 100644
--- a/tests/recipes/test_reportlab.py
+++ b/tests/recipes/test_reportlab.py
@@ -32,6 +32,7 @@ def test_prebuild_arch(self):
patch('sh.patch'), \
patch('pythonforandroid.recipe.touch'), \
patch('sh.unzip'), \
+ patch('pythonforandroid.recipe.Recipe.is_patched', lambda *a: False), \
patch('os.path.isfile'):
self.recipe.prebuild_arch(self.arch)
# makes sure placeholder got replaced with library and include paths
From 2a4746d7918593c2da8a0a7f03c175652818f920 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Mathias=20Lindstr=C3=B6m?=
<8863149+kuzeyron@users.noreply.github.com>
Date: Thu, 9 Oct 2025 02:57:29 +0300
Subject: [PATCH 141/158] Commented out the os.environ so we are seeing less
spam (#3204)
---
.../bootstraps/common/build/jni/application/src/start.c | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
index 1862208943..a494c77dae 100644
--- a/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
+++ b/pythonforandroid/bootstraps/common/build/jni/application/src/start.c
@@ -281,8 +281,8 @@ int main(int argc, char *argv[]) {
" self.__buffer = lines[-1]\n"
"sys.stdout = sys.stderr = LogFile()\n"
"print('Android path', sys.path)\n"
- "import os\n"
- "print('os.environ is', os.environ)\n"
+ "# import os\n"
+ "# print('os.environ is', os.environ)\n"
"print('Android kivy bootstrap done. __name__ is', __name__)");
#if PY_MAJOR_VERSION < 3
From c63d8e2a91ce7180811a37eae489f772fb766a9d Mon Sep 17 00:00:00 2001
From: Kartavya Shukla <87070473+Novfensec@users.noreply.github.com>
Date: Thu, 9 Oct 2025 15:02:29 +0530
Subject: [PATCH 142/158] Fix broadcast receiver (#3168)
* Update broadcastReceiver
* Undo unnecessary changes.
* Update broadcast.py
---
.../recipes/android/src/android/broadcast.py | 33 ++++++++++++++++---
1 file changed, 29 insertions(+), 4 deletions(-)
diff --git a/pythonforandroid/recipes/android/src/android/broadcast.py b/pythonforandroid/recipes/android/src/android/broadcast.py
index 750e9c87ef..cc8e06aee4 100644
--- a/pythonforandroid/recipes/android/src/android/broadcast.py
+++ b/pythonforandroid/recipes/android/src/android/broadcast.py
@@ -1,9 +1,12 @@
# -------------------------------------------------------------------
# Broadcast receiver bridge
-
+import logging
from jnius import autoclass, PythonJavaClass, java_method
from android.config import JAVA_NAMESPACE, JNI_NAMESPACE, ACTIVITY_CLASS_NAME, SERVICE_CLASS_NAME
+logger = logging.getLogger("BroadcastReceiver")
+logger.setLevel(logging.DEBUG)
+
class BroadcastReceiver(object):
@@ -22,6 +25,7 @@ def onReceive(self, context, intent):
def __init__(self, callback, actions=None, categories=None):
super().__init__()
self.callback = callback
+ self._is_registered = False
if not actions and not categories:
raise Exception('You need to define at least actions or categories')
@@ -58,15 +62,36 @@ def _expand_partial_name(partial_name):
self.receiver_filter.addCategory(x)
def start(self):
- Handler = autoclass('android.os.Handler')
+
+ if hasattr(self, 'handlerthread') and self.handlerthread.isAlive():
+ logger.debug("HandlerThread already running, skipping start")
+ return
+
+ HandlerThread = autoclass('android.os.HandlerThread')
+ self.handlerthread = HandlerThread('handlerthread')
self.handlerthread.start()
+
+ if self._is_registered:
+ logger.info("Already registered.")
+ return
+
+ Handler = autoclass('android.os.Handler')
self.handler = Handler(self.handlerthread.getLooper())
self.context.registerReceiver(
self.receiver, self.receiver_filter, None, self.handler)
+ self._is_registered = True
def stop(self):
- self.context.unregisterReceiver(self.receiver)
- self.handlerthread.quit()
+ try:
+ self.context.unregisterReceiver(self.receiver)
+ self._is_registered = False
+ except Exception as e:
+ logger.error("unregisterReceiver failed: %s", e)
+
+ if hasattr(self, 'handlerthread'):
+ self.handlerthread.quitSafely()
+ self.handlerthread = None
+ self.handler = None
@property
def context(self):
From ba6fb7bad26e8f7cf778d978a57f306ea6b7855b Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 12 Oct 2025 06:23:35 +0530
Subject: [PATCH 143/158] `cffi`: fix build
---
pythonforandroid/recipe.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 8808ffc0d2..e93bc125e5 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -977,7 +977,7 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
self.patch_shebangs(self._host_recipe.local_bin, self.real_hostpython_location)
env["PATH"] = self._host_recipe.local_bin + ":" + self._host_recipe.site_bin + ":" + env["PATH"]
- host_env = self.get_hostrecipe_env()
+ host_env = self.get_hostrecipe_env(arch)
env['PYTHONPATH'] = host_env["PYTHONPATH"]
if not self.call_hostpython_via_targetpython:
From c32eda8baaba2d9528b9eea60a1ebb29ea5d9c23 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Mon, 14 Jul 2025 23:22:41 +0530
Subject: [PATCH 144/158] `recipes`: new pycairo recipe
---
pythonforandroid/recipe.py | 7 +-
pythonforandroid/recipes/libcairo/__init__.py | 85 +++++++++++++++++++
pythonforandroid/recipes/libcairo/meson.patch | 64 ++++++++++++++
pythonforandroid/recipes/pycairo/__init__.py | 26 ++++++
pythonforandroid/recipes/pycairo/meson.patch | 24 ++++++
setup.py | 2 +-
6 files changed, 206 insertions(+), 2 deletions(-)
create mode 100644 pythonforandroid/recipes/libcairo/__init__.py
create mode 100644 pythonforandroid/recipes/libcairo/meson.patch
create mode 100644 pythonforandroid/recipes/pycairo/__init__.py
create mode 100644 pythonforandroid/recipes/pycairo/meson.patch
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 8808ffc0d2..f9a6793c6a 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1347,6 +1347,10 @@ class MesonRecipe(PyProjectRecipe):
meson_version = "1.4.0"
ninja_version = "1.11.1.1"
+ skip_python = False
+ '''If true, skips all Python build and installation steps.
+ Useful for Meson projects written purely in C/C++ without Python bindings.'''
+
def sanitize_flags(self, *flag_strings):
return " ".join(flag_strings).strip().split(" ")
@@ -1428,7 +1432,8 @@ def build_arch(self, arch):
]:
if dep not in self.hostpython_prerequisites:
self.hostpython_prerequisites.append(dep)
- super().build_arch(arch)
+ if not self.skip_python:
+ super().build_arch(arch)
class RustCompiledComponentsRecipe(PyProjectRecipe):
diff --git a/pythonforandroid/recipes/libcairo/__init__.py b/pythonforandroid/recipes/libcairo/__init__.py
new file mode 100644
index 0000000000..14ca37d531
--- /dev/null
+++ b/pythonforandroid/recipes/libcairo/__init__.py
@@ -0,0 +1,85 @@
+from pythonforandroid.recipe import Recipe, MesonRecipe
+from os.path import join, exists
+from pythonforandroid.util import ensure_dir, current_directory
+from pythonforandroid.logger import shprint
+from multiprocessing import cpu_count
+import sh
+
+
+class LibCairoRecipe(MesonRecipe):
+ name = 'libcairo'
+ version = '1.18.4'
+ url = 'https://gitlab.freedesktop.org/cairo/cairo/-/archive/{version}/cairo-{version}.tar.bz2'
+ skip_python = True
+ depends = ["png", "freetype"]
+ patches = ["meson.patch"]
+ built_libraries = {
+ 'libcairo.so': 'install/lib',
+ 'libpixman-1.so': 'install/lib',
+ 'libcairo-script-interpreter.so': 'install/lib'
+ }
+
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ cpufeatures = join(self.ctx.ndk.ndk_dir, "sources/android/cpufeatures")
+ lib_dir = join(cpufeatures, "obj", "local", arch.arch)
+ env["CFLAGS"] += f" -I{cpufeatures}"
+ env["LDFLAGS"] += f" -L{lib_dir} -lcpufeatures"
+ return env
+
+ def should_build(self, arch):
+ return Recipe.should_build(self, arch)
+
+ def build_arch(self, arch):
+ super().build_arch(arch)
+ build_dir = self.get_build_dir(arch.arch)
+ install_dir = join(build_dir, 'install')
+ ensure_dir(install_dir)
+ env = self.get_recipe_env(arch)
+
+ lib_dir = self.ctx.get_libs_dir(arch.arch)
+ png_include = self.get_recipe('png', self.ctx).get_build_dir(arch.arch)
+ freetype_inc = join(self.get_recipe('freetype', self.ctx).get_build_dir(arch), "include")
+
+ with current_directory(build_dir):
+
+ cpufeatures_dir = join(self.ctx.ndk.ndk_dir, "sources/android/cpufeatures")
+ lib_file = join(cpufeatures_dir, "obj", "local", arch.arch, "libcpufeatures.a")
+
+ if not exists(lib_file):
+ shprint(
+ sh.Command(join(self.ctx.ndk_dir, "ndk-build")),
+ f"NDK_PROJECT_PATH={cpufeatures_dir}",
+ f"APP_BUILD_SCRIPT={cpufeatures_dir}/Android.mk",
+ f"APP_ABI={arch.arch}",
+ "APP_PLATFORM=latest",
+ _env=env
+ )
+
+ shprint(sh.meson, 'setup', 'builddir',
+ '--cross-file', join("/tmp", "android.meson.cross"),
+ f'--prefix={install_dir}',
+ '-Dpng=enabled',
+ '-Dzlib=enabled',
+ '-Dglib=disabled',
+ '-Dgtk_doc=false',
+ '-Dsymbol-lookup=disabled',
+
+ # deps
+ f'-Dpng_include_dir={png_include}',
+ f'-Dpng_lib_dir={lib_dir}',
+ f'-Dfreetype_include_dir={freetype_inc}',
+ f'-Dfreetype_lib_dir={lib_dir}',
+ _env=env)
+
+ shprint(sh.ninja, '-C', 'builddir', '-j', str(cpu_count()), _env=env)
+ # macOS fix: sometimes Ninja creates a dummy 'lib' file instead of a directory.
+ # So we remove and recreate the install directory using shell commands,
+ # since os.remove/os.makedirs behave inconsistently in this build env.
+ shprint(sh.rm, '-rf', install_dir)
+ shprint(sh.mkdir, install_dir)
+
+ shprint(sh.ninja, '-C', 'builddir', 'install', _env=env)
+
+
+recipe = LibCairoRecipe()
diff --git a/pythonforandroid/recipes/libcairo/meson.patch b/pythonforandroid/recipes/libcairo/meson.patch
new file mode 100644
index 0000000000..0e119c5cc2
--- /dev/null
+++ b/pythonforandroid/recipes/libcairo/meson.patch
@@ -0,0 +1,64 @@
+diff '--color=auto' -uNr cairo-1.18.4/meson.build cairo-1.18.4.mod/meson.build
+--- cairo-1.18.4/meson.build 2025-03-08 18:53:25.000000000 +0530
++++ cairo-1.18.4.mod/meson.build 2025-07-14 20:42:56.226164648 +0530
+@@ -235,11 +235,13 @@
+ conf.set('HAVE_ZLIB', 1)
+ endif
+
+-png_dep = dependency('libpng',
+- required: get_option('png'),
+- version: libpng_required_version,
+- fallback: ['libpng', 'libpng_dep']
++png_inc = include_directories(get_option('png_include_dir'))
++png_lib = cc.find_library('png16', dirs: [get_option('png_lib_dir')], required: true)
++png_dep = declare_dependency(
++ include_directories: png_inc,
++ dependencies: [png_lib]
+ )
++
+ if png_dep.found()
+ feature_conf.set('CAIRO_HAS_SVG_SURFACE', 1)
+ feature_conf.set('CAIRO_HAS_PNG_FUNCTIONS', 1)
+@@ -265,7 +267,7 @@
+
+ # Disable fontconfig by default on platforms where it is optional
+ fontconfig_option = get_option('fontconfig')
+-fontconfig_required = host_machine.system() not in ['windows', 'darwin']
++fontconfig_required = false
+ fontconfig_option = fontconfig_option.disable_auto_if(not fontconfig_required)
+
+ fontconfig_dep = dependency('fontconfig',
+@@ -304,11 +306,14 @@
+ freetype_required = host_machine.system() not in ['windows', 'darwin']
+ freetype_option = freetype_option.disable_auto_if(not freetype_required)
+
+-freetype_dep = dependency('freetype2',
+- required: freetype_option,
+- version: freetype_required_version,
+- fallback: ['freetype2', 'freetype_dep'],
++freetype_inc = include_directories(get_option('freetype_include_dir'))
++freetype_lib = cc.find_library('freetype', dirs: [get_option('freetype_lib_dir')], required: true)
++
++freetype_dep = declare_dependency(
++ include_directories: freetype_inc,
++ dependencies: [freetype_lib]
+ )
++
+ if freetype_dep.found()
+ feature_conf.set('CAIRO_HAS_FT_FONT', 1)
+ built_features += [{
+diff '--color=auto' -uNr cairo-1.18.4/meson.options cairo-1.18.4.mod/meson.options
+--- cairo-1.18.4/meson.options 2025-03-08 18:53:25.000000000 +0530
++++ cairo-1.18.4.mod/meson.options 2025-07-14 20:43:00.473191452 +0530
+@@ -28,3 +28,11 @@
+ # Documentation
+ option('gtk_doc', type : 'boolean', value : false,
+ description: 'Build the Cairo API reference (depends on gtk-doc)')
++
++# Deps
++
++option('png_include_dir', type: 'string', value: '', description: 'Path to PNG headers')
++option('png_lib_dir', type: 'string', value: '', description: 'Path to PNG library')
++option('freetype_include_dir', type: 'string', value: '', description: 'Path to FreeType headers')
++option('freetype_lib_dir', type: 'string', value: '', description: 'Path to FreeType library')
++
diff --git a/pythonforandroid/recipes/pycairo/__init__.py b/pythonforandroid/recipes/pycairo/__init__.py
new file mode 100644
index 0000000000..2e9474568b
--- /dev/null
+++ b/pythonforandroid/recipes/pycairo/__init__.py
@@ -0,0 +1,26 @@
+from pythonforandroid.recipe import MesonRecipe
+from os.path import join
+
+
+class PyCairoRecipe(MesonRecipe):
+ version = '1.28.0'
+ url = 'https://github.com/pygobject/pycairo/releases/download/v{version}/pycairo-{version}.tar.gz'
+ name = 'pycairo'
+ site_packages_name = 'cairo'
+ depends = ['libcairo']
+ patches = ["meson.patch"]
+
+ def build_arch(self, arch):
+
+ include_path = join(self.get_recipe('libcairo', self.ctx).get_build_dir(arch), "install", "include", "cairo")
+ lib_path = self.ctx.get_libs_dir(arch.arch)
+
+ self.extra_build_args += [
+ f'-Csetup-args=-Dcairo_include={include_path}',
+ f'-Csetup-args=-Dcairo_lib={lib_path}',
+ ]
+
+ super().build_arch(arch)
+
+
+recipe = PyCairoRecipe()
diff --git a/pythonforandroid/recipes/pycairo/meson.patch b/pythonforandroid/recipes/pycairo/meson.patch
new file mode 100644
index 0000000000..aabbf5ae3f
--- /dev/null
+++ b/pythonforandroid/recipes/pycairo/meson.patch
@@ -0,0 +1,24 @@
+diff '--color=auto' -uNr pycairo-1.28.0/cairo/meson.build pycairo-1.28.0.mod/cairo/meson.build
+--- pycairo-1.28.0/cairo/meson.build 2025-04-15 00:22:30.000000000 +0530
++++ pycairo-1.28.0.mod/cairo/meson.build 2025-07-14 21:56:34.782983845 +0530
+@@ -28,7 +28,10 @@
+ fs.copyfile(python_file, python_file)
+ endforeach
+
+-cairo_dep = dependency('cairo', version: cair_version_req, required: cc.get_id() != 'msvc')
++cairo_dep = declare_dependency(
++ include_directories: include_directories(get_option('cairo_include')),
++ link_args: ['-L' + get_option('cairo_lib'), '-lcairo']
++)
+
+ if cc.get_id() == 'msvc' and not cairo_dep.found()
+ if cc.has_header('cairo.h')
+diff '--color=auto' -uNr pycairo-1.28.0/meson_options.txt pycairo-1.28.0.mod/meson_options.txt
+--- pycairo-1.28.0/meson_options.txt 2025-04-15 00:22:30.000000000 +0530
++++ pycairo-1.28.0.mod/meson_options.txt 2025-07-14 21:56:52.824191314 +0530
+@@ -1,3 +1,5 @@
+ option('python', type : 'string', value : 'python3')
+ option('tests', type : 'boolean', value : true, description : 'build unit tests')
+ option('wheel', type : 'boolean', value : false, description : 'build for a Python wheel')
++option('cairo_include', type: 'string', value: '', description: 'Path to cairo headers')
++option('cairo_lib', type: 'string', value: '', description: 'Path to cairo libraries')
diff --git a/setup.py b/setup.py
index c9cebd2314..0bf5b124a5 100644
--- a/setup.py
+++ b/setup.py
@@ -21,7 +21,7 @@
# https://github.com/kivy/buildozer/issues/722
install_reqs = [
'appdirs', 'colorama>=0.3.3', 'jinja2',
- 'sh>=2, <3.0; sys_platform!="win32"',
+ 'sh>=2, <3.0; sys_platform!="win32"', 'meson', 'ninja',
'build', 'toml', 'packaging', 'setuptools', 'wheel~=0.43.0'
]
# (build and toml are used by pythonpackage.py)
From 691959c74ad47b2431f615ddd327a8439346db28 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Tue, 14 Oct 2025 19:22:37 +0530
Subject: [PATCH 145/158] `matplotlib`: fix build (#3230)
Thanks a lot!
---
pythonforandroid/archs.py | 2 +-
.../recipes/matplotlib/__init__.py | 94 +++----------------
.../matplotlib/libfreetype.pc.template | 10 --
.../recipes/matplotlib/meson.patch | 21 +++++
.../recipes/matplotlib/setup.cfg.template | 38 --------
.../recipes/matplotlib/skip_macos.patch | 12 ---
6 files changed, 37 insertions(+), 140 deletions(-)
delete mode 100644 pythonforandroid/recipes/matplotlib/libfreetype.pc.template
create mode 100644 pythonforandroid/recipes/matplotlib/meson.patch
delete mode 100644 pythonforandroid/recipes/matplotlib/setup.cfg.template
delete mode 100644 pythonforandroid/recipes/matplotlib/skip_macos.patch
diff --git a/pythonforandroid/archs.py b/pythonforandroid/archs.py
index c2c1f91fc7..d09cee0ba9 100644
--- a/pythonforandroid/archs.py
+++ b/pythonforandroid/archs.py
@@ -132,7 +132,7 @@ def get_env(self, with_flags_in_cc=True):
env['CPPFLAGS'] = ' '.join(self.common_cppflags).format(
ctx=self.ctx,
command_prefix=self.command_prefix,
- python_includes=join(self.ctx.python_recipe.get_build_dir(self.arch), 'Include')
+ python_includes=join(Recipe.get_recipe("python3", self.ctx).include_root(self.arch))
)
# LDFLAGS: Link the extra global link paths first before anything else
diff --git a/pythonforandroid/recipes/matplotlib/__init__.py b/pythonforandroid/recipes/matplotlib/__init__.py
index 6fffd45596..f3ac80bddc 100644
--- a/pythonforandroid/recipes/matplotlib/__init__.py
+++ b/pythonforandroid/recipes/matplotlib/__init__.py
@@ -1,93 +1,29 @@
-from pythonforandroid.recipe import PyProjectRecipe
-from pythonforandroid.util import ensure_dir
+from pythonforandroid.recipe import MesonRecipe
+from pythonforandroid.logger import shprint
from os.path import join
-import shutil
+import sh
-class MatplotlibRecipe(PyProjectRecipe):
- version = '3.8.4'
+class MatplotlibRecipe(MesonRecipe):
+ version = '3.10.7'
url = 'https://github.com/matplotlib/matplotlib/archive/v{version}.zip'
- patches = ["skip_macos.patch"]
- depends = ['kiwisolver', 'numpy', 'pillow', 'setuptools', 'freetype']
+ depends = ['kiwisolver', 'numpy', 'pillow']
python_depends = ['cycler', 'fonttools', 'packaging', 'pyparsing', 'python-dateutil']
+ hostpython_prerequisites = ["setuptools_scm>=7"]
+ patches = ["meson.patch"]
need_stl_shared = True
- def generate_libraries_pc_files(self, arch):
- """
- Create *.pc files for libraries that `matplotib` depends on.
-
- Because, for unix platforms, the mpl install script uses `pkg-config`
- to detect libraries installed in non standard locations (our case...
- well...we don't even install the libraries...so we must trick a little
- the mlp install).
- """
- pkg_config_path = self.get_recipe_env(arch)['PKG_CONFIG_PATH']
- ensure_dir(pkg_config_path)
-
- lib_to_pc_file = {
- # `pkg-config` search for version freetype2.pc, our current
- # version for freetype, but we have our recipe named without
- # the version...so we add it in here for our pc file
- 'freetype': 'freetype2.pc',
- }
-
- for lib_name in {'freetype'}:
- pc_template_file = join(
- self.get_recipe_dir(),
- f'lib{lib_name}.pc.template'
- )
- # read template file into buffer
- with open(pc_template_file) as template_file:
- text_buffer = template_file.read()
- # set the library absolute path and library version
- lib_recipe = self.get_recipe(lib_name, self.ctx)
- text_buffer = text_buffer.replace(
- 'path_to_built', lib_recipe.get_build_dir(arch.arch),
- )
- text_buffer = text_buffer.replace(
- 'library_version', lib_recipe.version,
- )
-
- # write the library pc file into our defined dir `PKG_CONFIG_PATH`
- pc_dest_file = join(pkg_config_path, lib_to_pc_file[lib_name])
- with open(pc_dest_file, 'w') as pc_file:
- pc_file.write(text_buffer)
-
- def prebuild_arch(self, arch):
- shutil.copyfile(
- join(self.get_recipe_dir(), "setup.cfg.template"),
- join(self.get_build_dir(arch), "mplsetup.cfg"),
- )
- self.generate_libraries_pc_files(arch)
-
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
-
- # we make use of the same directory than `XDG_CACHE_HOME`, for our
- # custom library pc files, so we have all the install files that we
- # generate at the same place
- env['XDG_CACHE_HOME'] = join(self.get_build_dir(arch), 'p4a_files')
- env['PKG_CONFIG_PATH'] = env['XDG_CACHE_HOME']
-
- # creating proper *.pc files for our libraries does not seem enough to
- # success with our build (without depending on system development
- # libraries), but if we tell the compiler where to find our libraries
- # and includes, then the install success :)
- freetype = self.get_recipe('freetype', self.ctx)
- free_lib_dir = join(freetype.get_build_dir(arch.arch), 'objs', '.libs')
- free_inc_dir = join(freetype.get_build_dir(arch.arch), 'include')
- env['CFLAGS'] += f' -I{free_inc_dir}'
- env['LDFLAGS'] += f' -L{free_lib_dir}'
-
- # `freetype` could be built with `harfbuzz` support,
- # so we also include the necessary flags...just to be sure
- if 'harfbuzz' in self.ctx.recipe_build_order:
- harfbuzz = self.get_recipe('harfbuzz', self.ctx)
- harf_build = harfbuzz.get_build_dir(arch.arch)
- env['CFLAGS'] += f' -I{harf_build} -I{join(harf_build, "src")}'
- env['LDFLAGS'] += f' -L{join(harf_build, "src", ".libs")}'
+ env['CXXFLAGS'] += ' -Wno-c++11-narrowing'
return env
+ def build_arch(self, arch):
+ python_path = join(self.ctx.python_recipe.get_build_dir(arch), "android-build", "python3")
+ self.extra_build_args += [f'-Csetup-args=-Dpython3_program={python_path}']
+ shprint(sh.cp, self.real_hostpython_location, python_path)
+ super().build_arch(arch)
+
recipe = MatplotlibRecipe()
diff --git a/pythonforandroid/recipes/matplotlib/libfreetype.pc.template b/pythonforandroid/recipes/matplotlib/libfreetype.pc.template
deleted file mode 100644
index df5ef288dc..0000000000
--- a/pythonforandroid/recipes/matplotlib/libfreetype.pc.template
+++ /dev/null
@@ -1,10 +0,0 @@
-prefix=path_to_built
-exec_prefix=${prefix}
-includedir=${prefix}/include
-libdir=${exec_prefix}/objs/.libs
-
-Name: freetype2
-Description: The freetype2 library
-Version: library_version
-Cflags: -I${includedir}
-Libs: -L${libdir} -lfreetype
diff --git a/pythonforandroid/recipes/matplotlib/meson.patch b/pythonforandroid/recipes/matplotlib/meson.patch
new file mode 100644
index 0000000000..babab09002
--- /dev/null
+++ b/pythonforandroid/recipes/matplotlib/meson.patch
@@ -0,0 +1,21 @@
+diff '--color=auto' -uNr matplotlib-3.10.7/meson.build matplotlib-3.10.7.mod/meson.build
+--- matplotlib-3.10.7/meson.build 2025-10-09 04:16:31.000000000 +0530
++++ matplotlib-3.10.7.mod/meson.build 2025-10-12 10:19:29.664280049 +0530
+@@ -36,7 +36,7 @@
+
+ # https://mesonbuild.com/Python-module.html
+ py_mod = import('python')
+-py3 = py_mod.find_installation(pure: false)
++py3 = py_mod.find_installation(get_option('python3_program'), pure: false)
+ py3_dep = py3.dependency()
+
+ pybind11_dep = dependency('pybind11', version: '>=2.13.2')
+diff '--color=auto' -uNr matplotlib-3.10.7/meson.options matplotlib-3.10.7.mod/meson.options
+--- matplotlib-3.10.7/meson.options 2025-10-09 04:16:31.000000000 +0530
++++ matplotlib-3.10.7.mod/meson.options 2025-10-12 10:19:23.762042521 +0530
+@@ -28,3 +28,5 @@
+ # default is determined by fallback.
+ option('rcParams-backend', type: 'string', value: 'auto',
+ description: 'Set default backend at runtime')
++
++option('python3_program', type : 'string', value : '', description : 'Path to Python 3 executable')
diff --git a/pythonforandroid/recipes/matplotlib/setup.cfg.template b/pythonforandroid/recipes/matplotlib/setup.cfg.template
deleted file mode 100644
index 96ef80d4d2..0000000000
--- a/pythonforandroid/recipes/matplotlib/setup.cfg.template
+++ /dev/null
@@ -1,38 +0,0 @@
-# Rename this file to mplsetup.cfg to modify Matplotlib's build options.
-
-[libs]
-# By default, Matplotlib builds with LTO, which may be slow if you re-compile
-# often, and don't need the space saving/speedup.
-enable_lto = False
-# By default, Matplotlib downloads and builds its own copies of FreeType and of
-# Qhull. You may set the following to True to instead link against a system
-# FreeType/Qhull. As an exception, Matplotlib defaults to the system version
-# of FreeType on AIX.
-system_freetype = True
-#system_qhull = False
-
-[packages]
-# There are a number of data subpackages from Matplotlib that are
-# considered optional. All except 'tests' data (meaning the baseline
-# image files) are installed by default, but that can be changed here.
-#tests = False
-
-[gui_support]
-# Matplotlib supports multiple GUI toolkits, known as backends.
-# The MacOSX backend requires the Cocoa headers included with XCode.
-# You can select whether to build it by uncommenting the following line.
-# It is never built on Linux or Windows, regardless of the config value.
-#
-macosx = False
-
-[rc_options]
-# User-configurable options
-#
-# Default backend, one of: Agg, Cairo, GTK3Agg, GTK3Cairo, GTK4Agg, GTK4Cairo,
-# MacOSX, Pdf, Ps, QtAgg, QtCairo, SVG, TkAgg, WX, WXAgg.
-#
-# The Agg, Ps, Pdf and SVG backends do not require external dependencies. Do
-# not choose MacOSX if you have disabled the relevant extension modules. The
-# default is determined by fallback.
-#
-#backend = Agg
\ No newline at end of file
diff --git a/pythonforandroid/recipes/matplotlib/skip_macos.patch b/pythonforandroid/recipes/matplotlib/skip_macos.patch
deleted file mode 100644
index 7652750769..0000000000
--- a/pythonforandroid/recipes/matplotlib/skip_macos.patch
+++ /dev/null
@@ -1,12 +0,0 @@
-diff '--color=auto' -uNr matplotlib-3.8.4/setupext.py matplotlib-3.8.4.mod/setupext.py
---- matplotlib-3.8.4/setupext.py 2024-04-04 04:06:51.000000000 +0530
-+++ matplotlib-3.8.4.mod/setupext.py 2024-04-30 19:31:39.608063438 +0530
-@@ -782,7 +782,7 @@
- name = 'macosx'
-
- def check(self):
-- if sys.platform != 'darwin':
-+ if True: #sys.platform != 'darwin':
- raise Skipped("Mac OS-X only")
- return super().check()
-
From af20c5f2408f4e0d6242476a37d43fc2ea68cdb6 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Thu, 27 Mar 2025 02:58:42 +0530
Subject: [PATCH 146/158] `scipy`: update to `v1.15.2`
---
ci/constants.py | 5 +-
pythonforandroid/recipe.py | 1 +
pythonforandroid/recipes/fortran/__init__.py | 188 +++
pythonforandroid/recipes/lapack/__init__.py | 80 --
.../recipes/libopenblas/__init__.py | 50 +
pythonforandroid/recipes/scipy/__init__.py | 137 +-
pythonforandroid/recipes/scipy/meson.patch | 44 +
pythonforandroid/recipes/scipy/setup.py.patch | 1098 -----------------
pythonforandroid/recipes/scipy/wrapper.py | 62 +
9 files changed, 407 insertions(+), 1258 deletions(-)
create mode 100644 pythonforandroid/recipes/fortran/__init__.py
delete mode 100644 pythonforandroid/recipes/lapack/__init__.py
create mode 100644 pythonforandroid/recipes/libopenblas/__init__.py
create mode 100644 pythonforandroid/recipes/scipy/meson.patch
delete mode 100644 pythonforandroid/recipes/scipy/setup.py.patch
create mode 100644 pythonforandroid/recipes/scipy/wrapper.py
diff --git a/ci/constants.py b/ci/constants.py
index 382a4a0bfe..357d74d539 100644
--- a/ci/constants.py
+++ b/ci/constants.py
@@ -25,8 +25,9 @@ class TargetPython(Enum):
# mpmath package with a version >= 0.19 required
'sympy',
'vlc',
- # need extra gfortran NDK system add-on
- 'lapack', 'scipy',
+ # GitHub CI runs out of storage while building it
+ 'scipy',
+ 'fortran',
# Outdated and there's a chance that is now useless.
'zope_interface',
# Requires zope_interface, which is broken.
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 8808ffc0d2..5080be9bf3 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1364,6 +1364,7 @@ def get_recipe_meson_options(self, arch):
"cpp_args": self.sanitize_flags(env["CXXFLAGS"], env["CPPFLAGS"]),
"c_link_args": self.sanitize_flags(env["LDFLAGS"]),
"cpp_link_args": self.sanitize_flags(env["LDFLAGS"]),
+ "fortran_link_args": self.sanitize_flags(env["LDFLAGS"]),
},
"properties": {
"needs_exe_wrapper": True,
diff --git a/pythonforandroid/recipes/fortran/__init__.py b/pythonforandroid/recipes/fortran/__init__.py
new file mode 100644
index 0000000000..62148671ec
--- /dev/null
+++ b/pythonforandroid/recipes/fortran/__init__.py
@@ -0,0 +1,188 @@
+import os
+import subprocess
+import shutil
+import sh
+from pathlib import Path
+from os.path import join
+from pythonforandroid.recipe import Recipe
+from pythonforandroid.recommendations import read_ndk_version
+from pythonforandroid.logger import info, shprint, info_main
+from pythonforandroid.util import ensure_dir
+import hashlib
+
+FLANG_FILES = {
+ "package-flang-aarch64.tar.bz2": "bf01399513e3b435224d9a9f656b72a0965a23fdd8c3c26af0f7c32f2a5f3403",
+ "package-flang-host.tar.bz2": "3ea2c0e8125ededddf9b3f23c767b8e37816e140ac934c76ace19a168fefdf83",
+ "package-flang-x86_64.tar.bz2": "afe7e391355c71e7b0c8ee71a3002e83e2e524ad61810238815facf3030be6e6",
+ "package-install.tar.bz2": "169b75f6125dc7b95e1d30416147a05d135da6cbe9cc8432d48f5b8633ac38db",
+}
+
+
+class GFortranRecipe(Recipe):
+ # flang support in NDK by @termux (on github)
+ name = "fortran"
+ toolchain_ver = 0
+ url = "https://github.com/termux/ndk-toolchain-clang-with-flang/releases/download/"
+
+ def match_sha256(self, file_path, expected_hash):
+ sha256 = hashlib.sha256()
+ with open(file_path, "rb") as f:
+ for chunk in iter(lambda: f.read(8192), b""):
+ sha256.update(chunk)
+ file_hash = sha256.hexdigest()
+ return file_hash == expected_hash
+
+ @property
+ def ndk_version(self):
+ ndk_version = read_ndk_version(self.ctx.ndk_dir)
+ minor_to_letter = {0: ""}
+ minor_to_letter.update(
+ {n + 1: chr(i) for n, i in enumerate(range(ord("b"), ord("b") + 25))}
+ )
+ return f"{ndk_version.major}{minor_to_letter[ndk_version.minor]}"
+
+ def get_cache_dir(self):
+ dir_name = self.get_dir_name()
+ return join(self.ctx.build_dir, "other_builds", dir_name)
+
+ def get_fortran_dir(self):
+ toolchain_name = f"android-r{self.ndk_version}-api-{self.ctx.ndk_api}"
+ return join(
+ self.get_cache_dir(), f"{toolchain_name}-flang-v{self.toolchain_ver}"
+ )
+
+ def get_incomplete_files(self):
+ incomplete_files = []
+ cache_dir = self.get_cache_dir()
+ for file, sha256sum in FLANG_FILES.items():
+ _file = join(cache_dir, file)
+ if not (os.path.exists(_file) and self.match_sha256(_file, sha256sum)):
+ incomplete_files.append(file)
+ return incomplete_files
+
+ def download_if_necessary(self):
+ assert self.ndk_version == "28c"
+ if len(self.get_incomplete_files()) == 0:
+ return
+ self.download()
+
+ def download(self):
+ cache_dir = self.get_cache_dir()
+ ensure_dir(cache_dir)
+ for file in self.get_incomplete_files():
+ _file = join(cache_dir, file)
+ if os.path.exists(_file):
+ os.remove(_file)
+ self.download_file(f"{self.url}r{join(self.ndk_version, file)}", _file)
+
+ def extract_tar(self, file_path: Path, dest: Path, strip=1):
+ shprint(
+ sh.tar,
+ "xf",
+ str(file_path),
+ "--strip-components",
+ str(strip),
+ "-C",
+ str(dest) if dest else ".",
+ )
+
+ def create_flang_wrapper(self, path: Path, target: str):
+ script = f"""#!/usr/bin/env bash
+if [ "$1" != "-cpp" ] && [ "$1" != "-fc1" ]; then
+ `dirname $0`/flang-new --target={target}{self.ctx.ndk_api} -D__ANDROID_API__={self.ctx.ndk_api} "$@"
+else
+ `dirname $0`/flang-new "$@"
+fi
+"""
+ path.write_text(script)
+ path.chmod(0o755)
+
+ def unpack(self, arch):
+ info_main("Unpacking fortran")
+
+ flang_folder = self.get_fortran_dir()
+ if os.path.exists(flang_folder):
+ info("{} is already unpacked, skipping".format(self.name))
+ return
+
+ toolchain_path = Path(
+ join(self.ctx.ndk_dir, "toolchains/llvm/prebuilt/linux-x86_64")
+ )
+ cache_dir = Path(os.path.abspath(self.get_cache_dir()))
+
+ # clean tmp folder
+ tmp_folder = Path(os.path.abspath(f"{flang_folder}-tmp"))
+ shutil.rmtree(tmp_folder, ignore_errors=True)
+ tmp_folder.mkdir(parents=True)
+ os.chdir(tmp_folder)
+
+ self.extract_tar(cache_dir / "package-install.tar.bz2", None, strip=4)
+ self.extract_tar(cache_dir / "package-flang-host.tar.bz2", None)
+
+ sysroot_path = tmp_folder / "sysroot"
+ shutil.copytree(toolchain_path / "sysroot", sysroot_path)
+
+ self.extract_tar(
+ cache_dir / "package-flang-aarch64.tar.bz2",
+ sysroot_path / "usr/lib/aarch64-linux-android",
+ )
+ self.extract_tar(
+ cache_dir / "package-flang-x86_64.tar.bz2",
+ sysroot_path / "usr/lib/x86_64-linux-android",
+ )
+
+ # Fix lib/clang paths
+ version_output = subprocess.check_output(
+ [str(tmp_folder / "bin/clang"), "--version"], text=True
+ )
+ clang_version = next(
+ (line for line in version_output.splitlines() if "clang version" in line),
+ "",
+ )
+ major_ver = clang_version.split("clang version ")[-1].split(".")[0]
+
+ lib_path = tmp_folder / f"lib/clang/{major_ver}/lib"
+ src_lib_path = toolchain_path / f"lib/clang/{major_ver}/lib"
+ shutil.rmtree(lib_path, ignore_errors=True)
+ lib_path.mkdir(parents=True)
+
+ for item in src_lib_path.iterdir():
+ shprint(sh.cp, "-r", str(item), str(lib_path))
+
+ # Create flang wrappers
+ targets = [
+ "aarch64-linux-android",
+ "armv7a-linux-androideabi",
+ "i686-linux-android",
+ "x86_64-linux-android",
+ ]
+
+ for target in targets:
+ wrapper_path = tmp_folder / f"bin/{target}-flang"
+ self.create_flang_wrapper(wrapper_path, target)
+ shutil.copy(
+ wrapper_path, tmp_folder / f"bin/{target}{self.ctx.ndk_api}-flang"
+ )
+
+ tmp_folder.rename(flang_folder)
+
+ @property
+ def bin_path(self):
+ return f"{self.get_fortran_dir()}/bin"
+
+ def get_host_platform(self, arch):
+ return {
+ "arm64-v8a": "aarch64-linux-android",
+ "armeabi-v7a": "armv7a-linux-androideabi",
+ "x86_64": "x86_64-linux-android",
+ "x86": "i686-linux-android",
+ }[arch]
+
+ def get_fortran_bin(self, arch):
+ return join(self.bin_path, f"{self.get_host_platform(arch)}-flang")
+
+ def get_fortran_flags(self, arch):
+ return f"--target={self.get_host_platform(arch)}{self.ctx.ndk_api} -D__ANDROID_API__={self.ctx.ndk_api}"
+
+
+recipe = GFortranRecipe()
diff --git a/pythonforandroid/recipes/lapack/__init__.py b/pythonforandroid/recipes/lapack/__init__.py
deleted file mode 100644
index b6124dc285..0000000000
--- a/pythonforandroid/recipes/lapack/__init__.py
+++ /dev/null
@@ -1,80 +0,0 @@
-'''
-known to build with cmake version 3.23.2 and NDK r21e.
-See https://gitlab.kitware.com/cmake/cmake/-/issues/18739
-'''
-
-from pythonforandroid.recipe import Recipe
-from pythonforandroid.logger import shprint
-from pythonforandroid.util import current_directory, ensure_dir, BuildInterruptingException
-from multiprocessing import cpu_count
-from os.path import join
-import sh
-import shutil
-from os import environ
-from pythonforandroid.util import build_platform, rmdir
-
-arch_to_sysroot = {'armeabi': 'arm', 'armeabi-v7a': 'arm', 'arm64-v8a': 'arm64'}
-
-
-def arch_to_toolchain(arch):
- if 'arm' in arch.arch:
- return arch.command_prefix
- return arch.arch
-
-
-class LapackRecipe(Recipe):
-
- name = 'lapack'
- version = 'v3.10.1'
- url = 'https://github.com/Reference-LAPACK/lapack/archive/{version}.tar.gz'
- libdir = 'build/install/lib'
- built_libraries = {'libblas.so': libdir, 'liblapack.so': libdir, 'libcblas.so': libdir}
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
-
- ndk_dir = environ.get("LEGACY_NDK")
- if ndk_dir is None:
- raise BuildInterruptingException("Please set the environment variable 'LEGACY_NDK' to point to a NDK location with gcc/gfortran support (supported NDK version: 'r21e')")
-
- GCC_VER = '4.9'
- HOST = build_platform
-
- sysroot_suffix = arch_to_sysroot.get(arch.arch, arch.arch)
- sysroot = f"{ndk_dir}/platforms/{env['NDK_API']}/arch-{sysroot_suffix}"
- FC = f"{ndk_dir}/toolchains/{arch_to_toolchain(arch)}-{GCC_VER}/prebuilt/{HOST}/bin/{arch.command_prefix}-gfortran"
- env['FC'] = f'{FC} --sysroot={sysroot}'
- if shutil.which(FC) is None:
- raise BuildInterruptingException(f"{FC} not found. See https://github.com/mzakharo/android-gfortran")
- return env
-
- def build_arch(self, arch):
- source_dir = self.get_build_dir(arch.arch)
- build_target = join(source_dir, 'build')
- install_target = join(build_target, 'install')
-
- ensure_dir(build_target)
- with current_directory(build_target):
- env = self.get_recipe_env(arch)
- ndk_dir = environ["LEGACY_NDK"]
- rmdir('CMakeFiles')
- shprint(sh.rm, '-f', 'CMakeCache.txt', _env=env)
- opts = [
- '-DCMAKE_SYSTEM_NAME=Android',
- '-DCMAKE_POSITION_INDEPENDENT_CODE=1',
- '-DCMAKE_ANDROID_ARCH_ABI={arch}'.format(arch=arch.arch),
- '-DCMAKE_ANDROID_NDK=' + ndk_dir,
- '-DCMAKE_ANDROID_API={api}'.format(api=self.ctx.ndk_api),
- '-DCMAKE_BUILD_TYPE=Release',
- '-DCMAKE_INSTALL_PREFIX={}'.format(install_target),
- '-DCBLAS=ON',
- '-DBUILD_SHARED_LIBS=ON',
- ]
- if arch.arch == 'armeabi-v7a':
- opts.append('-DCMAKE_ANDROID_ARM_NEON=ON')
- shprint(sh.cmake, source_dir, *opts, _env=env)
- shprint(sh.make, '-j' + str(cpu_count()), _env=env)
- shprint(sh.make, 'install', _env=env)
-
-
-recipe = LapackRecipe()
diff --git a/pythonforandroid/recipes/libopenblas/__init__.py b/pythonforandroid/recipes/libopenblas/__init__.py
new file mode 100644
index 0000000000..e88feb6686
--- /dev/null
+++ b/pythonforandroid/recipes/libopenblas/__init__.py
@@ -0,0 +1,50 @@
+from pythonforandroid.recipe import Recipe
+from pythonforandroid.logger import shprint
+from pythonforandroid.util import current_directory, ensure_dir
+from multiprocessing import cpu_count
+from os.path import join
+import sh
+from pythonforandroid.util import rmdir
+
+
+class LibOpenBlasRecipe(Recipe):
+
+ version = "0.3.29"
+ url = "https://github.com/OpenMathLib/OpenBLAS/archive/refs/tags/v{version}.tar.gz"
+ built_libraries = {"libopenblas.so": "build/lib"}
+ min_ndk_api_support = 24 # complex math functions support
+
+ def build_arch(self, arch):
+ source_dir = self.get_build_dir(arch.arch)
+ build_target = join(source_dir, "build")
+
+ ensure_dir(build_target)
+ with current_directory(build_target):
+ env = self.get_recipe_env(arch)
+ rmdir("CMakeFiles")
+ shprint(sh.rm, "-f", "CMakeCache.txt", _env=env)
+
+ opts = [
+ # default cmake options
+ "-DCMAKE_SYSTEM_NAME=Android",
+ "-DCMAKE_ANDROID_ARCH_ABI={arch}".format(arch=arch.arch),
+ "-DCMAKE_ANDROID_NDK=" + self.ctx.ndk_dir,
+ "-DCMAKE_ANDROID_API={api}".format(api=self.ctx.ndk_api),
+ "-DCMAKE_BUILD_TYPE=Release",
+ "-DBUILD_SHARED_LIBS=ON",
+ "-DC_LAPACK=ON",
+ "-DTARGET={target}".format(
+ target={
+ "arm64-v8a": "ARMV8",
+ "armeabi-v7a": "ARMV7",
+ "x86_64": "CORE2",
+ "x86": "CORE2",
+ }[arch.arch]
+ ),
+ ]
+
+ shprint(sh.cmake, source_dir, *opts, _env=env)
+ shprint(sh.make, "-j" + str(cpu_count()), _env=env)
+
+
+recipe = LibOpenBlasRecipe()
diff --git a/pythonforandroid/recipes/scipy/__init__.py b/pythonforandroid/recipes/scipy/__init__.py
index 242ca04234..8cf46dab11 100644
--- a/pythonforandroid/recipes/scipy/__init__.py
+++ b/pythonforandroid/recipes/scipy/__init__.py
@@ -1,92 +1,73 @@
-from multiprocessing import cpu_count
-from os.path import join
-from os import environ
-import sh
-from pythonforandroid.logger import shprint
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe, Recipe
-from pythonforandroid.util import build_platform, current_directory
+import os
+from os.path import join, dirname, basename
+from pythonforandroid.recipe import MesonRecipe, Recipe
+from pythonforandroid.logger import warning
+from pathlib import Path
-def arch_to_toolchain(arch):
- if 'arm' in arch.arch:
- return arch.command_prefix
- return arch.arch
+class ScipyRecipe(MesonRecipe):
-
-class ScipyRecipe(CompiledComponentsPythonRecipe):
-
- version = 'maintenance/1.11.x'
- url = 'git+https://github.com/scipy/scipy.git'
- git_commit = 'b430bf54b5064465983813e2cfef3fcb86c3df07' # version 1.11.3
- site_packages_name = 'scipy'
- hostpython_prerequisites = ['numpy']
- depends = ['setuptools', 'cython', 'numpy', 'lapack', 'pybind11']
- call_hostpython_via_targetpython = False
+ version = "v1.16.2"
+ url = "git+https://github.com/scipy/scipy.git"
+ depends = ["numpy", "libopenblas", "fortran"]
need_stl_shared = True
- patches = ["setup.py.patch"]
-
- def build_compiled_components(self, arch):
- self.setup_extra_args = ['-j', str(cpu_count())]
- super().build_compiled_components(arch)
- self.setup_extra_args = []
-
- def rebuild_compiled_components(self, arch, env):
- self.setup_extra_args = ['-j', str(cpu_count())]
- super().rebuild_compiled_components(arch, env)
- self.setup_extra_args = []
-
- def download_file(self, url, target, cwd=None):
- super().download_file(url, target, cwd=cwd)
- with current_directory(target):
- shprint(sh.git, 'fetch', '--unshallow')
- shprint(sh.git, 'checkout', self.git_commit)
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ meson_version = "1.5.0"
+ hostpython_prerequisites = ["numpy", "Cython>=3.0.8"]
+ patches = ["meson.patch"]
+
+ def get_recipe_meson_options(self, arch):
+ options = super().get_recipe_meson_options(arch)
+ options["binaries"]["python"] = self.ctx.python_recipe.python_exe
+ options["binaries"]["fortran"] = self.place_wrapper(arch)
+ options["properties"]["numpy-include-dir"] = join(
+ self.ctx.get_python_install_dir(arch.arch), "numpy/_core/include"
+ )
+ self.ensure_args(
+ "-Csetup-args=-Dblas=openblas",
+ "-Csetup-args=-Dlapack=openblas",
+ f"-Csetup-args=-Dopenblas_libdir={self.ctx.get_libs_dir(arch.arch)}",
+ f'-Csetup-args=-Dopenblas_incldir={join(Recipe.get_recipe("libopenblas", self.ctx).get_build_dir(arch.arch), "build")}',
+ "-Csetup-args=-Duse-pythran=false",
+ )
+ return options
+
+ def place_wrapper(self, arch):
+ compiler = Recipe.get_recipe("fortran", self.ctx).get_fortran_bin(arch.arch)
+ file = join(self.get_recipe_dir(), "wrapper.py")
+ with open(file, "r") as _file:
+ data = _file.read()
+ _file.close()
+ data = data.replace("@COMPILER@", compiler)
+ # custom compiler
+ # taken from: https://github.com/termux/termux-packages/blob/master/packages/python-scipy/
+ m_compiler = Path(join(dirname(compiler), basename(compiler) + "-scipy"))
+ m_compiler.write_text(data)
+ m_compiler.chmod(0o755)
+ self.patch_shebang(str(m_compiler), self.real_hostpython_location)
+ return str(m_compiler)
+
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
arch_env = arch.get_env()
-
- env['LDFLAGS'] = arch_env['LDFLAGS']
- env['LDFLAGS'] += ' -L{} -lpython{}'.format(
+ env["LDFLAGS"] = arch_env["LDFLAGS"]
+ env["LDFLAGS"] += " -L{} -lpython{}".format(
self.ctx.python_recipe.link_root(arch.arch),
self.ctx.python_recipe.link_version,
)
+ return env
- ndk_dir = environ["LEGACY_NDK"]
- GCC_VER = '4.9'
- HOST = build_platform
- suffix = '64' if '64' in arch.arch else ''
-
- prefix = arch.command_prefix
- CLANG_BIN = f'{ndk_dir}/toolchains/llvm/prebuilt/{HOST}/bin/'
- GCC = f'{ndk_dir}/toolchains/{arch_to_toolchain(arch)}-{GCC_VER}/prebuilt/{HOST}'
- libgfortran = f'{GCC}/{prefix}/lib{suffix}'
- numpylib = self.ctx.get_python_install_dir(arch.arch) + '/numpy'
- arch_cflags = ' '.join(arch.arch_cflags)
- LDSHARED_opts = f'-target {arch.target} {arch_cflags} ' + ' '.join(arch.common_ldshared)
-
- # TODO: add pythran support
- env['SCIPY_USE_PYTHRAN'] = '0'
-
- lapack_dir = join(Recipe.get_recipe('lapack', self.ctx).get_build_dir(arch.arch), 'build', 'install')
- env['LAPACK'] = f'{lapack_dir}/lib'
- env['BLAS'] = env['LAPACK']
-
- # compilers
- env['F77'] = f'{GCC}/bin/{prefix}-gfortran'
- env['F90'] = f'{GCC}/bin/{prefix}-gfortran'
- env['CC'] = f'{CLANG_BIN}clang -target {arch.target} {arch_cflags}'
- env['CXX'] = f'{CLANG_BIN}clang++ -target {arch.target} {arch_cflags}'
-
- # scipy expects ldshared to be a single executable without options
- env['LDSHARED'] = f'{CLANG_BIN}/clang'
+ def build_arch(self, arch):
+ if arch.arch not in ["arm64-v8a", "x86_64"]:
+ warning(
+ "SciPy supports only 64-bit Android architectures: arm64-v8a and x86_64; skipping build."
+ )
+ return
- # erase the default NDK C++ include options
- env['CPPFLAGS'] = '-DANDROID'
+ if os.name != "posix":
+ warning("Building SciPy is only supported on Linux; skipping.")
+ return
- # configure linker
- env['LDFLAGS'] += f' {LDSHARED_opts} -L{libgfortran} -L{numpylib}/core/lib -L{numpylib}/random/lib'
- env['LDFLAGS'] += f' -l{self.stl_lib_name}'
- return env
+ super().build_arch(arch)
recipe = ScipyRecipe()
diff --git a/pythonforandroid/recipes/scipy/meson.patch b/pythonforandroid/recipes/scipy/meson.patch
new file mode 100644
index 0000000000..1fa91e9276
--- /dev/null
+++ b/pythonforandroid/recipes/scipy/meson.patch
@@ -0,0 +1,44 @@
+diff '--color=auto' -uNr scipy.git/meson.options scipy.git.patch/meson.options
+--- scipy.git/meson.options 2025-03-27 02:55:14.586853766 +0530
++++ scipy.git.patch/meson.options 2025-03-27 02:07:29.736674085 +0530
+@@ -2,6 +2,8 @@
+ description: 'option for BLAS library switching')
+ option('lapack', type: 'string', value: 'openblas',
+ description: 'option for LAPACK library switching')
++option('openblas_incldir', type: 'string', value: '', description: 'OpenBLAS include directory')
++option('openblas_libdir', type: 'string', value: '', description: 'OpenBLAS library directory')
+ option('use-g77-abi', type: 'boolean', value: false,
+ description: 'If set to true, forces using g77 compatibility wrappers ' +
+ 'for LAPACK functions. The default is to use gfortran ' +
+diff '--color=auto' -uNr scipy.git/scipy/meson.build scipy.git.patch/scipy/meson.build
+--- scipy.git/scipy/meson.build 2025-03-27 02:55:14.632428649 +0530
++++ scipy.git.patch/scipy/meson.build 2025-03-27 11:25:33.756445056 +0530
+@@ -268,10 +268,18 @@
+ endif
+ endif
+
++openblas_inc = get_option('openblas_incldir')
++openblas_lib = get_option('openblas_libdir')
++
++openblas_dep = declare_dependency(
++ include_directories: include_directories(openblas_inc),
++ link_args: ['-L' + openblas_lib, '-lopenblas']
++)
++
+ # pkg-config uses a lower-case name while CMake uses a capitalized name, so try
+ # that too to make the fallback detection with CMake work
+ if blas_name == 'openblas'
+- blas = dependency(['openblas', 'OpenBLAS'])
++ blas = openblas_dep
+ elif blas_name != 'scipy-openblas' # if so, we found it already
+ blas = dependency(blas_name)
+ endif
+@@ -295,7 +303,7 @@
+ # use that - no need to run the full detection twice.
+ lapack = blas
+ elif lapack_name == 'openblas'
+- lapack = dependency(['openblas', 'OpenBLAS'])
++ lapack = openblas_dep
+ else
+ lapack = dependency(lapack_name)
+ endif
diff --git a/pythonforandroid/recipes/scipy/setup.py.patch b/pythonforandroid/recipes/scipy/setup.py.patch
deleted file mode 100644
index 9fbc0ab5fb..0000000000
--- a/pythonforandroid/recipes/scipy/setup.py.patch
+++ /dev/null
@@ -1,1098 +0,0 @@
-diff '--color=auto' -uNr scipy/_setup.py scipy.mod/_setup.py
---- scipy/_setup.py 2023-10-30 19:20:36.545524745 +0530
-+++ scipy.mod/_setup.py 1970-01-01 05:30:00.000000000 +0530
-@@ -1,545 +0,0 @@
--#!/usr/bin/env python
--"""SciPy: Scientific Library for Python
--
--SciPy (pronounced "Sigh Pie") is open-source software for mathematics,
--science, and engineering. The SciPy library
--depends on NumPy, which provides convenient and fast N-dimensional
--array manipulation. The SciPy library is built to work with NumPy
--arrays, and provides many user-friendly and efficient numerical
--routines such as routines for numerical integration and optimization.
--Together, they run on all popular operating systems, are quick to
--install, and are free of charge. NumPy and SciPy are easy to use,
--but powerful enough to be depended upon by some of the world's
--leading scientists and engineers. If you need to manipulate
--numbers on a computer and display or publish the results,
--give SciPy a try!
--
--"""
--
--
--# IMPORTANT:
--#
--# THIS FILE IS INTENTIONALLY RENAMED FROM setup.py TO _setup.py
--# IT IS ONLY KEPT IN THE REPO BECAUSE conda-forge STILL NEEDS IT
--# FOR BUILDING SCIPY ON WINDOWS. IT SHOULD NOT BE USED BY ANYONE
--# ELSE. USE `pip install .` OR ANOTHER INSTALL COMMAND USING A
--# BUILD FRONTEND LIKE pip OR pypa/build TO INSTALL SCIPY FROM SOURCE.
--#
--# SEE http://scipy.github.io/devdocs/building/index.html FOR BUILD
--# INSTRUCTIONS.
--
--
--DOCLINES = (__doc__ or '').split("\n")
--
--import os
--import sys
--import subprocess
--import textwrap
--import warnings
--import sysconfig
--from tools.version_utils import write_version_py, get_version_info
--from tools.version_utils import IS_RELEASE_BRANCH
--import importlib
--
--
--if sys.version_info[:2] < (3, 9):
-- raise RuntimeError("Python version >= 3.9 required.")
--
--import builtins
--
--
--CLASSIFIERS = """\
--Development Status :: 5 - Production/Stable
--Intended Audience :: Science/Research
--Intended Audience :: Developers
--License :: OSI Approved :: BSD License
--Programming Language :: C
--Programming Language :: Python
--Programming Language :: Python :: 3
--Programming Language :: Python :: 3.9
--Programming Language :: Python :: 3.10
--Programming Language :: Python :: 3.11
--Topic :: Software Development :: Libraries
--Topic :: Scientific/Engineering
--Operating System :: Microsoft :: Windows
--Operating System :: POSIX :: Linux
--Operating System :: POSIX
--Operating System :: Unix
--Operating System :: MacOS
--
--"""
--
--
--# BEFORE importing setuptools, remove MANIFEST. Otherwise it may not be
--# properly updated when the contents of directories change (true for distutils,
--# not sure about setuptools).
--if os.path.exists('MANIFEST'):
-- os.remove('MANIFEST')
--
--# This is a bit hackish: we are setting a global variable so that the main
--# scipy __init__ can detect if it is being loaded by the setup routine, to
--# avoid attempting to load components that aren't built yet. While ugly, it's
--# a lot more robust than what was previously being used.
--builtins.__SCIPY_SETUP__ = True
--
--
--def check_submodules():
-- """ verify that the submodules are checked out and clean
-- use `git submodule update --init`; on failure
-- """
-- if not os.path.exists('.git'):
-- return
-- with open('.gitmodules') as f:
-- for l in f:
-- if 'path' in l:
-- p = l.split('=')[-1].strip()
-- if not os.path.exists(p):
-- raise ValueError('Submodule %s missing' % p)
--
--
-- proc = subprocess.Popen(['git', 'submodule', 'status'],
-- stdout=subprocess.PIPE)
-- status, _ = proc.communicate()
-- status = status.decode("ascii", "replace")
-- for line in status.splitlines():
-- if line.startswith('-') or line.startswith('+'):
-- raise ValueError('Submodule not clean: %s' % line)
--
--
--class concat_license_files():
-- """Merge LICENSE.txt and LICENSES_bundled.txt for sdist creation
--
-- Done this way to keep LICENSE.txt in repo as exact BSD 3-clause (see
-- NumPy gh-13447). This makes GitHub state correctly how SciPy is licensed.
-- """
-- def __init__(self):
-- self.f1 = 'LICENSE.txt'
-- self.f2 = 'LICENSES_bundled.txt'
--
-- def __enter__(self):
-- """Concatenate files and remove LICENSES_bundled.txt"""
-- with open(self.f1, 'r') as f1:
-- self.bsd_text = f1.read()
--
-- with open(self.f1, 'a') as f1:
-- with open(self.f2, 'r') as f2:
-- self.bundled_text = f2.read()
-- f1.write('\n\n')
-- f1.write(self.bundled_text)
--
-- def __exit__(self, exception_type, exception_value, traceback):
-- """Restore content of both files"""
-- with open(self.f1, 'w') as f:
-- f.write(self.bsd_text)
--
--
--from distutils.command.sdist import sdist
--class sdist_checked(sdist):
-- """ check submodules on sdist to prevent incomplete tarballs """
-- def run(self):
-- check_submodules()
-- with concat_license_files():
-- sdist.run(self)
--
--
--def get_build_ext_override():
-- """
-- Custom build_ext command to tweak extension building.
-- """
-- from numpy.distutils.command.build_ext import build_ext as npy_build_ext
-- if int(os.environ.get('SCIPY_USE_PYTHRAN', 1)):
-- try:
-- import pythran
-- from pythran.dist import PythranBuildExt
-- except ImportError:
-- BaseBuildExt = npy_build_ext
-- else:
-- BaseBuildExt = PythranBuildExt[npy_build_ext]
-- _pep440 = importlib.import_module('scipy._lib._pep440')
-- if _pep440.parse(pythran.__version__) < _pep440.Version('0.11.0'):
-- raise RuntimeError("The installed `pythran` is too old, >= "
-- "0.11.0 is needed, {} detected. Please "
-- "upgrade Pythran, or use `export "
-- "SCIPY_USE_PYTHRAN=0`.".format(
-- pythran.__version__))
-- else:
-- BaseBuildExt = npy_build_ext
--
-- class build_ext(BaseBuildExt):
-- def finalize_options(self):
-- super().finalize_options()
--
-- # Disable distutils parallel build, due to race conditions
-- # in numpy.distutils (Numpy issue gh-15957)
-- if self.parallel:
-- print("NOTE: -j build option not supported. Set NPY_NUM_BUILD_JOBS=4 "
-- "for parallel build.")
-- self.parallel = None
--
-- def build_extension(self, ext):
-- # When compiling with GNU compilers, use a version script to
-- # hide symbols during linking.
-- if self.__is_using_gnu_linker(ext):
-- export_symbols = self.get_export_symbols(ext)
-- text = '{global: %s; local: *; };' % (';'.join(export_symbols),)
--
-- script_fn = os.path.join(self.build_temp, 'link-version-{}.map'.format(ext.name))
-- with open(script_fn, 'w') as f:
-- f.write(text)
-- # line below fixes gh-8680
-- ext.extra_link_args = [arg for arg in ext.extra_link_args if not "version-script" in arg]
-- ext.extra_link_args.append('-Wl,--version-script=' + script_fn)
--
-- # Allow late configuration
-- hooks = getattr(ext, '_pre_build_hook', ())
-- _run_pre_build_hooks(hooks, (self, ext))
--
-- super().build_extension(ext)
--
-- def __is_using_gnu_linker(self, ext):
-- if not sys.platform.startswith('linux'):
-- return False
--
-- # Fortran compilation with gfortran uses it also for
-- # linking. For the C compiler, we detect gcc in a similar
-- # way as distutils does it in
-- # UnixCCompiler.runtime_library_dir_option
-- if ext.language == 'f90':
-- is_gcc = (self._f90_compiler.compiler_type in ('gnu', 'gnu95'))
-- elif ext.language == 'f77':
-- is_gcc = (self._f77_compiler.compiler_type in ('gnu', 'gnu95'))
-- else:
-- is_gcc = False
-- if self.compiler.compiler_type == 'unix':
-- cc = sysconfig.get_config_var("CC")
-- if not cc:
-- cc = ""
-- compiler_name = os.path.basename(cc.split(" ")[0])
-- is_gcc = "gcc" in compiler_name or "g++" in compiler_name
-- return is_gcc and sysconfig.get_config_var('GNULD') == 'yes'
--
-- return build_ext
--
--
--def get_build_clib_override():
-- """
-- Custom build_clib command to tweak library building.
-- """
-- from numpy.distutils.command.build_clib import build_clib as old_build_clib
--
-- class build_clib(old_build_clib):
-- def finalize_options(self):
-- super().finalize_options()
--
-- # Disable parallelization (see build_ext above)
-- self.parallel = None
--
-- def build_a_library(self, build_info, lib_name, libraries):
-- # Allow late configuration
-- hooks = build_info.get('_pre_build_hook', ())
-- _run_pre_build_hooks(hooks, (self, build_info))
-- old_build_clib.build_a_library(self, build_info, lib_name, libraries)
--
-- return build_clib
--
--
--def _run_pre_build_hooks(hooks, args):
-- """Call a sequence of pre-build hooks, if any"""
-- if hooks is None:
-- hooks = ()
-- elif not hasattr(hooks, '__iter__'):
-- hooks = (hooks,)
-- for hook in hooks:
-- hook(*args)
--
--
--def generate_cython():
-- cwd = os.path.abspath(os.path.dirname(__file__))
-- print("Cythonizing sources")
-- p = subprocess.call([sys.executable,
-- os.path.join(cwd, 'tools', 'cythonize.py'),
-- 'scipy'],
-- cwd=cwd)
-- if p != 0:
-- # Could be due to a too old pip version and build isolation, check that
-- try:
-- # Note, pip may not be installed or not have been used
-- import pip
-- except (ImportError, ModuleNotFoundError):
-- raise RuntimeError("Running cythonize failed!")
-- else:
-- _pep440 = importlib.import_module('scipy._lib._pep440')
-- if _pep440.parse(pip.__version__) < _pep440.Version('18.0.0'):
-- raise RuntimeError("Cython not found or too old. Possibly due "
-- "to `pip` being too old, found version {}, "
-- "needed is >= 18.0.0.".format(
-- pip.__version__))
-- else:
-- raise RuntimeError("Running cythonize failed!")
--
--
--def parse_setuppy_commands():
-- """Check the commands and respond appropriately. Disable broken commands.
--
-- Return a boolean value for whether or not to run the build or not (avoid
-- parsing Cython and template files if False).
-- """
-- args = sys.argv[1:]
--
-- if not args:
-- # User forgot to give an argument probably, let setuptools handle that.
-- return True
--
-- info_commands = ['--help-commands', '--name', '--version', '-V',
-- '--fullname', '--author', '--author-email',
-- '--maintainer', '--maintainer-email', '--contact',
-- '--contact-email', '--url', '--license', '--description',
-- '--long-description', '--platforms', '--classifiers',
-- '--keywords', '--provides', '--requires', '--obsoletes']
--
-- for command in info_commands:
-- if command in args:
-- return False
--
-- # Note that 'alias', 'saveopts' and 'setopt' commands also seem to work
-- # fine as they are, but are usually used together with one of the commands
-- # below and not standalone. Hence they're not added to good_commands.
-- good_commands = ('develop', 'sdist', 'build', 'build_ext', 'build_py',
-- 'build_clib', 'build_scripts', 'bdist_wheel', 'bdist_rpm',
-- 'bdist_wininst', 'bdist_msi', 'bdist_mpkg')
--
-- for command in good_commands:
-- if command in args:
-- return True
--
-- # The following commands are supported, but we need to show more
-- # useful messages to the user
-- if 'install' in args:
-- print(textwrap.dedent("""
-- Note: for reliable uninstall behaviour and dependency installation
-- and uninstallation, please use pip instead of using
-- `setup.py install`:
--
-- - `pip install .` (from a git repo or downloaded source
-- release)
-- - `pip install scipy` (last SciPy release on PyPI)
--
-- """))
-- return True
--
-- if '--help' in args or '-h' in sys.argv[1]:
-- print(textwrap.dedent("""
-- SciPy-specific help
-- -------------------
--
-- To install SciPy from here with reliable uninstall, we recommend
-- that you use `pip install .`. To install the latest SciPy release
-- from PyPI, use `pip install scipy`.
--
-- For help with build/installation issues, please ask on the
-- scipy-user mailing list. If you are sure that you have run
-- into a bug, please report it at https://github.com/scipy/scipy/issues.
--
-- Setuptools commands help
-- ------------------------
-- """))
-- return False
--
--
-- # The following commands aren't supported. They can only be executed when
-- # the user explicitly adds a --force command-line argument.
-- bad_commands = dict(
-- test="""
-- `setup.py test` is not supported. Use one of the following
-- instead:
--
-- - `python runtests.py` (to build and test)
-- - `python runtests.py --no-build` (to test installed scipy)
-- - `>>> scipy.test()` (run tests for installed scipy
-- from within an interpreter)
-- """,
-- upload="""
-- `setup.py upload` is not supported, because it's insecure.
-- Instead, build what you want to upload and upload those files
-- with `twine upload -s <filenames>` instead.
-- """,
-- upload_docs="`setup.py upload_docs` is not supported",
-- easy_install="`setup.py easy_install` is not supported",
-- clean="""
-- `setup.py clean` is not supported, use one of the following instead:
--
-- - `git clean -xdf` (cleans all files)
-- - `git clean -Xdf` (cleans all versioned files, doesn't touch
-- files that aren't checked into the git repo)
-- """,
-- check="`setup.py check` is not supported",
-- register="`setup.py register` is not supported",
-- bdist_dumb="`setup.py bdist_dumb` is not supported",
-- bdist="`setup.py bdist` is not supported",
-- flake8="`setup.py flake8` is not supported, use flake8 standalone",
-- build_sphinx="`setup.py build_sphinx` is not supported, see doc/README.md",
-- )
-- bad_commands['nosetests'] = bad_commands['test']
-- for command in ('upload_docs', 'easy_install', 'bdist', 'bdist_dumb',
-- 'register', 'check', 'install_data', 'install_headers',
-- 'install_lib', 'install_scripts', ):
-- bad_commands[command] = "`setup.py %s` is not supported" % command
--
-- for command in bad_commands.keys():
-- if command in args:
-- print(textwrap.dedent(bad_commands[command]) +
-- "\nAdd `--force` to your command to use it anyway if you "
-- "must (unsupported).\n")
-- sys.exit(1)
--
-- # Commands that do more than print info, but also don't need Cython and
-- # template parsing.
-- other_commands = ['egg_info', 'install_egg_info', 'rotate']
-- for command in other_commands:
-- if command in args:
-- return False
--
-- # If we got here, we didn't detect what setup.py command was given
-- warnings.warn("Unrecognized setuptools command ('{}'), proceeding with "
-- "generating Cython sources and expanding templates".format(
-- ' '.join(sys.argv[1:])))
-- return True
--
--def check_setuppy_command():
-- run_build = parse_setuppy_commands()
-- if run_build:
-- try:
-- pkgname = 'numpy'
-- import numpy
-- pkgname = 'pybind11'
-- import pybind11
-- except ImportError as exc: # We do not have our build deps installed
-- print(textwrap.dedent(
-- """Error: '%s' must be installed before running the build.
-- """
-- % (pkgname,)))
-- sys.exit(1)
--
-- return run_build
--
--def configuration(parent_package='', top_path=None):
-- from numpy.distutils.system_info import get_info, NotFoundError
-- from numpy.distutils.misc_util import Configuration
--
-- lapack_opt = get_info('lapack_opt')
--
-- if not lapack_opt:
-- if sys.platform == "darwin":
-- msg = ('No BLAS/LAPACK libraries found. '
-- 'Note: Accelerate is no longer supported.')
-- else:
-- msg = 'No BLAS/LAPACK libraries found.'
-- msg += ("\n"
-- "To build Scipy from sources, BLAS & LAPACK libraries "
-- "need to be installed.\n"
-- "See site.cfg.example in the Scipy source directory and\n"
-- "https://docs.scipy.org/doc/scipy/dev/contributor/building.html "
-- "for details.")
-- raise NotFoundError(msg)
--
-- config = Configuration(None, parent_package, top_path)
-- config.set_options(ignore_setup_xxx_py=True,
-- assume_default_configuration=True,
-- delegate_options_to_subpackages=True,
-- quiet=True)
--
-- config.add_subpackage('scipy')
-- config.add_data_files(('scipy', '*.txt'))
--
-- config.get_version('scipy/version.py')
--
-- return config
--
--
--def setup_package():
-- # In maintenance branch, change np_maxversion to N+3 if numpy is at N
-- # Update here, in pyproject.toml, and in scipy/__init__.py
-- # Rationale: SciPy builds without deprecation warnings with N; deprecations
-- # in N+1 will turn into errors in N+3
-- # For Python versions, if releases is (e.g.) <=3.9.x, set bound to 3.10
-- np_minversion = '1.21.6'
-- np_maxversion = '1.28.0'
-- python_minversion = '3.9'
-- python_maxversion = '3.13'
-- if IS_RELEASE_BRANCH:
-- req_np = 'numpy>={},<{}'.format(np_minversion, np_maxversion)
-- req_py = '>={},<{}'.format(python_minversion, python_maxversion)
-- else:
-- req_np = 'numpy>={}'.format(np_minversion)
-- req_py = '>={}'.format(python_minversion)
--
-- # Rewrite the version file every time
-- write_version_py('.')
--
-- cmdclass = {'sdist': sdist_checked}
--
-- metadata = dict(
-- name='scipy',
-- maintainer="SciPy Developers",
-- maintainer_email="scipy-dev@python.org",
-- description=DOCLINES[0],
-- long_description="\n".join(DOCLINES[2:]),
-- url="https://www.scipy.org",
-- download_url="https://github.com/scipy/scipy/releases",
-- project_urls={
-- "Bug Tracker": "https://github.com/scipy/scipy/issues",
-- "Documentation": "https://docs.scipy.org/doc/scipy/reference/",
-- "Source Code": "https://github.com/scipy/scipy",
-- },
-- license='BSD',
-- cmdclass=cmdclass,
-- classifiers=[_f for _f in CLASSIFIERS.split('\n') if _f],
-- platforms=["Windows", "Linux", "Solaris", "Mac OS-X", "Unix"],
-- install_requires=[req_np],
-- python_requires=req_py,
-- zip_safe=False,
-- )
--
-- if "--force" in sys.argv:
-- run_build = True
-- sys.argv.remove('--force')
-- else:
-- # Raise errors for unsupported commands, improve help output, etc.
-- run_build = check_setuppy_command()
--
-- # Disable OSX Accelerate, it has too old LAPACK
-- os.environ['ACCELERATE'] = 'None'
--
-- # This import is here because it needs to be done before importing setup()
-- # from numpy.distutils, but after the MANIFEST removing and sdist import
-- # higher up in this file.
-- from setuptools import setup
--
-- if run_build:
-- from numpy.distutils.core import setup
--
-- # Customize extension building
-- cmdclass['build_ext'] = get_build_ext_override()
-- cmdclass['build_clib'] = get_build_clib_override()
--
-- if not 'sdist' in sys.argv:
-- # Generate Cython sources, unless we're creating an sdist
-- # Cython is a build dependency, and shipping generated .c files
-- # can cause problems (see gh-14199)
-- generate_cython()
--
-- metadata['configuration'] = configuration
-- else:
-- # Don't import numpy here - non-build actions are required to succeed
-- # without NumPy for example when pip is used to install Scipy when
-- # NumPy is not yet present in the system.
--
-- # Version number is added to metadata inside configuration() if build
-- # is run.
-- metadata['version'] = get_version_info('.')[0]
--
-- setup(**metadata)
--
--
--if __name__ == '__main__':
-- setup_package()
-diff '--color=auto' -uNr scipy/setup.py scipy.mod/setup.py
---- scipy/setup.py 1970-01-01 05:30:00.000000000 +0530
-+++ scipy.mod/setup.py 2023-10-30 19:22:02.921729484 +0530
-@@ -0,0 +1,545 @@
-+#!/usr/bin/env python
-+"""SciPy: Scientific Library for Python
-+
-+SciPy (pronounced "Sigh Pie") is open-source software for mathematics,
-+science, and engineering. The SciPy library
-+depends on NumPy, which provides convenient and fast N-dimensional
-+array manipulation. The SciPy library is built to work with NumPy
-+arrays, and provides many user-friendly and efficient numerical
-+routines such as routines for numerical integration and optimization.
-+Together, they run on all popular operating systems, are quick to
-+install, and are free of charge. NumPy and SciPy are easy to use,
-+but powerful enough to be depended upon by some of the world's
-+leading scientists and engineers. If you need to manipulate
-+numbers on a computer and display or publish the results,
-+give SciPy a try!
-+
-+"""
-+
-+
-+# IMPORTANT:
-+#
-+# THIS FILE IS INTENTIONALLY RENAMED FROM setup.py TO _setup.py
-+# IT IS ONLY KEPT IN THE REPO BECAUSE conda-forge STILL NEEDS IT
-+# FOR BUILDING SCIPY ON WINDOWS. IT SHOULD NOT BE USED BY ANYONE
-+# ELSE. USE `pip install .` OR ANOTHER INSTALL COMMAND USING A
-+# BUILD FRONTEND LIKE pip OR pypa/build TO INSTALL SCIPY FROM SOURCE.
-+#
-+# SEE http://scipy.github.io/devdocs/building/index.html FOR BUILD
-+# INSTRUCTIONS.
-+
-+
-+DOCLINES = (__doc__ or '').split("\n")
-+
-+import os
-+import sys
-+import subprocess
-+import textwrap
-+import warnings
-+import sysconfig
-+from tools.version_utils import write_version_py, get_version_info
-+from tools.version_utils import IS_RELEASE_BRANCH
-+import importlib
-+
-+
-+if sys.version_info[:2] < (3, 9):
-+ raise RuntimeError("Python version >= 3.9 required.")
-+
-+import builtins
-+
-+
-+CLASSIFIERS = """\
-+Development Status :: 5 - Production/Stable
-+Intended Audience :: Science/Research
-+Intended Audience :: Developers
-+License :: OSI Approved :: BSD License
-+Programming Language :: C
-+Programming Language :: Python
-+Programming Language :: Python :: 3
-+Programming Language :: Python :: 3.9
-+Programming Language :: Python :: 3.10
-+Programming Language :: Python :: 3.11
-+Topic :: Software Development :: Libraries
-+Topic :: Scientific/Engineering
-+Operating System :: Microsoft :: Windows
-+Operating System :: POSIX :: Linux
-+Operating System :: POSIX
-+Operating System :: Unix
-+Operating System :: MacOS
-+
-+"""
-+
-+
-+# BEFORE importing setuptools, remove MANIFEST. Otherwise it may not be
-+# properly updated when the contents of directories change (true for distutils,
-+# not sure about setuptools).
-+if os.path.exists('MANIFEST'):
-+ os.remove('MANIFEST')
-+
-+# This is a bit hackish: we are setting a global variable so that the main
-+# scipy __init__ can detect if it is being loaded by the setup routine, to
-+# avoid attempting to load components that aren't built yet. While ugly, it's
-+# a lot more robust than what was previously being used.
-+builtins.__SCIPY_SETUP__ = True
-+
-+
-+def check_submodules():
-+ """ verify that the submodules are checked out and clean
-+ use `git submodule update --init`; on failure
-+ """
-+ if not os.path.exists('.git'):
-+ return
-+ with open('.gitmodules') as f:
-+ for l in f:
-+ if 'path' in l:
-+ p = l.split('=')[-1].strip()
-+ if not os.path.exists(p):
-+ raise ValueError('Submodule %s missing' % p)
-+
-+
-+ proc = subprocess.Popen(['git', 'submodule', 'status'],
-+ stdout=subprocess.PIPE)
-+ status, _ = proc.communicate()
-+ status = status.decode("ascii", "replace")
-+ for line in status.splitlines():
-+ if line.startswith('-') or line.startswith('+'):
-+ raise ValueError('Submodule not clean: %s' % line)
-+
-+
-+class concat_license_files():
-+ """Merge LICENSE.txt and LICENSES_bundled.txt for sdist creation
-+
-+ Done this way to keep LICENSE.txt in repo as exact BSD 3-clause (see
-+ NumPy gh-13447). This makes GitHub state correctly how SciPy is licensed.
-+ """
-+ def __init__(self):
-+ self.f1 = 'LICENSE.txt'
-+ self.f2 = 'LICENSES_bundled.txt'
-+
-+ def __enter__(self):
-+ """Concatenate files and remove LICENSES_bundled.txt"""
-+ with open(self.f1, 'r') as f1:
-+ self.bsd_text = f1.read()
-+
-+ with open(self.f1, 'a') as f1:
-+ with open(self.f2, 'r') as f2:
-+ self.bundled_text = f2.read()
-+ f1.write('\n\n')
-+ f1.write(self.bundled_text)
-+
-+ def __exit__(self, exception_type, exception_value, traceback):
-+ """Restore content of both files"""
-+ with open(self.f1, 'w') as f:
-+ f.write(self.bsd_text)
-+
-+
-+from distutils.command.sdist import sdist
-+class sdist_checked(sdist):
-+ """ check submodules on sdist to prevent incomplete tarballs """
-+ def run(self):
-+ check_submodules()
-+ with concat_license_files():
-+ sdist.run(self)
-+
-+
-+def get_build_ext_override():
-+ """
-+ Custom build_ext command to tweak extension building.
-+ """
-+ from numpy.distutils.command.build_ext import build_ext as npy_build_ext
-+ if int(os.environ.get('SCIPY_USE_PYTHRAN', 1)):
-+ try:
-+ import pythran
-+ from pythran.dist import PythranBuildExt
-+ except ImportError:
-+ BaseBuildExt = npy_build_ext
-+ else:
-+ BaseBuildExt = PythranBuildExt[npy_build_ext]
-+ _pep440 = importlib.import_module('scipy._lib._pep440')
-+ if _pep440.parse(pythran.__version__) < _pep440.Version('0.11.0'):
-+ raise RuntimeError("The installed `pythran` is too old, >= "
-+ "0.11.0 is needed, {} detected. Please "
-+ "upgrade Pythran, or use `export "
-+ "SCIPY_USE_PYTHRAN=0`.".format(
-+ pythran.__version__))
-+ else:
-+ BaseBuildExt = npy_build_ext
-+
-+ class build_ext(BaseBuildExt):
-+ def finalize_options(self):
-+ super().finalize_options()
-+
-+ # Disable distutils parallel build, due to race conditions
-+ # in numpy.distutils (Numpy issue gh-15957)
-+ if self.parallel:
-+ print("NOTE: -j build option not supported. Set NPY_NUM_BUILD_JOBS=4 "
-+ "for parallel build.")
-+ self.parallel = None
-+
-+ def build_extension(self, ext):
-+ # When compiling with GNU compilers, use a version script to
-+ # hide symbols during linking.
-+ if self.__is_using_gnu_linker(ext):
-+ export_symbols = self.get_export_symbols(ext)
-+ text = '{global: %s; local: *; };' % (';'.join(export_symbols),)
-+
-+ script_fn = os.path.join(self.build_temp, 'link-version-{}.map'.format(ext.name))
-+ with open(script_fn, 'w') as f:
-+ f.write(text)
-+ # line below fixes gh-8680
-+ ext.extra_link_args = [arg for arg in ext.extra_link_args if not "version-script" in arg]
-+ ext.extra_link_args.append('-Wl,--version-script=' + script_fn)
-+
-+ # Allow late configuration
-+ hooks = getattr(ext, '_pre_build_hook', ())
-+ _run_pre_build_hooks(hooks, (self, ext))
-+
-+ super().build_extension(ext)
-+
-+ def __is_using_gnu_linker(self, ext):
-+ if not sys.platform.startswith('linux'):
-+ return False
-+
-+ # Fortran compilation with gfortran uses it also for
-+ # linking. For the C compiler, we detect gcc in a similar
-+ # way as distutils does it in
-+ # UnixCCompiler.runtime_library_dir_option
-+ if ext.language == 'f90':
-+ is_gcc = (self._f90_compiler.compiler_type in ('gnu', 'gnu95'))
-+ elif ext.language == 'f77':
-+ is_gcc = (self._f77_compiler.compiler_type in ('gnu', 'gnu95'))
-+ else:
-+ is_gcc = False
-+ if self.compiler.compiler_type == 'unix':
-+ cc = sysconfig.get_config_var("CC")
-+ if not cc:
-+ cc = ""
-+ compiler_name = os.path.basename(cc.split(" ")[0])
-+ is_gcc = "gcc" in compiler_name or "g++" in compiler_name
-+ return is_gcc and sysconfig.get_config_var('GNULD') == 'yes'
-+
-+ return build_ext
-+
-+
-+def get_build_clib_override():
-+ """
-+ Custom build_clib command to tweak library building.
-+ """
-+ from numpy.distutils.command.build_clib import build_clib as old_build_clib
-+
-+ class build_clib(old_build_clib):
-+ def finalize_options(self):
-+ super().finalize_options()
-+
-+ # Disable parallelization (see build_ext above)
-+ self.parallel = None
-+
-+ def build_a_library(self, build_info, lib_name, libraries):
-+ # Allow late configuration
-+ hooks = build_info.get('_pre_build_hook', ())
-+ _run_pre_build_hooks(hooks, (self, build_info))
-+ old_build_clib.build_a_library(self, build_info, lib_name, libraries)
-+
-+ return build_clib
-+
-+
-+def _run_pre_build_hooks(hooks, args):
-+ """Call a sequence of pre-build hooks, if any"""
-+ if hooks is None:
-+ hooks = ()
-+ elif not hasattr(hooks, '__iter__'):
-+ hooks = (hooks,)
-+ for hook in hooks:
-+ hook(*args)
-+
-+
-+def generate_cython():
-+ cwd = os.path.abspath(os.path.dirname(__file__))
-+ print("Cythonizing sources")
-+ p = subprocess.call([sys.executable,
-+ os.path.join(cwd, 'tools', 'cythonize.py'),
-+ 'scipy'],
-+ cwd=cwd)
-+ if p != 0:
-+ # Could be due to a too old pip version and build isolation, check that
-+ try:
-+ # Note, pip may not be installed or not have been used
-+ import pip
-+ except (ImportError, ModuleNotFoundError):
-+ raise RuntimeError("Running cythonize failed!")
-+ else:
-+ _pep440 = importlib.import_module('scipy._lib._pep440')
-+ if _pep440.parse(pip.__version__) < _pep440.Version('18.0.0'):
-+ raise RuntimeError("Cython not found or too old. Possibly due "
-+ "to `pip` being too old, found version {}, "
-+ "needed is >= 18.0.0.".format(
-+ pip.__version__))
-+ else:
-+ raise RuntimeError("Running cythonize failed!")
-+
-+
-+def parse_setuppy_commands():
-+ """Check the commands and respond appropriately. Disable broken commands.
-+
-+ Return a boolean value for whether or not to run the build or not (avoid
-+ parsing Cython and template files if False).
-+ """
-+ args = sys.argv[1:]
-+
-+ if not args:
-+ # User forgot to give an argument probably, let setuptools handle that.
-+ return True
-+
-+ info_commands = ['--help-commands', '--name', '--version', '-V',
-+ '--fullname', '--author', '--author-email',
-+ '--maintainer', '--maintainer-email', '--contact',
-+ '--contact-email', '--url', '--license', '--description',
-+ '--long-description', '--platforms', '--classifiers',
-+ '--keywords', '--provides', '--requires', '--obsoletes']
-+
-+ for command in info_commands:
-+ if command in args:
-+ return False
-+
-+ # Note that 'alias', 'saveopts' and 'setopt' commands also seem to work
-+ # fine as they are, but are usually used together with one of the commands
-+ # below and not standalone. Hence they're not added to good_commands.
-+ good_commands = ('develop', 'sdist', 'build', 'build_ext', 'build_py',
-+ 'build_clib', 'build_scripts', 'bdist_wheel', 'bdist_rpm',
-+ 'bdist_wininst', 'bdist_msi', 'bdist_mpkg')
-+
-+ for command in good_commands:
-+ if command in args:
-+ return True
-+
-+ # The following commands are supported, but we need to show more
-+ # useful messages to the user
-+ if 'install' in args:
-+ print(textwrap.dedent("""
-+ Note: for reliable uninstall behaviour and dependency installation
-+ and uninstallation, please use pip instead of using
-+ `setup.py install`:
-+
-+ - `pip install .` (from a git repo or downloaded source
-+ release)
-+ - `pip install scipy` (last SciPy release on PyPI)
-+
-+ """))
-+ return True
-+
-+ if '--help' in args or '-h' in sys.argv[1]:
-+ print(textwrap.dedent("""
-+ SciPy-specific help
-+ -------------------
-+
-+ To install SciPy from here with reliable uninstall, we recommend
-+ that you use `pip install .`. To install the latest SciPy release
-+ from PyPI, use `pip install scipy`.
-+
-+ For help with build/installation issues, please ask on the
-+ scipy-user mailing list. If you are sure that you have run
-+ into a bug, please report it at https://github.com/scipy/scipy/issues.
-+
-+ Setuptools commands help
-+ ------------------------
-+ """))
-+ return False
-+
-+
-+ # The following commands aren't supported. They can only be executed when
-+ # the user explicitly adds a --force command-line argument.
-+ bad_commands = dict(
-+ test="""
-+ `setup.py test` is not supported. Use one of the following
-+ instead:
-+
-+ - `python runtests.py` (to build and test)
-+ - `python runtests.py --no-build` (to test installed scipy)
-+ - `>>> scipy.test()` (run tests for installed scipy
-+ from within an interpreter)
-+ """,
-+ upload="""
-+ `setup.py upload` is not supported, because it's insecure.
-+ Instead, build what you want to upload and upload those files
-+ with `twine upload -s <filenames>` instead.
-+ """,
-+ upload_docs="`setup.py upload_docs` is not supported",
-+ easy_install="`setup.py easy_install` is not supported",
-+ clean="""
-+ `setup.py clean` is not supported, use one of the following instead:
-+
-+ - `git clean -xdf` (cleans all files)
-+ - `git clean -Xdf` (cleans all versioned files, doesn't touch
-+ files that aren't checked into the git repo)
-+ """,
-+ check="`setup.py check` is not supported",
-+ register="`setup.py register` is not supported",
-+ bdist_dumb="`setup.py bdist_dumb` is not supported",
-+ bdist="`setup.py bdist` is not supported",
-+ flake8="`setup.py flake8` is not supported, use flake8 standalone",
-+ build_sphinx="`setup.py build_sphinx` is not supported, see doc/README.md",
-+ )
-+ bad_commands['nosetests'] = bad_commands['test']
-+ for command in ('upload_docs', 'easy_install', 'bdist', 'bdist_dumb',
-+ 'register', 'check', 'install_data', 'install_headers',
-+ 'install_lib', 'install_scripts', ):
-+ bad_commands[command] = "`setup.py %s` is not supported" % command
-+
-+ for command in bad_commands.keys():
-+ if command in args:
-+ print(textwrap.dedent(bad_commands[command]) +
-+ "\nAdd `--force` to your command to use it anyway if you "
-+ "must (unsupported).\n")
-+ sys.exit(1)
-+
-+ # Commands that do more than print info, but also don't need Cython and
-+ # template parsing.
-+ other_commands = ['egg_info', 'install_egg_info', 'rotate']
-+ for command in other_commands:
-+ if command in args:
-+ return False
-+
-+ # If we got here, we didn't detect what setup.py command was given
-+ warnings.warn("Unrecognized setuptools command ('{}'), proceeding with "
-+ "generating Cython sources and expanding templates".format(
-+ ' '.join(sys.argv[1:])))
-+ return True
-+
-+def check_setuppy_command():
-+ run_build = parse_setuppy_commands()
-+ if run_build:
-+ try:
-+ pkgname = 'numpy'
-+ import numpy
-+ pkgname = 'pybind11'
-+ import pybind11
-+ except ImportError as exc: # We do not have our build deps installed
-+ print(textwrap.dedent(
-+ """Error: '%s' must be installed before running the build.
-+ """
-+ % (pkgname,)))
-+ sys.exit(1)
-+
-+ return run_build
-+
-+def configuration(parent_package='', top_path=None):
-+ from numpy.distutils.system_info import get_info, NotFoundError
-+ from numpy.distutils.misc_util import Configuration
-+
-+ lapack_opt = get_info('lapack_opt')
-+
-+ if not lapack_opt:
-+ if sys.platform == "darwin":
-+ msg = ('No BLAS/LAPACK libraries found. '
-+ 'Note: Accelerate is no longer supported.')
-+ else:
-+ msg = 'No BLAS/LAPACK libraries found.'
-+ msg += ("\n"
-+ "To build Scipy from sources, BLAS & LAPACK libraries "
-+ "need to be installed.\n"
-+ "See site.cfg.example in the Scipy source directory and\n"
-+ "https://docs.scipy.org/doc/scipy/dev/contributor/building.html "
-+ "for details.")
-+ raise NotFoundError(msg)
-+
-+ config = Configuration(None, parent_package, top_path)
-+ config.set_options(ignore_setup_xxx_py=True,
-+ assume_default_configuration=True,
-+ delegate_options_to_subpackages=True,
-+ quiet=True)
-+
-+ config.add_subpackage('scipy')
-+ config.add_data_files(('scipy', '*.txt'))
-+
-+ config.get_version('scipy/version.py')
-+
-+ return config
-+
-+
-+def setup_package():
-+ # In maintenance branch, change np_maxversion to N+3 if numpy is at N
-+ # Update here, in pyproject.toml, and in scipy/__init__.py
-+ # Rationale: SciPy builds without deprecation warnings with N; deprecations
-+ # in N+1 will turn into errors in N+3
-+ # For Python versions, if releases is (e.g.) <=3.9.x, set bound to 3.10
-+ np_minversion = '1.21.6'
-+ np_maxversion = '1.28.0'
-+ python_minversion = '3.9'
-+ python_maxversion = '3.13'
-+ if IS_RELEASE_BRANCH:
-+ req_np = 'numpy>={},<{}'.format(np_minversion, np_maxversion)
-+ req_py = '>={},<{}'.format(python_minversion, python_maxversion)
-+ else:
-+ req_np = 'numpy>={}'.format(np_minversion)
-+ req_py = '>={}'.format(python_minversion)
-+
-+ # Rewrite the version file every time
-+ write_version_py('.')
-+
-+ cmdclass = {'sdist': sdist_checked}
-+
-+ metadata = dict(
-+ name='scipy',
-+ maintainer="SciPy Developers",
-+ maintainer_email="scipy-dev@python.org",
-+ description=DOCLINES[0],
-+ long_description="\n".join(DOCLINES[2:]),
-+ url="https://www.scipy.org",
-+ download_url="https://github.com/scipy/scipy/releases",
-+ project_urls={
-+ "Bug Tracker": "https://github.com/scipy/scipy/issues",
-+ "Documentation": "https://docs.scipy.org/doc/scipy/reference/",
-+ "Source Code": "https://github.com/scipy/scipy",
-+ },
-+ license='BSD',
-+ cmdclass=cmdclass,
-+ classifiers=[_f for _f in CLASSIFIERS.split('\n') if _f],
-+ platforms=["Windows", "Linux", "Solaris", "Mac OS-X", "Unix"],
-+ install_requires=[req_np],
-+ python_requires=req_py,
-+ zip_safe=False,
-+ )
-+
-+ if "--force" in sys.argv:
-+ run_build = True
-+ sys.argv.remove('--force')
-+ else:
-+ # Raise errors for unsupported commands, improve help output, etc.
-+ run_build = check_setuppy_command()
-+
-+ # Disable OSX Accelerate, it has too old LAPACK
-+ os.environ['ACCELERATE'] = 'None'
-+
-+ # This import is here because it needs to be done before importing setup()
-+ # from numpy.distutils, but after the MANIFEST removing and sdist import
-+ # higher up in this file.
-+ from setuptools import setup
-+
-+ if run_build:
-+ from numpy.distutils.core import setup
-+
-+ # Customize extension building
-+ cmdclass['build_ext'] = get_build_ext_override()
-+ cmdclass['build_clib'] = get_build_clib_override()
-+
-+ if not 'sdist' in sys.argv:
-+ # Generate Cython sources, unless we're creating an sdist
-+ # Cython is a build dependency, and shipping generated .c files
-+ # can cause problems (see gh-14199)
-+ generate_cython()
-+
-+ metadata['configuration'] = configuration
-+ else:
-+ # Don't import numpy here - non-build actions are required to succeed
-+ # without NumPy for example when pip is used to install Scipy when
-+ # NumPy is not yet present in the system.
-+
-+ # Version number is added to metadata inside configuration() if build
-+ # is run.
-+ metadata['version'] = get_version_info('.')[0]
-+
-+ setup(**metadata)
-+
-+
-+if __name__ == '__main__':
-+ setup_package()
diff --git a/pythonforandroid/recipes/scipy/wrapper.py b/pythonforandroid/recipes/scipy/wrapper.py
new file mode 100644
index 0000000000..ca0e60d22e
--- /dev/null
+++ b/pythonforandroid/recipes/scipy/wrapper.py
@@ -0,0 +1,62 @@
+#!/usr/bin/python3
+
+# Taken from https://github.com/termux/termux-packages/blob/master/packages/python-scipy/wrapper.py.in
+
+import os
+import subprocess
+import sys
+import typing
+
+"""
+This wrapper is used to ignore or replace some unsupported flags for flang-new.
+
+It will operate as follows:
+
+1. Ignore `-Minform=inform` and `-fdiagnostics-color`.
+ They are added by meson automatically, but are not supported by flang-new yet.
+2. Remove `-lflang` and `-lpgmath`.
+ It exists in classic-flang but doesn't exist in flang-new.
+3. Replace `-Oz` to `-O2`.
+ `-Oz` is not supported by flang-new.
+4. Replace `-module` to `-J`.
+ See https://github.com/llvm/llvm-project/issues/66969
+5. Ignore `-MD`, `-MQ file` and `-MF file`.
+ They generates files used by GNU make but we're using ninja.
+6. Ignore `-fvisibility=hidden`.
+ It is not supported by flang-new, and ignoring it will not break the functionality,
+ as scipy also uses version script for shared libraries.
+"""
+
+COMPLIER_PATH = "@COMPILER@"
+
+
+def main(argv: typing.List[str]):
+ cwd = os.getcwd()
+ argv_new = []
+ i = 0
+ while i < len(argv):
+ arg = argv[i]
+ if arg in [
+ "-Minform=inform",
+ "-lflang",
+ "-lpgmath",
+ "-MD",
+ "-fvisibility=hidden",
+ ] or arg.startswith("-fdiagnostics-color"):
+ pass
+ elif arg == "-Oz":
+ argv_new.append("-O2")
+ elif arg == "-module":
+ argv_new.append("-J")
+ elif arg in ["-MQ", "-MF"]:
+ i += 1
+ else:
+ argv_new.append(arg)
+ i += 1
+
+ args = [COMPLIER_PATH] + argv_new
+ subprocess.check_call(args, env=os.environ, cwd=cwd, text=True)
+
+
+if __name__ == "__main__":
+ main(sys.argv[1:])
From 96112666e3f3831bef9d6414234d31942458fa0a Mon Sep 17 00:00:00 2001
From: clayote <public@zacharyspector.com>
Date: Thu, 23 Oct 2025 06:42:59 +1300
Subject: [PATCH 147/158] Patch out the `ast.Str` reference in Kivy, for Python
3.14 compatibility (#3241)
* `python`: update to `3.14.0`
* Patch out the reference to `ast.Str` in `kivy.lang.parser`
It broke Python 3.14, and I don't think Kivy even supports Python 3.6 anymore
---------
Co-authored-by: Ansh Dadwal <anshdadwal298@gmail.com>
---
pythonforandroid/recipes/kivy/__init__.py | 6 +++++-
pythonforandroid/recipes/kivy/no-ast-str.patch | 16 ++++++++++++++++
2 files changed, 21 insertions(+), 1 deletion(-)
create mode 100644 pythonforandroid/recipes/kivy/no-ast-str.patch
diff --git a/pythonforandroid/recipes/kivy/__init__.py b/pythonforandroid/recipes/kivy/__init__.py
index d9ca543f36..c545bbc4c1 100644
--- a/pythonforandroid/recipes/kivy/__init__.py
+++ b/pythonforandroid/recipes/kivy/__init__.py
@@ -32,7 +32,11 @@ class KivyRecipe(PyProjectRecipe):
# sdl-gl-swapwindow-nogil.patch is needed to avoid a deadlock.
# See: https://github.com/kivy/kivy/pull/8025
# WARNING: Remove this patch when a new Kivy version is released.
- patches = [("sdl-gl-swapwindow-nogil.patch", is_kivy_affected_by_deadlock_issue), "use_cython.patch"]
+ patches = [
+ ("sdl-gl-swapwindow-nogil.patch", is_kivy_affected_by_deadlock_issue),
+ "use_cython.patch",
+ "no-ast-str.patch"
+ ]
@property
def need_stl_shared(self):
diff --git a/pythonforandroid/recipes/kivy/no-ast-str.patch b/pythonforandroid/recipes/kivy/no-ast-str.patch
new file mode 100644
index 0000000000..09ac0efa74
--- /dev/null
+++ b/pythonforandroid/recipes/kivy/no-ast-str.patch
@@ -0,0 +1,16 @@
+diff -ur kivy-2.3.1b/kivy/lang/parser.py kivy-2.3.1/kivy/lang/parser.py
+--- kivy-2.3.1b/kivy/lang/parser.py 2025-10-19 13:04:51.542798827 +1300
++++ kivy-2.3.1/kivy/lang/parser.py 2025-10-19 13:05:16.007104601 +1300
+@@ -230,11 +230,7 @@
+
+ if isinstance(node, (ast.JoinedStr, ast.BoolOp)):
+ for n in node.values:
+- if isinstance(n, ast.Str):
+- # NOTE: required for python3.6
+- yield from cls.get_names_from_expression(n.s)
+- else:
+- yield from cls.get_names_from_expression(n.value)
++ yield from cls.get_names_from_expression(n.value)
+
+ if isinstance(node, ast.BinOp):
+ yield from cls.get_names_from_expression(node.right)
From 7593f9d62439b5864f7e6204fe382c424e11ad57 Mon Sep 17 00:00:00 2001
From: broccoliboy <broccoliboy@gmail.com>
Date: Sat, 25 Oct 2025 11:37:02 -0500
Subject: [PATCH 148/158] Proxy environment variables passthrough to recipes
(#3126)
* pass common proxy env vars to recipe env to fix pip build issues on corporate networks
* remove whitespace on blank lines
---
pythonforandroid/recipe.py | 5 +++++
1 file changed, 5 insertions(+)
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 5ca2d91a7d..7017e682ec 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -535,6 +535,11 @@ def get_recipe_env(self, arch=None, with_flags_in_cc=True):
if arch is None:
arch = self.filtered_archs[0]
env = arch.get_env(with_flags_in_cc=with_flags_in_cc)
+
+ for proxy_key in ['HTTP_PROXY', 'http_proxy', 'HTTPS_PROXY', 'https_proxy']:
+ if proxy_key in environ:
+ env[proxy_key] = environ[proxy_key]
+
return env
def prebuild_arch(self, arch):
From b4accdf8b1771faa112a32fbbe4c61e3d8b5cdf8 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Tue, 14 Oct 2025 23:46:04 +0530
Subject: [PATCH 149/158] `cffi`: fix build
---
pythonforandroid/recipes/cffi/__init__.py | 21 +++++++------------
.../recipes/cryptography/__init__.py | 4 ++--
pythonforandroid/recipes/libffi/__init__.py | 2 +-
3 files changed, 10 insertions(+), 17 deletions(-)
diff --git a/pythonforandroid/recipes/cffi/__init__.py b/pythonforandroid/recipes/cffi/__init__.py
index f0c25a92c9..11446df950 100644
--- a/pythonforandroid/recipes/cffi/__init__.py
+++ b/pythonforandroid/recipes/cffi/__init__.py
@@ -1,22 +1,19 @@
import os
-from pythonforandroid.recipe import CompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class CffiRecipe(CompiledComponentsPythonRecipe):
+class CffiRecipe(PyProjectRecipe):
"""
Extra system dependencies: autoconf, automake and libtool.
"""
name = 'cffi'
- version = '1.15.1'
- url = 'https://pypi.python.org/packages/source/c/cffi/cffi-{version}.tar.gz'
+ version = '2.0.0'
+ url = 'https://github.com/python-cffi/cffi/archive/refs/tags/v{version}.tar.gz'
- depends = ['setuptools', 'pycparser', 'libffi']
+ depends = ['pycparser', 'libffi']
patches = ['disable-pkg-config.patch']
- # call_hostpython_via_targetpython = False
- install_in_hostpython = True
-
def get_hostrecipe_env(self, arch=None):
# fixes missing ffi.h on some host systems (e.g. gentoo)
env = super().get_hostrecipe_env(arch)
@@ -25,8 +22,8 @@ def get_hostrecipe_env(self, arch=None):
env['FFI_INC'] = ",".join(includes)
return env
- def get_recipe_env(self, arch=None):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch=None, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
libffi = self.get_recipe('libffi', self.ctx)
includes = libffi.get_include_dirs(arch)
env['CFLAGS'] = ' -I'.join([env.get('CFLAGS', '')] + includes)
@@ -36,10 +33,6 @@ def get_recipe_env(self, arch=None):
env['LDFLAGS'] += ' -L{}'.format(os.path.join(self.ctx.bootstrap.build_dir, 'libs', arch.arch))
# required for libc and libdl
env['LDFLAGS'] += ' -L{}'.format(arch.ndk_lib_dir_versioned)
- env['PYTHONPATH'] = ':'.join([
- self.ctx.get_site_packages_dir(arch),
- env['BUILDLIB_PATH'],
- ])
env['LDFLAGS'] += ' -L{}'.format(self.ctx.python_recipe.link_root(arch.arch))
env['LDFLAGS'] += ' -lpython{}'.format(self.ctx.python_recipe.link_version)
return env
diff --git a/pythonforandroid/recipes/cryptography/__init__.py b/pythonforandroid/recipes/cryptography/__init__.py
index c6a91a13d7..e54eaa1bd8 100644
--- a/pythonforandroid/recipes/cryptography/__init__.py
+++ b/pythonforandroid/recipes/cryptography/__init__.py
@@ -5,9 +5,9 @@
class CryptographyRecipe(RustCompiledComponentsRecipe):
name = 'cryptography'
- version = '42.0.1'
+ version = '46.0.3'
url = 'https://github.com/pyca/cryptography/archive/refs/tags/{version}.tar.gz'
- depends = ['openssl']
+ depends = ['openssl', 'cffi']
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
diff --git a/pythonforandroid/recipes/libffi/__init__.py b/pythonforandroid/recipes/libffi/__init__.py
index 767881b793..5636dffa7d 100644
--- a/pythonforandroid/recipes/libffi/__init__.py
+++ b/pythonforandroid/recipes/libffi/__init__.py
@@ -35,7 +35,7 @@ def build_arch(self, arch):
shprint(sh.make, '-j', str(cpu_count()), 'libffi.la', _env=env)
def get_include_dirs(self, arch):
- return [join(self.get_build_dir(arch.arch), 'include')]
+ return [join(self.get_build_dir(arch), 'include')]
recipe = LibffiRecipe()
From e1bd24977c6b2fa5011173d1be596cab9153bfd9 Mon Sep 17 00:00:00 2001
From: clayote <public@zacharyspector.com>
Date: Tue, 28 Oct 2025 02:44:32 +1300
Subject: [PATCH 150/158] Update to Python 3.14, remove distutils (#3242)
* Use `pip install` instead of `setup.py install` in `recipe.py`
On Python 3.14, many recipes, notably `msgpack`, no longer byte-
compile with `setup.py install`, because the way that command
byte-compiles relies on `distutils`, which is gone.
* Import from setuptools and not distutils in the `android` recipe
* Remove freetype's distutils "fallback"
It was actually using distutils every time, apparently?
* Bump `freetype` version to match that of `freetype-py`
* On second thought, use the latest freetype version
* Use the latest *released* freetype-py version
* Tell setuptools what freetype-py version we have
* Turn `FreetypePyRecipe` into a `PyProjectRecipe`
I don't think this will require the version hack
* Put the `setuptools-scm` version hack back in
* Fix `FreetypePyRecipe.get_recipe_env`
* Switch to pythonhosted FreetypePy
`setuptools-scm` can't deal with the github hosted one, it seems.
* Update apsw recipe, use pythonhosted package
* Fix `LibffiRecipe.get_include_dirs` for `arch=None`
* Use recent libffi and cffi
* Require setuptools for every recipe
* `python`: update to `3.14.0`
* Use `pip install` instead of `setup.py install` in `recipe.py`
On Python 3.14, many recipes, notably `msgpack`, no longer byte-
compile with `setup.py install`, because the way that command
byte-compiles relies on `distutils`, which is gone.
* Import from setuptools and not distutils in the `android` recipe
* Remove freetype's distutils "fallback"
It was actually using distutils every time, apparently?
* Bump `freetype` version to match that of `freetype-py`
* On second thought, use the latest freetype version
* Use the latest *released* freetype-py version
* Tell setuptools what freetype-py version we have
* Turn `FreetypePyRecipe` into a `PyProjectRecipe`
I don't think this will require the version hack
* Put the `setuptools-scm` version hack back in
* Fix `FreetypePyRecipe.get_recipe_env`
* Switch to pythonhosted FreetypePy
`setuptools-scm` can't deal with the github hosted one, it seems.
* Update apsw recipe, use pythonhosted package
* Require setuptools for every recipe
* `python`: update to `3.14.0`
* Remove 'six' from `build_order` in `TestToolchainCL.test_create`
* Add 'six' to `python_modules` in `TestToolchainCL.test_create`
* Remove 'setuptools' from 'build_order' in `TestToolchainCL.test_create`
I guess because the hostpython already has it, there's no need to build it again?
* Revert the change to `modules_build_dir` in `Python3Recipe.create_python_bundle`
* Remove old versions from `modules_build_dir` expected value in `TestPython3Recipe.test_create_python_bundle`
* Roll back to an earlier apsw version
* only load libpython once
* fix pip install path
* fix build
* shift foundPython check
* fix newline
* remove unnecessary test
---------
Co-authored-by: Ansh Dadwal <anshdadwal298@gmail.com>
---
.../java/org/kivy/android/PythonUtil.java | 3 +-
pythonforandroid/recipe.py | 27 ++++-----
pythonforandroid/recipes/android/src/setup.py | 2 +-
pythonforandroid/recipes/apsw/__init__.py | 30 +++-------
.../recipes/freetype-py/__init__.py | 13 ++--
.../freetype-py/fall-back-to-distutils.patch | 15 -----
.../recipes/freetype-py/fix_import.patch | 8 +++
pythonforandroid/recipes/freetype/__init__.py | 3 +-
.../recipes/hostpython3/__init__.py | 2 +-
pythonforandroid/recipes/python3/__init__.py | 17 ++----
pythonforandroid/recipes/six/__init__.py | 10 ----
pythonforandroid/recipes/sqlite3/Android.mk | 11 ----
pythonforandroid/recipes/sqlite3/__init__.py | 57 ++++++++----------
tests/recipes/test_python3.py | 60 -------------------
tests/test_toolchain.py | 4 +-
15 files changed, 71 insertions(+), 191 deletions(-)
delete mode 100644 pythonforandroid/recipes/freetype-py/fall-back-to-distutils.patch
create mode 100644 pythonforandroid/recipes/freetype-py/fix_import.patch
delete mode 100644 pythonforandroid/recipes/six/__init__.py
delete mode 100644 pythonforandroid/recipes/sqlite3/Android.mk
diff --git a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
index 065f43c3bd..abf21dd47b 100644
--- a/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
+++ b/pythonforandroid/bootstraps/common/build/src/main/java/org/kivy/android/PythonUtil.java
@@ -51,7 +51,7 @@ protected static ArrayList<String> getLibraries(File libsDir) {
addLibraryIfExists(libsList, name, libsDir);
}
- for (int v = 5; v <= 14; v++) {
+ for (int v = 14; v >= 5; v--) {
libsList.add("python3." + v + (v <= 7 ? "m" : ""));
}
@@ -63,6 +63,7 @@ public static void loadLibraries(File filesDir, File libsDir) {
boolean foundPython = false;
for (String lib : getLibraries(libsDir)) {
+ if (lib.startsWith("python") && foundPython) {continue;}
Log.v(TAG, "Loading library: " + lib);
try {
System.loadLibrary(lib);
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 7017e682ec..23b1d6ab79 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -883,7 +883,7 @@ class PythonRecipe(Recipe):
on python2 or python3 which can break the dependency graph
'''
- hostpython_prerequisites = []
+ hostpython_prerequisites = ['setuptools']
'''List of hostpython packages required to build a recipe'''
_host_recipe = None
@@ -1025,18 +1025,11 @@ def install_python_package(self, arch, name=None, env=None, is_dir=True):
hostpython = sh.Command(self.hostpython_location)
hpenv = env.copy()
with current_directory(self.get_build_dir(arch.arch)):
-
- if isfile("setup.py"):
- shprint(hostpython, 'setup.py', 'install', '-O2',
- '--root={}'.format(self.ctx.get_python_install_dir(arch.arch)),
- '--install-lib=.',
- _env=hpenv, *self.setup_extra_args)
-
- # If asked, also install in the hostpython build dir
- if self.install_in_hostpython:
- self.install_hostpython_package(arch)
- else:
- warning("`PythonRecipe.install_python_package` called without `setup.py` file!")
+ shprint(hostpython, '-m', 'pip', 'install', '.',
+ '--compile', '--target',
+ self.ctx.get_python_install_dir(arch.arch),
+ _env=hpenv, *self.setup_extra_args
+ )
def get_hostrecipe_env(self, arch=None):
env = environ.copy()
@@ -1053,8 +1046,8 @@ def hostpython_site_dir(self):
def install_hostpython_package(self, arch):
env = self.get_hostrecipe_env(arch)
real_hostpython = sh.Command(self.real_hostpython_location)
- shprint(real_hostpython, 'setup.py', 'install', '-O2',
- '--install-lib=Lib/site-packages',
+ shprint(real_hostpython, '-m', 'pip', 'install', '.',
+ '--compile',
'--root={}'.format(self._host_recipe.site_root),
_env=env, *self.setup_extra_args)
@@ -1100,7 +1093,7 @@ class CompiledComponentsPythonRecipe(PythonRecipe):
def build_arch(self, arch):
'''Build any cython components, then install the Python module by
- calling setup.py install with the target Python dir.
+ calling pip install with the target Python dir.
'''
Recipe.build_arch(self, arch)
self.install_hostpython_prerequisites()
@@ -1149,7 +1142,7 @@ class CythonRecipe(PythonRecipe):
def build_arch(self, arch):
'''Build any cython components, then install the Python module by
- calling setup.py install with the target Python dir.
+ calling pip install with the target Python dir.
'''
Recipe.build_arch(self, arch)
self.build_cython_components(arch)
diff --git a/pythonforandroid/recipes/android/src/setup.py b/pythonforandroid/recipes/android/src/setup.py
index 8bf4512e05..8182ba9c58 100755
--- a/pythonforandroid/recipes/android/src/setup.py
+++ b/pythonforandroid/recipes/android/src/setup.py
@@ -1,4 +1,4 @@
-from distutils.core import setup, Extension
+from setuptools import setup, Extension
from Cython.Build import cythonize
import os
diff --git a/pythonforandroid/recipes/apsw/__init__.py b/pythonforandroid/recipes/apsw/__init__.py
index 42ad3ba337..825d5ced40 100644
--- a/pythonforandroid/recipes/apsw/__init__.py
+++ b/pythonforandroid/recipes/apsw/__init__.py
@@ -1,32 +1,16 @@
-from pythonforandroid.recipe import PythonRecipe
-from pythonforandroid.toolchain import current_directory, shprint
-import sh
+from pythonforandroid.recipe import PyProjectRecipe
-class ApswRecipe(PythonRecipe):
- version = '3.15.0-r1'
- url = 'https://github.com/rogerbinns/apsw/archive/{version}.tar.gz'
- depends = ['sqlite3', 'setuptools']
- call_hostpython_via_targetpython = False
+class ApswRecipe(PyProjectRecipe):
+ version = '3.50.4.0'
+ url = 'https://github.com/rogerbinns/apsw/releases/download/{version}/apsw-{version}.tar.gz'
+ depends = ['sqlite3']
site_packages_name = 'apsw'
- def build_arch(self, arch):
- env = self.get_recipe_env(arch)
- with current_directory(self.get_build_dir(arch.arch)):
- # Build python bindings
- hostpython = sh.Command(self.hostpython_location)
- shprint(hostpython,
- 'setup.py',
- 'build_ext',
- '--enable=fts4', _env=env)
- # Install python bindings
- super().build_arch(arch)
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
sqlite_recipe = self.get_recipe('sqlite3', self.ctx)
env['CFLAGS'] += ' -I' + sqlite_recipe.get_build_dir(arch.arch)
- env['LDFLAGS'] += ' -L' + sqlite_recipe.get_lib_dir(arch)
env['LIBS'] = env.get('LIBS', '') + ' -lsqlite3'
return env
diff --git a/pythonforandroid/recipes/freetype-py/__init__.py b/pythonforandroid/recipes/freetype-py/__init__.py
index 7be2f2e10c..0967cfb5d0 100644
--- a/pythonforandroid/recipes/freetype-py/__init__.py
+++ b/pythonforandroid/recipes/freetype-py/__init__.py
@@ -1,12 +1,17 @@
-from pythonforandroid.recipe import PythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class FreetypePyRecipe(PythonRecipe):
- version = '2.2.0'
+class FreetypePyRecipe(PyProjectRecipe):
+ version = '2.5.1'
url = 'https://github.com/rougier/freetype-py/archive/refs/tags/v{version}.tar.gz'
+ patches = ["fix_import.patch"]
depends = ['freetype']
- patches = ['fall-back-to-distutils.patch']
site_packages_name = 'freetype'
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env["SETUPTOOLS_SCM_PRETEND_VERSION_FOR_freetype_py"] = self.version
+ return env
+
recipe = FreetypePyRecipe()
diff --git a/pythonforandroid/recipes/freetype-py/fall-back-to-distutils.patch b/pythonforandroid/recipes/freetype-py/fall-back-to-distutils.patch
deleted file mode 100644
index 0f06f1854a..0000000000
--- a/pythonforandroid/recipes/freetype-py/fall-back-to-distutils.patch
+++ /dev/null
@@ -1,15 +0,0 @@
-diff -ruN freetype-py.orig/setup.py freetype-py/setup.py
---- freetype-py.orig/setup.py 2020-07-09 20:58:51.000000000 +0700
-+++ freetype-py/setup.py 2022-03-02 19:28:17.948831134 +0700
-@@ -12,7 +12,10 @@
- from io import open
- from os import path
-
--from setuptools import setup
-+try:
-+ from setuptools import setup
-+except ImportError:
-+ from distutils.core import setup
-
- if os.environ.get("FREETYPEPY_BUNDLE_FT"):
- print("# Will build and bundle FreeType.")
diff --git a/pythonforandroid/recipes/freetype-py/fix_import.patch b/pythonforandroid/recipes/freetype-py/fix_import.patch
new file mode 100644
index 0000000000..03ebcae776
--- /dev/null
+++ b/pythonforandroid/recipes/freetype-py/fix_import.patch
@@ -0,0 +1,8 @@
+diff '--color=auto' -uNr freetype-py-2.5.1/MANIFEST.in freetype-py-2.5.1.mod/MANIFEST.in
+--- freetype-py-2.5.1/MANIFEST.in 2024-08-29 23:12:30.000000000 +0530
++++ freetype-py-2.5.1.mod/MANIFEST.in 2025-10-26 11:54:45.052025521 +0530
+@@ -9,3 +9,4 @@
+ include LICENSE.txt
+ include README.rst
+ include setup-build-freetype.py
++recursive-include _custom_build *.py
diff --git a/pythonforandroid/recipes/freetype/__init__.py b/pythonforandroid/recipes/freetype/__init__.py
index e5ddfe1424..c584cba02c 100644
--- a/pythonforandroid/recipes/freetype/__init__.py
+++ b/pythonforandroid/recipes/freetype/__init__.py
@@ -24,7 +24,7 @@ class FreetypeRecipe(Recipe):
https://sourceforge.net/projects/freetype/files/freetype2/2.5.3/
"""
- version = '2.10.1'
+ version = '2.14.1'
url = 'https://download.savannah.gnu.org/releases/freetype/freetype-{version}.tar.gz' # noqa
built_libraries = {'libfreetype.so': 'objs/.libs'}
@@ -77,6 +77,7 @@ def build_arch(self, arch, with_harfbuzz=False):
'--host={}'.format(arch.command_prefix),
'--prefix={}'.format(prefix_path),
'--without-bzip2',
+ '--without-brotli',
'--with-png=no',
}
if not harfbuzz_in_recipes:
diff --git a/pythonforandroid/recipes/hostpython3/__init__.py b/pythonforandroid/recipes/hostpython3/__init__.py
index 094660fada..afc4df4955 100644
--- a/pythonforandroid/recipes/hostpython3/__init__.py
+++ b/pythonforandroid/recipes/hostpython3/__init__.py
@@ -36,7 +36,7 @@ class HostPython3Recipe(Recipe):
:class:`~pythonforandroid.python.HostPythonRecipe`
'''
- version = '3.11.13'
+ version = '3.14.0'
url = 'https://github.com/python/cpython/archive/refs/tags/v{version}.tar.gz'
'''The default url to download our host python recipe. This url will
diff --git a/pythonforandroid/recipes/python3/__init__.py b/pythonforandroid/recipes/python3/__init__.py
index 81aee7c66e..fb69b9a366 100644
--- a/pythonforandroid/recipes/python3/__init__.py
+++ b/pythonforandroid/recipes/python3/__init__.py
@@ -54,7 +54,7 @@ class Python3Recipe(TargetPythonRecipe):
:class:`~pythonforandroid.python.GuestPythonRecipe`
'''
- version = '3.11.13'
+ version = '3.14.0'
_p_version = Version(version)
url = 'https://github.com/python/cpython/archive/refs/tags/v{version}.tar.gz'
name = 'python3'
@@ -262,7 +262,7 @@ def add_flags(include_flags, link_dirs, link_libs):
info('Activating flags for sqlite3')
recipe = Recipe.get_recipe('sqlite3', self.ctx)
add_flags(' -I' + recipe.get_build_dir(arch.arch),
- ' -L' + recipe.get_lib_dir(arch), ' -lsqlite3')
+ ' -L' + recipe.get_build_dir(arch.arch), ' -lsqlite3')
info('Activating flags for libffi')
recipe = Recipe.get_recipe('libffi', self.ctx)
@@ -388,19 +388,12 @@ def create_python_bundle(self, dirn, arch):
copying all the modules and standard library to the right
place.
"""
- # Todo: find a better way to find the build libs folder
- modules_build_dir = join(
+ modules_build_dir = glob.glob(join(
self.get_build_dir(arch.arch),
'android-build',
'build',
- 'lib.{}{}-{}-{}'.format(
- # android is now supported platform
- "android" if self._p_version.minor >= 13 else "linux",
- '2' if self.version[0] == '2' else '',
- arch.command_prefix.split('-')[0],
- self.major_minor_version_string
- ))
-
+ 'lib.*'
+ ))[0]
# Compile to *.pyc the python modules
self.compile_python_files(modules_build_dir)
# Compile to *.pyc the standard python library
diff --git a/pythonforandroid/recipes/six/__init__.py b/pythonforandroid/recipes/six/__init__.py
deleted file mode 100644
index 3be8ce7578..0000000000
--- a/pythonforandroid/recipes/six/__init__.py
+++ /dev/null
@@ -1,10 +0,0 @@
-from pythonforandroid.recipe import PythonRecipe
-
-
-class SixRecipe(PythonRecipe):
- version = '1.15.0'
- url = 'https://pypi.python.org/packages/source/s/six/six-{version}.tar.gz'
- depends = ['setuptools']
-
-
-recipe = SixRecipe()
diff --git a/pythonforandroid/recipes/sqlite3/Android.mk b/pythonforandroid/recipes/sqlite3/Android.mk
deleted file mode 100644
index 57bc81573d..0000000000
--- a/pythonforandroid/recipes/sqlite3/Android.mk
+++ /dev/null
@@ -1,11 +0,0 @@
-LOCAL_PATH := $(call my-dir)/..
-
-include $(CLEAR_VARS)
-
-LOCAL_SRC_FILES := sqlite3.c
-
-LOCAL_MODULE := sqlite3
-
-LOCAL_CFLAGS := -DSQLITE_ENABLE_FTS4 -D_FILE_OFFSET_BITS=32 -DSQLITE_ENABLE_JSON1
-
-include $(BUILD_SHARED_LIBRARY)
diff --git a/pythonforandroid/recipes/sqlite3/__init__.py b/pythonforandroid/recipes/sqlite3/__init__.py
index 1f4292c1eb..4265a07077 100644
--- a/pythonforandroid/recipes/sqlite3/__init__.py
+++ b/pythonforandroid/recipes/sqlite3/__init__.py
@@ -1,36 +1,27 @@
-from os.path import join
-import shutil
-
-from pythonforandroid.recipe import NDKRecipe
-from pythonforandroid.util import ensure_dir
-
-
-class Sqlite3Recipe(NDKRecipe):
- version = '3.35.5'
- # Don't forget to change the URL when changing the version
- url = 'https://www.sqlite.org/2021/sqlite-amalgamation-3350500.zip'
- generated_libraries = ['sqlite3']
-
- def should_build(self, arch):
- return not self.has_libs(arch, 'libsqlite3.so')
-
- def prebuild_arch(self, arch):
- super().prebuild_arch(arch)
- # Copy the Android make file
- ensure_dir(join(self.get_build_dir(arch.arch), 'jni'))
- shutil.copyfile(join(self.get_recipe_dir(), 'Android.mk'),
- join(self.get_build_dir(arch.arch), 'jni/Android.mk'))
-
- def build_arch(self, arch, *extra_args):
- super().build_arch(arch)
- # Copy the shared library
- shutil.copyfile(join(self.get_build_dir(arch.arch), 'libs', arch.arch, 'libsqlite3.so'),
- join(self.ctx.get_libs_dir(arch.arch), 'libsqlite3.so'))
-
- def get_recipe_env(self, arch):
- env = super().get_recipe_env(arch)
- env['NDK_PROJECT_PATH'] = self.get_build_dir(arch.arch)
- return env
+import sh
+from pythonforandroid.logger import shprint
+from pythonforandroid.util import current_directory
+from pythonforandroid.recipe import Recipe
+from multiprocessing import cpu_count
+
+
+class Sqlite3Recipe(Recipe):
+ version = '3.50.4'
+ url = 'https://github.com/sqlite/sqlite/archive/refs/tags/version-{version}.tar.gz'
+ built_libraries = {'libsqlite3.so': '.'}
+
+ def build_arch(self, arch):
+ env = self.get_recipe_env(arch)
+ build_dir = self.get_build_dir(arch.arch)
+ config_args = {
+ '--host={}'.format(arch.command_prefix),
+ '--prefix={}'.format(build_dir),
+ '--disable-tcl',
+ }
+ with current_directory(build_dir):
+ configure = sh.Command('./configure')
+ shprint(configure, *config_args, _env=env)
+ shprint(sh.make, '-j', str(cpu_count()), _env=env)
recipe = Sqlite3Recipe()
diff --git a/tests/recipes/test_python3.py b/tests/recipes/test_python3.py
index 01d58f7d27..57b7ef8e01 100644
--- a/tests/recipes/test_python3.py
+++ b/tests/recipes/test_python3.py
@@ -142,63 +142,3 @@ def test_build_arch_wrong_ndk_api(self):
# restore recipe's ctx or we could get failures with other test,
# since we share `self.recipe with all the tests of the class
self.recipe.ctx.ndk_api = self.ctx.ndk_api
-
- @mock.patch('shutil.copystat')
- @mock.patch('shutil.copyfile')
- @mock.patch("pythonforandroid.util.chdir")
- @mock.patch("pythonforandroid.util.makedirs")
- @mock.patch("pythonforandroid.util.walk")
- @mock.patch("pythonforandroid.recipes.python3.sh.find")
- @mock.patch("pythonforandroid.recipes.python3.sh.cp")
- @mock.patch("pythonforandroid.recipes.python3.sh.zip")
- @mock.patch("pythonforandroid.recipes.python3.subprocess.call")
- def test_create_python_bundle(
- self,
- mock_subprocess,
- mock_sh_zip,
- mock_sh_cp,
- mock_sh_find,
- mock_walk,
- mock_makedirs,
- mock_chdir,
- mock_copyfile,
- mock_copystat,
- ):
- fake_compile_dir = '/fake/compile/dir'
- simulated_walk_result = [
- ["/fake_dir", ["__pycache__", "Lib"], ["README", "setup.py"]],
- ["/fake_dir/Lib", ["ctypes"], ["abc.pyc", "abc.py"]],
- ["/fake_dir/Lib/ctypes", [], ["util.pyc", "util.py"]],
- ]
- mock_walk.return_value = simulated_walk_result
- self.recipe.create_python_bundle(fake_compile_dir, self.arch)
-
- recipe_build_dir = self.recipe.get_build_dir(self.arch.arch)
- modules_build_dir = join(
- recipe_build_dir,
- 'android-build',
- 'build',
- 'lib.{}{}-{}-{}'.format(
- 'android' if self.recipe.version[2] >= "3" else 'linux',
- '2' if self.recipe.version[0] == '2' else '',
- self.arch.command_prefix.split('-')[0],
- self.recipe.major_minor_version_string
- ))
- expected_sp_paths = [
- modules_build_dir,
- join(recipe_build_dir, 'Lib'),
- self.ctx.get_python_install_dir(self.arch.arch),
- ]
- for n, (sp_call, kw) in enumerate(mock_subprocess.call_args_list):
- self.assertEqual(sp_call[0][-1], expected_sp_paths[n])
-
- # we expect two calls to `walk_valid_filens`
- self.assertEqual(len(mock_walk.call_args_list), 2)
-
- mock_sh_zip.assert_called()
- mock_sh_cp.assert_called()
- mock_sh_find.assert_called()
- mock_makedirs.assert_called()
- mock_chdir.assert_called()
- mock_copyfile.assert_called()
- mock_copystat.assert_called()
diff --git a/tests/test_toolchain.py b/tests/test_toolchain.py
index 874453f981..03b008fd35 100644
--- a/tests/test_toolchain.py
+++ b/tests/test_toolchain.py
@@ -84,9 +84,9 @@ def test_create(self):
]
build_order = [
'hostpython3', 'libffi', 'openssl', 'sqlite3', 'python3',
- 'genericndkbuild', 'setuptools', 'six', 'pyjnius', 'android',
+ 'genericndkbuild', 'pyjnius', 'android',
]
- python_modules = []
+ python_modules = ['six']
context = mock.ANY
project_dir = None
assert m_build_recipes.call_args_list == [
From 96f2401547fa0f5c3bb3838d78b3318b0cdc0045 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 1 Nov 2025 15:42:22 +0530
Subject: [PATCH 151/158] `pycryptodome`: update to 3.23.0
---
pythonforandroid/recipes/pycryptodome/__init__.py | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/pythonforandroid/recipes/pycryptodome/__init__.py b/pythonforandroid/recipes/pycryptodome/__init__.py
index 9418600a29..6600368b21 100644
--- a/pythonforandroid/recipes/pycryptodome/__init__.py
+++ b/pythonforandroid/recipes/pycryptodome/__init__.py
@@ -1,10 +1,10 @@
-from pythonforandroid.recipe import PythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class PycryptodomeRecipe(PythonRecipe):
- version = '3.6.3'
- url = 'https://github.com/Legrandin/pycryptodome/archive/v{version}.tar.gz'
- depends = ['setuptools', 'cffi']
+class PycryptodomeRecipe(PyProjectRecipe):
+ version = '3.23.0'
+ url = 'https://github.com/Legrandin/pycryptodome/archive/refs/tags/v{version}.tar.gz'
+ depends = ['cffi']
recipe = PycryptodomeRecipe()
From 386e68a50cb5bc487478ee88439b6162d6c0a032 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 1 Nov 2025 15:53:50 +0530
Subject: [PATCH 152/158] remove patchelf dependency
---
Dockerfile | 1 -
doc/source/quickstart.rst | 1 -
pythonforandroid/recipe.py | 2 +-
3 files changed, 1 insertion(+), 3 deletions(-)
diff --git a/Dockerfile b/Dockerfile
index 408a0802f7..c36bdbafcd 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -71,7 +71,6 @@ RUN ${RETRY} apt -y update -qq > /dev/null \
make \
openjdk-17-jdk \
patch \
- patchelf \
pkg-config \
python3 \
python3-dev \
diff --git a/doc/source/quickstart.rst b/doc/source/quickstart.rst
index f9aea86996..7438805065 100644
--- a/doc/source/quickstart.rst
+++ b/doc/source/quickstart.rst
@@ -86,7 +86,6 @@ the following command (re-adapted from the `Dockerfile` we use to perform CI bui
make \
openjdk-17-jdk \
patch \
- patchelf \
pkg-config \
python3 \
python3-dev \
diff --git a/pythonforandroid/recipe.py b/pythonforandroid/recipe.py
index 23b1d6ab79..9c086e0c83 100644
--- a/pythonforandroid/recipe.py
+++ b/pythonforandroid/recipe.py
@@ -1311,7 +1311,7 @@ def build_arch(self, arch):
return
self.install_hostpython_prerequisites(
- packages=["build[virtualenv]", "pip", "setuptools"] + self.hostpython_prerequisites
+ packages=["build[virtualenv]", "pip", "setuptools", "patchelf"] + self.hostpython_prerequisites
)
self.patch_shebangs(self._host_recipe.site_bin, self.real_hostpython_location)
From 63b9deddd3c0c5d6baf4020e3608c20ab58e7b1f Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sat, 1 Nov 2025 22:45:01 +0530
Subject: [PATCH 153/158] `pydantic`: update to 2.41.4
---
pythonforandroid/recipes/pydantic-core/__init__.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/pythonforandroid/recipes/pydantic-core/__init__.py b/pythonforandroid/recipes/pydantic-core/__init__.py
index bf76a65d0a..2ca49cf5dd 100644
--- a/pythonforandroid/recipes/pydantic-core/__init__.py
+++ b/pythonforandroid/recipes/pydantic-core/__init__.py
@@ -2,7 +2,7 @@
class PydanticcoreRecipe(RustCompiledComponentsRecipe):
- version = "2.16.1"
+ version = "2.41.4"
url = "https://github.com/pydantic/pydantic-core/archive/refs/tags/v{version}.tar.gz"
site_packages_name = "pydantic_core"
From ab95ee6300e6e95afc066907d1dd7e5ebc09a11c Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 1 Nov 2025 16:09:08 +0000
Subject: [PATCH 154/158] :white_check_mark: unit test patching &
checkdependencies modules
- tests/test_patching.py: tests covering platform detection,
architecture checks, API level comparisons, version checking, and
logical conjunctions used in recipe patch conditionals
- tests/test_checkdependencies.py: tests covering module import
verification, version requirement checking, and environment variable
handling
---
tests/test_checkdependencies.py | 195 +++++++++++++++++++
tests/test_patching.py | 326 ++++++++++++++++++++++++++++++++
2 files changed, 521 insertions(+)
create mode 100644 tests/test_checkdependencies.py
create mode 100644 tests/test_patching.py
diff --git a/tests/test_checkdependencies.py b/tests/test_checkdependencies.py
new file mode 100644
index 0000000000..c629893c51
--- /dev/null
+++ b/tests/test_checkdependencies.py
@@ -0,0 +1,195 @@
+import sys
+from unittest import mock
+
+from pythonforandroid import checkdependencies
+
+
+class TestCheckPythonDependencies:
+ """Test check_python_dependencies function."""
+
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_all_modules_present(self, mock_import):
+ """Test that check_python_dependencies completes when all modules are present."""
+ # Mock all required modules
+ mock_colorama = mock.Mock()
+ mock_colorama.__version__ = '0.4.0'
+ mock_sh = mock.Mock()
+ mock_sh.__version__ = '1.12'
+ mock_appdirs = mock.Mock()
+ mock_jinja2 = mock.Mock()
+
+ def import_side_effect(name):
+ if name == 'colorama':
+ return mock_colorama
+ elif name == 'sh':
+ return mock_sh
+ elif name == 'appdirs':
+ return mock_appdirs
+ elif name == 'jinja2':
+ return mock_jinja2
+ raise ImportError(f"No module named '{name}'")
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', {
+ 'colorama': mock_colorama,
+ 'sh': mock_sh,
+ 'appdirs': mock_appdirs,
+ 'jinja2': mock_jinja2
+ }):
+ checkdependencies.check_python_dependencies()
+
+ @mock.patch('builtins.exit')
+ @mock.patch('builtins.print')
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_missing_module_without_version(self, mock_import, mock_print, mock_exit):
+ """Test error message when module without version requirement is missing."""
+ modules_dict = {}
+
+ def import_side_effect(name):
+ if name == 'appdirs':
+ raise ImportError(f"No module named '{name}'")
+ mock_mod = mock.Mock()
+ mock_mod.__version__ = '1.0'
+ modules_dict[name] = mock_mod
+ return mock_mod
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', modules_dict):
+ checkdependencies.check_python_dependencies()
+
+ # Verify error message was printed
+ error_calls = [str(call) for call in mock_print.call_args_list]
+ assert any('appdirs' in call and 'ERROR' in call for call in error_calls)
+ mock_exit.assert_called_once_with(1)
+
+ @mock.patch('builtins.exit')
+ @mock.patch('builtins.print')
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_missing_module_with_version(self, mock_import, mock_print, mock_exit):
+ """Test error message when module with version requirement is missing."""
+ modules_dict = {}
+
+ def import_side_effect(name):
+ if name == 'colorama':
+ raise ImportError(f"No module named '{name}'")
+ mock_mod = mock.Mock()
+ mock_mod.__version__ = '1.0'
+ modules_dict[name] = mock_mod
+ return mock_mod
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', modules_dict):
+ checkdependencies.check_python_dependencies()
+
+ # Verify error message includes version requirement
+ error_calls = [str(call) for call in mock_print.call_args_list]
+ assert any('colorama' in call and '0.3.3' in call for call in error_calls)
+ mock_exit.assert_called_once_with(1)
+
+ @mock.patch('builtins.exit')
+ @mock.patch('builtins.print')
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_module_version_too_old(self, mock_import, mock_print, mock_exit):
+ """Test error when module version is too old."""
+ mock_colorama = mock.Mock()
+ mock_colorama.__version__ = '0.2.0' # Too old, needs 0.3.3
+ modules_dict = {'colorama': mock_colorama}
+
+ def import_side_effect(name):
+ if name == 'colorama':
+ return mock_colorama
+ mock_mod = mock.Mock()
+ mock_mod.__version__ = '1.0'
+ modules_dict[name] = mock_mod
+ return mock_mod
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', modules_dict):
+ checkdependencies.check_python_dependencies()
+
+ # Verify error message about version
+ error_calls = [str(call) for call in mock_print.call_args_list]
+ assert any('version' in call.lower() and 'colorama' in call for call in error_calls)
+ mock_exit.assert_called_once_with(1)
+
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_module_version_acceptable(self, mock_import):
+ """Test that acceptable versions pass."""
+ mock_colorama = mock.Mock()
+ mock_colorama.__version__ = '0.4.0' # Newer than 0.3.3
+ mock_sh = mock.Mock()
+ mock_sh.__version__ = '1.12' # Newer than 1.10
+
+ def import_side_effect(name):
+ if name == 'colorama':
+ return mock_colorama
+ elif name == 'sh':
+ return mock_sh
+ mock_mod = mock.Mock()
+ return mock_mod
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', {
+ 'colorama': mock_colorama,
+ 'sh': mock_sh
+ }):
+ # Should complete without error
+ checkdependencies.check_python_dependencies()
+
+ @mock.patch('pythonforandroid.checkdependencies.import_module')
+ def test_module_without_version_attribute(self, mock_import):
+ """Test handling of modules that don't have __version__."""
+ mock_colorama = mock.Mock(spec=[]) # No __version__ attribute
+ modules_dict = {'colorama': mock_colorama}
+
+ def import_side_effect(name):
+ if name == 'colorama':
+ return mock_colorama
+ mock_mod = mock.Mock()
+ modules_dict[name] = mock_mod
+ return mock_mod
+
+ mock_import.side_effect = import_side_effect
+
+ with mock.patch.object(sys, 'modules', modules_dict):
+ # Should complete without error (version check is skipped)
+ checkdependencies.check_python_dependencies()
+
+
+class TestCheck:
+ """Test the main check() function."""
+
+ @mock.patch('pythonforandroid.checkdependencies.check_python_dependencies')
+ @mock.patch('pythonforandroid.checkdependencies.check_and_install_default_prerequisites')
+ def test_check_with_skip_prerequisites(self, mock_prereqs, mock_python_deps):
+ """Test check() skips prerequisites when SKIP_PREREQUISITES_CHECK=1."""
+ with mock.patch.dict('os.environ', {'SKIP_PREREQUISITES_CHECK': '1'}):
+ checkdependencies.check()
+
+ mock_prereqs.assert_not_called()
+ mock_python_deps.assert_called_once()
+
+ @mock.patch('pythonforandroid.checkdependencies.check_python_dependencies')
+ @mock.patch('pythonforandroid.checkdependencies.check_and_install_default_prerequisites')
+ def test_check_without_skip(self, mock_prereqs, mock_python_deps):
+ """Test check() runs prerequisites when SKIP_PREREQUISITES_CHECK is not set."""
+ with mock.patch.dict('os.environ', {}, clear=True):
+ checkdependencies.check()
+
+ mock_prereqs.assert_called_once()
+ mock_python_deps.assert_called_once()
+
+ @mock.patch('pythonforandroid.checkdependencies.check_python_dependencies')
+ @mock.patch('pythonforandroid.checkdependencies.check_and_install_default_prerequisites')
+ def test_check_with_skip_set_to_zero(self, mock_prereqs, mock_python_deps):
+ """Test check() runs prerequisites when SKIP_PREREQUISITES_CHECK=0."""
+ with mock.patch.dict('os.environ', {'SKIP_PREREQUISITES_CHECK': '0'}):
+ checkdependencies.check()
+
+ mock_prereqs.assert_called_once()
+ mock_python_deps.assert_called_once()
diff --git a/tests/test_patching.py b/tests/test_patching.py
new file mode 100644
index 0000000000..dd085f3402
--- /dev/null
+++ b/tests/test_patching.py
@@ -0,0 +1,326 @@
+from unittest import mock
+
+from pythonforandroid.patching import (
+ is_platform,
+ is_linux,
+ is_darwin,
+ is_windows,
+ is_arch,
+ is_api,
+ is_api_gt,
+ is_api_gte,
+ is_api_lt,
+ is_api_lte,
+ is_ndk,
+ is_version_gt,
+ is_version_lt,
+ version_starts_with,
+ will_build,
+ check_all,
+ check_any,
+)
+
+
+class TestPlatformChecks:
+ """Test platform detection functions."""
+
+ @mock.patch('pythonforandroid.patching.uname')
+ def test_is_platform_linux(self, mock_uname):
+ """Test is_platform returns check function for Linux."""
+ mock_uname.return_value = mock.Mock(system='Linux')
+ check_fn = is_platform('Linux')
+ assert check_fn(None, None)
+
+ @mock.patch('pythonforandroid.patching.uname')
+ def test_is_platform_darwin(self, mock_uname):
+ """Test is_platform returns check function for Darwin."""
+ mock_uname.return_value = mock.Mock(system='Darwin')
+ check_fn = is_platform('Darwin')
+ assert check_fn(None, None)
+
+ @mock.patch('pythonforandroid.patching.uname')
+ def test_is_platform_case_insensitive(self, mock_uname):
+ """Test is_platform is case insensitive."""
+ mock_uname.return_value = mock.Mock(system='LINUX')
+ check_fn = is_platform('linux')
+ assert check_fn(None, None)
+
+ @mock.patch('pythonforandroid.patching.uname')
+ def test_is_platform_mismatch(self, mock_uname):
+ """Test is_platform returns False for mismatched platform."""
+ mock_uname.return_value = mock.Mock(system='Linux')
+ check_fn = is_platform('Windows')
+ assert not check_fn(None, None)
+
+ def test_is_linux(self):
+ """Test is_linux constant function is defined."""
+ # is_linux is defined at module import time based on real platform
+ # We can only verify it's callable
+ assert callable(is_linux)
+
+ def test_is_darwin(self):
+ """Test is_darwin constant function is defined."""
+ # is_darwin is defined at module import time based on real platform
+ # We can only verify it's callable
+ assert callable(is_darwin)
+
+ def test_is_windows(self):
+ """Test is_windows constant function is defined."""
+ # is_windows is defined at module import time based on real platform
+ # We can only verify it's callable
+ assert callable(is_windows)
+
+
+class TestArchChecks:
+ """Test architecture check functions."""
+
+ def test_is_arch_match(self):
+ """Test is_arch returns True for matching architecture."""
+ mock_arch = mock.Mock(arch='armeabi-v7a')
+ check_fn = is_arch('armeabi-v7a')
+ assert check_fn(mock_arch)
+
+ def test_is_arch_mismatch(self):
+ """Test is_arch returns False for mismatched architecture."""
+ mock_arch = mock.Mock(arch='armeabi-v7a')
+ check_fn = is_arch('arm64-v8a')
+ assert not check_fn(mock_arch)
+
+
+class TestAndroidAPIChecks:
+ """Test Android API level comparison functions."""
+
+ def test_is_api_equal(self):
+ """Test is_api for equal API level."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 21
+ check_fn = is_api(21)
+ assert check_fn(None, mock_recipe)
+
+ def test_is_api_not_equal(self):
+ """Test is_api for unequal API level."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 21
+ check_fn = is_api(27)
+ assert not check_fn(None, mock_recipe)
+
+ def test_is_api_gt(self):
+ """Test is_api_gt for greater than comparison."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 27
+ check_fn = is_api_gt(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 21
+ assert not check_fn(None, mock_recipe)
+
+ def test_is_api_gte(self):
+ """Test is_api_gte for greater than or equal comparison."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 27
+ check_fn = is_api_gte(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 21
+ check_fn = is_api_gte(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 19
+ assert not check_fn(None, mock_recipe)
+
+ def test_is_api_lt(self):
+ """Test is_api_lt for less than comparison."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 19
+ check_fn = is_api_lt(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 21
+ assert not check_fn(None, mock_recipe)
+
+ def test_is_api_lte(self):
+ """Test is_api_lte for less than or equal comparison."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.android_api = 19
+ check_fn = is_api_lte(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 21
+ check_fn = is_api_lte(21)
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.ctx.android_api = 27
+ assert not check_fn(None, mock_recipe)
+
+
+class TestNDKChecks:
+ """Test NDK version check functions."""
+
+ def test_is_ndk_equal(self):
+ """Test is_ndk for equal NDK version."""
+ mock_ndk = mock.Mock(name='ndk_r21e')
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.ndk = mock_ndk
+ check_fn = is_ndk(mock_ndk)
+ assert check_fn(None, mock_recipe)
+
+ def test_is_ndk_not_equal(self):
+ """Test is_ndk for unequal NDK version."""
+ mock_ndk1 = mock.Mock(name='ndk_r21e')
+ mock_ndk2 = mock.Mock(name='ndk_r25c')
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.ndk = mock_ndk1
+ check_fn = is_ndk(mock_ndk2)
+ assert not check_fn(None, mock_recipe)
+
+
+class TestVersionChecks:
+ """Test recipe version comparison functions."""
+
+ def test_is_version_gt(self):
+ """Test is_version_gt for version comparison."""
+ mock_recipe = mock.Mock(version='2.0.0')
+ check_fn = is_version_gt('1.0.0')
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.version = '1.0.0'
+ assert not check_fn(None, mock_recipe)
+
+ def test_is_version_lt(self):
+ """Test is_version_lt for version comparison."""
+ mock_recipe = mock.Mock(version='1.0.0')
+ check_fn = is_version_lt('2.0.0')
+ assert check_fn(None, mock_recipe)
+
+ mock_recipe.version = '2.0.0'
+ assert not check_fn(None, mock_recipe)
+
+ def test_version_starts_with(self):
+ """Test version_starts_with for version prefix matching."""
+ mock_recipe = mock.Mock(version='1.15.2')
+ check_fn = version_starts_with('1.15')
+ assert check_fn(None, mock_recipe)
+
+ check_fn = version_starts_with('1.14')
+ assert not check_fn(None, mock_recipe)
+
+ check_fn = version_starts_with('2')
+ assert not check_fn(None, mock_recipe)
+
+
+class TestWillBuild:
+ """Test will_build function."""
+
+ def test_will_build_present(self):
+ """Test will_build returns True when recipe is in build order."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.recipe_build_order = ['python3', 'numpy', 'kivy']
+ check_fn = will_build('numpy')
+ assert check_fn(None, mock_recipe)
+
+ def test_will_build_absent(self):
+ """Test will_build returns False when recipe is not in build order."""
+ mock_recipe = mock.Mock()
+ mock_recipe.ctx.recipe_build_order = ['python3', 'numpy', 'kivy']
+ check_fn = will_build('scipy')
+ assert not check_fn(None, mock_recipe)
+
+
+class TestConjunctions:
+ """Test logical conjunction functions."""
+
+ def test_check_all_all_true(self):
+ """Test check_all returns True when all checks pass."""
+ def check1(_arch, _recipe):
+ return True
+
+ def check2(_arch, _recipe):
+ return True
+
+ def check3(_arch, _recipe):
+ return True
+
+ check_fn = check_all(check1, check2, check3)
+ assert check_fn(None, None)
+
+ def test_check_all_one_false(self):
+ """Test check_all returns False when one check fails."""
+ def check1(_arch, _recipe):
+ return True
+
+ def check2(_arch, _recipe):
+ return False
+
+ def check3(_arch, _recipe):
+ return True
+
+ check_fn = check_all(check1, check2, check3)
+ assert not check_fn(None, None)
+
+ def test_check_all_all_false(self):
+ """Test check_all returns False when all checks fail."""
+ def check1(_arch, _recipe):
+ return False
+
+ def check2(_arch, _recipe):
+ return False
+
+ check_fn = check_all(check1, check2)
+ assert not check_fn(None, None)
+
+ def test_check_any_one_true(self):
+ """Test check_any returns True when one check passes."""
+ def check1(_arch, _recipe):
+ return False
+
+ def check2(_arch, _recipe):
+ return True
+
+ def check3(_arch, _recipe):
+ return False
+
+ check_fn = check_any(check1, check2, check3)
+ assert check_fn(None, None)
+
+ def test_check_any_all_false(self):
+ """Test check_any returns False when all checks fail."""
+ def check1(_arch, _recipe):
+ return False
+
+ def check2(_arch, _recipe):
+ return False
+
+ check_fn = check_any(check1, check2)
+ assert not check_fn(None, None)
+
+ def test_check_any_all_true(self):
+ """Test check_any returns True when all checks pass."""
+ def check1(_arch, _recipe):
+ return True
+
+ def check2(_arch, _recipe):
+ return True
+
+ check_fn = check_any(check1, check2)
+ assert check_fn(None, None)
+
+ @mock.patch('pythonforandroid.patching.uname')
+ def test_combined_checks(self, mock_uname):
+ """Test combining multiple check functions with check_all and check_any."""
+ # Test check_all with is_platform and is_version_gt
+ mock_uname.return_value = mock.Mock(system='Linux')
+ mock_recipe = mock.Mock(version='2.0.0')
+
+ check_fn = check_all(
+ is_platform('Linux'),
+ is_version_gt('1.0.0')
+ )
+ assert check_fn(None, mock_recipe)
+
+ # Test check_any with is_platform and is_version_gt
+ mock_uname.return_value = mock.Mock(system='Windows')
+ check_fn = check_any(
+ is_platform('Linux'),
+ is_version_gt('1.0.0')
+ )
+ assert check_fn(None, mock_recipe)
From 17ca5f215352d9970f830145f4b84b2cec1f3b94 Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sat, 1 Nov 2025 22:26:41 +0000
Subject: [PATCH 155/158] :white_check_mark: Tests for entrypoints, bdistapk,
and util modules
Add test coverage for three more modules:
- entrypoints.py: 4 tests covering main() exception handling
(coverage: 33% -> 100%)
- bdistapk.py: 12 tests covering setuptools command handlers
(coverage: 0% -> 86%)
- util.py: 5 tests for load_source(), rmdir(), and
patch_wheel_setuptools_logging() (coverage: -> 95%)
---
tests/test_bdistapk.py | 200 ++++++++++++++++++++++++++++++++++++++
tests/test_entrypoints.py | 63 ++++++++++++
tests/test_util.py | 67 +++++++++++++
3 files changed, 330 insertions(+)
create mode 100644 tests/test_bdistapk.py
create mode 100644 tests/test_entrypoints.py
diff --git a/tests/test_bdistapk.py b/tests/test_bdistapk.py
new file mode 100644
index 0000000000..54d743ff18
--- /dev/null
+++ b/tests/test_bdistapk.py
@@ -0,0 +1,200 @@
+import sys
+from unittest import mock
+from setuptools.dist import Distribution
+
+from pythonforandroid.bdistapk import (
+ argv_contains,
+ BdistAPK,
+ BdistAAR,
+ BdistAAB,
+)
+
+
+class TestArgvContains:
+ """Test argv_contains helper function."""
+
+ def test_argv_contains_present(self):
+ """Test argv_contains returns True when argument is present."""
+ with mock.patch.object(sys, 'argv', ['prog', '--name=test', '--version=1.0']):
+ assert argv_contains('--name')
+ assert argv_contains('--version')
+
+ def test_argv_contains_partial_match(self):
+ """Test argv_contains returns True for partial matches."""
+ with mock.patch.object(sys, 'argv', ['prog', '--name=test']):
+ assert argv_contains('--name')
+ assert argv_contains('--nam')
+
+ def test_argv_contains_not_present(self):
+ """Test argv_contains returns False when argument is not present."""
+ with mock.patch.object(sys, 'argv', ['prog', '--name=test']):
+ assert not argv_contains('--package')
+ assert not argv_contains('--arch')
+
+
+class TestBdist:
+ """Test Bdist base class."""
+
+ def setup_method(self):
+ """Set up test fixtures."""
+ self.distribution = Distribution({
+ 'name': 'TestApp',
+ 'version': '1.0.0',
+ })
+ self.distribution.package_data = {'testapp': ['*.py', '*.kv']}
+
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ def test_initialize_options(self, mock_rmdir, mock_ensure_dir):
+ """Test initialize_options sets attributes from user_options."""
+ bdist = BdistAPK(self.distribution)
+ bdist.user_options = [('name=', None, None), ('version=', None, None)]
+
+ bdist.initialize_options()
+
+ assert hasattr(bdist, 'name')
+ assert hasattr(bdist, 'version')
+
+ @mock.patch('pythonforandroid.bdistapk.argv_contains')
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ def test_finalize_options_injects_defaults(
+ self, mock_rmdir, mock_ensure_dir, mock_argv_contains
+ ):
+ """Test finalize_options injects default name, package, version, arch."""
+ mock_argv_contains.return_value = False
+
+ with mock.patch.object(sys, 'argv', ['setup.py', 'apk']):
+ bdist = BdistAPK(self.distribution)
+ bdist.finalize_options()
+
+ # Check that defaults were added to sys.argv
+ argv_str = ' '.join(sys.argv)
+ assert '--name=' in argv_str or any('--name' in arg for arg in sys.argv)
+
+ @mock.patch('pythonforandroid.bdistapk.argv_contains')
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ def test_finalize_options_permissions_handling(
+ self, mock_rmdir, mock_ensure_dir, mock_argv_contains
+ ):
+ """Test finalize_options handles permissions list correctly."""
+ mock_argv_contains.side_effect = lambda x: x != '--permissions'
+
+ # Set up permissions in the distribution command options
+ self.distribution.command_options['apk'] = {
+ 'permissions': ('setup.py', ['INTERNET', 'CAMERA'])
+ }
+
+ with mock.patch.object(sys, 'argv', ['setup.py', 'apk']):
+ bdist = BdistAPK(self.distribution)
+ bdist.package_type = 'apk'
+ bdist.finalize_options()
+
+ # Check permissions were added
+ assert any('--permission=INTERNET' in arg for arg in sys.argv)
+ assert any('--permission=CAMERA' in arg for arg in sys.argv)
+
+ @mock.patch('pythonforandroid.entrypoints.main')
+ @mock.patch('pythonforandroid.bdistapk.argv_contains')
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ @mock.patch('pythonforandroid.bdistapk.copyfile')
+ @mock.patch('pythonforandroid.bdistapk.glob')
+ def test_run_calls_main(
+ self, mock_glob, mock_copyfile, mock_rmdir, mock_ensure_dir,
+ mock_argv_contains, mock_main
+ ):
+ """Test run() calls prepare_build_dir and then main()."""
+ mock_glob.return_value = ['testapp/main.py']
+ mock_argv_contains.return_value = False # Not using --launcher or --private
+
+ with mock.patch.object(sys, 'argv', ['setup.py', 'apk']):
+ bdist = BdistAPK(self.distribution)
+ bdist.arch = 'armeabi-v7a'
+ bdist.run()
+
+ mock_rmdir.assert_called()
+ mock_ensure_dir.assert_called()
+ mock_main.assert_called_once()
+ assert sys.argv[1] == 'apk'
+
+ @mock.patch('pythonforandroid.bdistapk.argv_contains')
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ @mock.patch('pythonforandroid.bdistapk.copyfile')
+ @mock.patch('pythonforandroid.bdistapk.glob')
+ @mock.patch('builtins.exit', side_effect=SystemExit(1))
+ def test_prepare_build_dir_no_main_py(
+ self, mock_exit, mock_glob, mock_copyfile,
+ mock_rmdir, mock_ensure_dir, mock_argv_contains
+ ):
+ """Test prepare_build_dir exits if no main.py found and not using launcher."""
+ mock_glob.return_value = ['testapp/helper.py']
+ mock_argv_contains.return_value = False # Not using --launcher
+
+ bdist = BdistAPK(self.distribution)
+ bdist.arch = 'armeabi-v7a'
+
+ # Expect SystemExit to be raised
+ try:
+ bdist.prepare_build_dir()
+ assert False, "Expected SystemExit to be raised"
+ except SystemExit:
+ pass
+
+ mock_exit.assert_called_once_with(1)
+
+ @mock.patch('pythonforandroid.bdistapk.argv_contains')
+ @mock.patch('pythonforandroid.bdistapk.ensure_dir')
+ @mock.patch('pythonforandroid.bdistapk.rmdir')
+ @mock.patch('pythonforandroid.bdistapk.copyfile')
+ @mock.patch('pythonforandroid.bdistapk.glob')
+ def test_prepare_build_dir_with_main_py(
+ self, mock_glob, mock_copyfile, mock_rmdir,
+ mock_ensure_dir, mock_argv_contains
+ ):
+ """Test prepare_build_dir succeeds when main.py is found."""
+ mock_glob.return_value = ['testapp/main.py', 'testapp/helper.py']
+ # Return False for all argv_contains checks (no --launcher, no --private)
+ mock_argv_contains.return_value = False
+
+ with mock.patch.object(sys, 'argv', ['setup.py', 'apk']):
+ bdist = BdistAPK(self.distribution)
+ bdist.arch = 'armeabi-v7a'
+ bdist.prepare_build_dir()
+
+ # Should have copied files (glob might return duplicates)
+ assert mock_copyfile.call_count >= 2
+ # Should have added --private argument
+ assert any('--private=' in arg for arg in sys.argv)
+
+
+class TestBdistSubclasses:
+ """Test BdistAPK, BdistAAR, BdistAAB subclasses."""
+
+ def setup_method(self):
+ """Set up test fixtures."""
+ self.distribution = Distribution({
+ 'name': 'TestApp',
+ 'version': '1.0.0',
+ })
+ self.distribution.package_data = {}
+
+ def test_bdist_apk_package_type(self):
+ """Test BdistAPK has correct package_type."""
+ bdist = BdistAPK(self.distribution)
+ assert bdist.package_type == 'apk'
+ assert bdist.description == 'Create an APK with python-for-android'
+
+ def test_bdist_aar_package_type(self):
+ """Test BdistAAR has correct package_type."""
+ bdist = BdistAAR(self.distribution)
+ assert bdist.package_type == 'aar'
+ assert bdist.description == 'Create an AAR with python-for-android'
+
+ def test_bdist_aab_package_type(self):
+ """Test BdistAAB has correct package_type."""
+ bdist = BdistAAB(self.distribution)
+ assert bdist.package_type == 'aab'
+ assert bdist.description == 'Create an AAB with python-for-android'
diff --git a/tests/test_entrypoints.py b/tests/test_entrypoints.py
new file mode 100644
index 0000000000..66b68d6b3a
--- /dev/null
+++ b/tests/test_entrypoints.py
@@ -0,0 +1,63 @@
+from unittest import mock
+
+from pythonforandroid.entrypoints import main
+from pythonforandroid.util import BuildInterruptingException
+
+
+class TestMain:
+ """Test the main entry point function."""
+
+ @mock.patch('pythonforandroid.toolchain.ToolchainCL')
+ @mock.patch('pythonforandroid.entrypoints.check_python_version')
+ def test_main_success(self, mock_check_version, mock_toolchain):
+ """Test main() executes successfully with valid Python version."""
+ main()
+
+ mock_check_version.assert_called_once()
+ mock_toolchain.assert_called_once()
+
+ @mock.patch('pythonforandroid.entrypoints.handle_build_exception')
+ @mock.patch('pythonforandroid.toolchain.ToolchainCL')
+ @mock.patch('pythonforandroid.entrypoints.check_python_version')
+ def test_main_build_interrupting_exception(
+ self, mock_check_version, mock_toolchain, mock_handler
+ ):
+ """Test main() catches BuildInterruptingException and handles it."""
+ exc = BuildInterruptingException("Build failed", "Try reinstalling")
+ mock_toolchain.side_effect = exc
+
+ main()
+
+ mock_check_version.assert_called_once()
+ mock_toolchain.assert_called_once()
+ mock_handler.assert_called_once_with(exc)
+
+ @mock.patch('pythonforandroid.toolchain.ToolchainCL')
+ @mock.patch('pythonforandroid.entrypoints.check_python_version')
+ def test_main_other_exception_propagates(
+ self, mock_check_version, mock_toolchain
+ ):
+ """Test main() allows non-BuildInterruptingException to propagate."""
+ mock_toolchain.side_effect = RuntimeError("Unexpected error")
+
+ try:
+ main()
+ assert False, "Expected RuntimeError to be raised"
+ except RuntimeError as e:
+ assert str(e) == "Unexpected error"
+
+ mock_check_version.assert_called_once()
+ mock_toolchain.assert_called_once()
+
+ @mock.patch('pythonforandroid.entrypoints.check_python_version')
+ def test_main_python_version_check_fails(self, mock_check_version):
+ """Test main() allows Python version check failure to propagate."""
+ mock_check_version.side_effect = SystemExit(1)
+
+ try:
+ main()
+ assert False, "Expected SystemExit to be raised"
+ except SystemExit as e:
+ assert e.code == 1
+
+ mock_check_version.assert_called_once()
diff --git a/tests/test_util.py b/tests/test_util.py
index 7a60bc73fb..744e17132e 100644
--- a/tests/test_util.py
+++ b/tests/test_util.py
@@ -230,3 +230,70 @@ def test_max_build_tool_version(self):
result = util.max_build_tool_version(build_tools_versions)
self.assertEqual(result, expected_result)
+
+ def test_load_source(self):
+ """
+ Test method :meth:`~pythonforandroid.util.load_source`.
+ We test loading a Python module from a file path using importlib.
+ """
+ with TemporaryDirectory() as temp_dir:
+ # Create a test module file
+ test_module_path = Path(temp_dir) / "test_module.py"
+ with open(test_module_path, "w") as f:
+ f.write("TEST_VALUE = 42\n")
+ f.write("def test_function():\n")
+ f.write(" return 'hello'\n")
+
+ # Load the module
+ loaded_module = util.load_source("test_module", str(test_module_path))
+
+ # Verify the module was loaded correctly
+ self.assertEqual(loaded_module.TEST_VALUE, 42)
+ self.assertEqual(loaded_module.test_function(), 'hello')
+
+ @mock.patch("pythonforandroid.util.exists")
+ @mock.patch("shutil.rmtree")
+ def test_rmdir_exists(self, mock_rmtree, mock_exists):
+ """
+ Test method :meth:`~pythonforandroid.util.rmdir` when directory exists.
+ We mock exists to return True and verify rmtree is called.
+ """
+ mock_exists.return_value = True
+ util.rmdir("/fake/directory")
+ mock_rmtree.assert_called_once_with("/fake/directory", False)
+
+ @mock.patch("pythonforandroid.util.exists")
+ @mock.patch("shutil.rmtree")
+ def test_rmdir_not_exists(self, mock_rmtree, mock_exists):
+ """
+ Test method :meth:`~pythonforandroid.util.rmdir` when directory doesn't exist.
+ We mock exists to return False and verify rmtree is not called.
+ """
+ mock_exists.return_value = False
+ util.rmdir("/fake/directory")
+ mock_rmtree.assert_not_called()
+
+ @mock.patch("pythonforandroid.util.exists")
+ @mock.patch("shutil.rmtree")
+ def test_rmdir_ignore_errors(self, mock_rmtree, mock_exists):
+ """
+ Test method :meth:`~pythonforandroid.util.rmdir` with ignore_errors flag.
+ We verify that the ignore_errors parameter is passed to rmtree.
+ """
+ mock_exists.return_value = True
+ util.rmdir("/fake/directory", ignore_errors=True)
+ mock_rmtree.assert_called_once_with("/fake/directory", True)
+
+ @mock.patch("pythonforandroid.util.mock")
+ def test_patch_wheel_setuptools_logging(self, mock_mock):
+ """
+ Test method :meth:`~pythonforandroid.util.patch_wheel_setuptools_logging`.
+ We verify it returns a mock.patch object for the wheel logging module.
+ """
+ mock_patch_obj = mock.Mock()
+ mock_mock.patch.return_value = mock_patch_obj
+
+ result = util.patch_wheel_setuptools_logging()
+
+ mock_mock.patch.assert_called_once_with("wheel._setuptools_logging.configure")
+ self.assertEqual(result, mock_patch_obj)
From 1c413d1f7c82c500259b451ff547d619648b53a6 Mon Sep 17 00:00:00 2001
From: Ansh Dadwal <anshdadwal298@gmail.com>
Date: Sun, 2 Nov 2025 16:35:03 +0530
Subject: [PATCH 156/158] `materialyoucolor`: shift to `PPR` (#3251)
---
.../recipes/materialyoucolor/__init__.py | 12 ++++++++----
1 file changed, 8 insertions(+), 4 deletions(-)
diff --git a/pythonforandroid/recipes/materialyoucolor/__init__.py b/pythonforandroid/recipes/materialyoucolor/__init__.py
index 32d44a2714..abbc2d7b39 100644
--- a/pythonforandroid/recipes/materialyoucolor/__init__.py
+++ b/pythonforandroid/recipes/materialyoucolor/__init__.py
@@ -1,11 +1,15 @@
-from pythonforandroid.recipe import CppCompiledComponentsPythonRecipe
+from pythonforandroid.recipe import PyProjectRecipe
-class MaterialyoucolorRecipe(CppCompiledComponentsPythonRecipe):
+class MaterialyoucolorRecipe(PyProjectRecipe):
stl_lib_name = "c++_shared"
- version = "2.0.9"
+ version = "2.0.10"
url = "https://github.com/T-Dynamos/materialyoucolor-python/releases/download/v{version}/materialyoucolor-{version}.tar.gz"
- depends = ["setuptools"]
+
+ def get_recipe_env(self, arch, **kwargs):
+ env = super().get_recipe_env(arch, **kwargs)
+ env['LDCXXSHARED'] = env['CXX'] + ' -shared'
+ return env
recipe = MaterialyoucolorRecipe()
From 62e8089e562f2732e9786c7744e7c72e5b2bca7e Mon Sep 17 00:00:00 2001
From: Andre Miras <andre.miras@gmail.com>
Date: Sun, 2 Nov 2025 17:09:35 +0000
Subject: [PATCH 157/158] :white_check_mark: Increase test coverage for logger,
prerequisites, and pythonpackage
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
Add comprehensive tests to improve coverage:
Logger module (66% → 86%):
- Color setup and configuration (never/always/auto modes)
- Utility functions (shorten_string, get_console_width)
- LevelDifferentiatingFormatter for all log levels
- shprint error handling with filters and critical failures
- Logging helpers (info_main, info_notify)
Prerequisites module (45% → 80%):
- Base Prerequisite class methods (is_valid, checker)
- Installation workflow (ask_to_install, install)
- JDK version checking and JAVA_HOME support
- Homebrew formula location helpers
- Main check_and_install workflow
Pythonpackage module:
- Parametrized tests for parse_as_folder_reference edge cases
- Filesystem path detection for relative paths and git URLs
- Dependency transformation with query params and fragments
- Error handling for package extraction and invalid metadata
---
tests/test_logger.py | 222 +++++++++++++++++++++-
tests/test_prerequisites.py | 303 ++++++++++++++++++++++++++++++
tests/test_pythonpackage_basic.py | 89 +++++++++
3 files changed, 613 insertions(+), 1 deletion(-)
diff --git a/tests/test_logger.py b/tests/test_logger.py
index 773e7e54a0..aa739ff5d5 100644
--- a/tests/test_logger.py
+++ b/tests/test_logger.py
@@ -1,8 +1,228 @@
+import logging
+import sh
+import pytest
import unittest
-from unittest.mock import MagicMock
+from unittest.mock import MagicMock, Mock, patch
from pythonforandroid import logger
+class TestColorSetup:
+ """Test color setup and configuration."""
+
+ def teardown_method(self):
+ """Reset color state after each test to avoid affecting other tests."""
+ logger.setup_color('never')
+
+ def test_setup_color_never(self):
+ """Test color disabled when set to 'never'."""
+ logger.setup_color('never')
+ assert not logger.Out_Style._enabled
+ assert not logger.Out_Fore._enabled
+ assert not logger.Err_Style._enabled
+ assert not logger.Err_Fore._enabled
+
+ def test_setup_color_always(self):
+ """Test color enabled when set to 'always'."""
+ logger.setup_color('always')
+ assert logger.Out_Style._enabled
+ assert logger.Out_Fore._enabled
+ assert logger.Err_Style._enabled
+ assert logger.Err_Fore._enabled
+
+ @patch('pythonforandroid.logger.stdout')
+ @patch('pythonforandroid.logger.stderr')
+ def test_setup_color_auto_with_tty(self, mock_stderr, mock_stdout):
+ """Test color enabled when auto and isatty() returns True."""
+ mock_stdout.isatty.return_value = True
+ mock_stderr.isatty.return_value = True
+ logger.setup_color('auto')
+ assert logger.Out_Style._enabled
+ assert logger.Err_Style._enabled
+
+
+class TestUtilityFunctions:
+ """Test logger utility functions."""
+
+ def test_shorten_string_short(self):
+ """Test shorten_string returns string unchanged when under limit."""
+ result = logger.shorten_string("short", 50)
+ assert result == "short"
+
+ def test_shorten_string_long(self):
+ """Test shorten_string truncates long strings correctly."""
+ long_string = "a" * 100
+ result = logger.shorten_string(long_string, 50)
+ assert "...(and" in result
+ assert "more)" in result
+ assert len(result) <= 50
+
+ def test_shorten_string_bytes(self):
+ """Test shorten_string handles bytes input."""
+ byte_string = b"test" * 50
+ result = logger.shorten_string(byte_string, 50)
+ assert "...(and" in result
+
+ @patch.dict('os.environ', {'COLUMNS': '120'})
+ def test_get_console_width_from_env(self):
+ """Test get_console_width reads from COLUMNS env var."""
+ width = logger.get_console_width()
+ assert width == 120
+
+ @patch.dict('os.environ', {}, clear=True)
+ @patch('os.popen')
+ def test_get_console_width_from_stty(self, mock_popen):
+ """Test get_console_width falls back to stty command."""
+ mock_popen.return_value.read.return_value = "40 80"
+ width = logger.get_console_width()
+ assert width == 80
+ mock_popen.assert_called_once_with('stty size', 'r')
+
+ @patch.dict('os.environ', {}, clear=True)
+ @patch('os.popen')
+ def test_get_console_width_default(self, mock_popen):
+ """Test get_console_width returns default when stty fails."""
+ mock_popen.return_value.read.side_effect = Exception("stty failed")
+ width = logger.get_console_width()
+ assert width == 100
+
+
+class TestLevelDifferentiatingFormatter:
+ """Test custom log message formatter."""
+
+ def test_format_error_level(self):
+ """Test formatter adds [ERROR] prefix for ERROR level."""
+ formatter = logger.LevelDifferentiatingFormatter('%(message)s')
+ record = logging.LogRecord(
+ name='test', level=40, pathname='', lineno=0,
+ msg='test error', args=(), exc_info=None
+ )
+ formatted = formatter.format(record)
+ assert '[ERROR]' in formatted
+
+ def test_format_warning_level(self):
+ """Test formatter adds [WARNING] prefix for WARNING level."""
+ formatter = logger.LevelDifferentiatingFormatter('%(message)s')
+ record = logging.LogRecord(
+ name='test', level=30, pathname='', lineno=0,
+ msg='test warning', args=(), exc_info=None
+ )
+ formatted = formatter.format(record)
+ assert '[WARNING]' in formatted
+
+ def test_format_info_level(self):
+ """Test formatter adds [INFO] prefix for INFO level."""
+ formatter = logger.LevelDifferentiatingFormatter('%(message)s')
+ record = logging.LogRecord(
+ name='test', level=20, pathname='', lineno=0,
+ msg='test info', args=(), exc_info=None
+ )
+ formatted = formatter.format(record)
+ assert '[INFO]' in formatted
+
+ def test_format_debug_level(self):
+ """Test formatter adds [DEBUG] prefix for DEBUG level."""
+ formatter = logger.LevelDifferentiatingFormatter('%(message)s')
+ record = logging.LogRecord(
+ name='test', level=10, pathname='', lineno=0,
+ msg='test debug', args=(), exc_info=None
+ )
+ formatted = formatter.format(record)
+ assert '[DEBUG]' in formatted
+
+
+class TestShprintErrorHandling:
+ """Test shprint error handling and edge cases."""
+
+ @patch('pythonforandroid.logger.get_console_width')
+ def test_shprint_with_filter(self, mock_width):
+ """Test shprint filters output with _filter parameter."""
+ mock_width.return_value = 100
+
+ command = MagicMock()
+ # Create a mock error with required attributes
+ error = Mock(spec=sh.ErrorReturnCode)
+ error.stdout = b'line1\nfiltered_line\nline3'
+ error.stderr = b''
+ command.side_effect = error
+
+ with pytest.raises(TypeError):
+ logger.shprint(command, _filter='filtered', _tail=10)
+
+ @patch('pythonforandroid.logger.get_console_width')
+ def test_shprint_with_filterout(self, mock_width):
+ """Test shprint excludes output with _filterout parameter."""
+ mock_width.return_value = 100
+
+ command = MagicMock()
+ error = Mock(spec=sh.ErrorReturnCode)
+ error.stdout = b'keep1\nexclude_line\nkeep2'
+ error.stderr = b''
+ command.side_effect = error
+
+ with pytest.raises(TypeError):
+ logger.shprint(command, _filterout='exclude', _tail=10)
+
+ @patch('pythonforandroid.logger.get_console_width')
+ @patch('pythonforandroid.logger.stdout')
+ @patch.dict('os.environ', {'P4A_FULL_DEBUG': '1'})
+ def test_shprint_full_debug_mode(self, mock_stdout, mock_width):
+ """Test shprint in P4A_FULL_DEBUG mode shows all output."""
+ mock_width.return_value = 100
+
+ command = MagicMock()
+ command.return_value = iter(['debug line 1\n', 'debug line 2\n'])
+
+ logger.shprint(command)
+ # In full debug mode, output is written directly to stdout
+ assert mock_stdout.write.called
+
+ @patch('pythonforandroid.logger.get_console_width')
+ @patch.dict('os.environ', {}, clear=True)
+ def test_shprint_critical_failure_exits(self, mock_width):
+ """Test shprint exits on critical command failure."""
+ mock_width.return_value = 100
+
+ command = MagicMock()
+
+ # Create a proper exception class that mimics sh.ErrorReturnCode
+ class MockErrorReturnCode(sh.ErrorReturnCode):
+ def __init__(self):
+ self.full_cmd = 'test'
+ self.stdout = b'output'
+ self.stderr = b'error'
+ self.exit_code = 1
+
+ error = MockErrorReturnCode()
+ command.side_effect = error
+
+ with patch('pythonforandroid.logger.exit', side_effect=SystemExit) as mock_exit:
+ with pytest.raises(SystemExit):
+ logger.shprint(command, _critical=True, _tail=5)
+ mock_exit.assert_called_once_with(1)
+
+
+class TestLoggingHelpers:
+ """Test logging helper functions."""
+
+ @patch('pythonforandroid.logger.logger')
+ def test_info_main(self, mock_logger):
+ """Test info_main logs with bright green formatting."""
+ logger.info_main('test', 'message')
+ mock_logger.info.assert_called_once()
+ # Verify the call contains color codes and text
+ call_args = mock_logger.info.call_args[0][0]
+ assert 'test' in call_args
+ assert 'message' in call_args
+
+ @patch('pythonforandroid.logger.info')
+ def test_info_notify(self, mock_info):
+ """Test info_notify logs with blue formatting."""
+ logger.info_notify('notification')
+ mock_info.assert_called_once()
+ call_args = mock_info.call_args[0][0]
+ assert 'notification' in call_args
+
+
class TestShprint(unittest.TestCase):
def test_unicode_encode(self):
diff --git a/tests/test_prerequisites.py b/tests/test_prerequisites.py
index 8d577fde1c..9d8bb071f2 100644
--- a/tests/test_prerequisites.py
+++ b/tests/test_prerequisites.py
@@ -2,8 +2,10 @@
from unittest import mock, skipIf
import sys
+import pytest
from pythonforandroid.prerequisites import (
+ Prerequisite,
JDKPrerequisite,
HomebrewPrerequisite,
OpenSSLPrerequisite,
@@ -13,6 +15,7 @@
PkgConfigPrerequisite,
CmakePrerequisite,
get_required_prerequisites,
+ check_and_install_default_prerequisites,
)
@@ -300,3 +303,303 @@ def test_default_linux_prerequisites_set(self):
[
],
)
+
+
+class TestPrerequisiteBaseClass:
+ """Test base Prerequisite class methods."""
+
+ @mock.patch('pythonforandroid.prerequisites.info')
+ @mock.patch.object(Prerequisite, 'checker')
+ def test_is_valid_when_met(self, mock_checker, mock_info):
+ """Test is_valid returns True when prerequisite is met."""
+ mock_checker.return_value = True
+ prerequisite = Prerequisite()
+ result = prerequisite.is_valid()
+ assert result == (True, "")
+ mock_info.assert_called()
+ assert "is met" in mock_info.call_args[0][0]
+
+ @mock.patch('pythonforandroid.prerequisites.warning')
+ @mock.patch.object(Prerequisite, 'checker')
+ def test_is_valid_when_not_met_non_mandatory(self, mock_checker, mock_warning):
+ """Test is_valid warns when non-mandatory prerequisite not met."""
+ mock_checker.return_value = False
+ prerequisite = Prerequisite()
+ prerequisite.mandatory = dict(linux=False, darwin=False)
+
+ result = prerequisite.is_valid()
+ assert result is None
+ mock_warning.assert_called()
+ assert "not met" in mock_warning.call_args[0][0]
+
+ @mock.patch('pythonforandroid.prerequisites.error')
+ @mock.patch.object(Prerequisite, 'checker')
+ @mock.patch('sys.platform', 'linux')
+ def test_is_valid_when_not_met_mandatory(self, mock_checker, mock_error):
+ """Test is_valid errors when mandatory prerequisite not met."""
+ mock_checker.return_value = False
+ prerequisite = Prerequisite()
+ prerequisite.mandatory = dict(linux=True, darwin=False)
+
+ result = prerequisite.is_valid()
+ assert result is None
+ mock_error.assert_called()
+ assert "not met" in mock_error.call_args[0][0]
+
+ @mock.patch('sys.platform', 'linux')
+ @mock.patch.object(Prerequisite, 'linux_checker')
+ def test_checker_calls_linux_checker(self, mock_linux_checker):
+ """Test checker dispatches to linux_checker on Linux."""
+ mock_linux_checker.return_value = True
+ prerequisite = Prerequisite()
+ result = prerequisite.checker()
+ assert result is True
+ mock_linux_checker.assert_called_once()
+
+ @mock.patch('sys.platform', 'darwin')
+ @mock.patch.object(Prerequisite, 'darwin_checker')
+ def test_checker_calls_darwin_checker(self, mock_darwin_checker):
+ """Test checker dispatches to darwin_checker on macOS."""
+ mock_darwin_checker.return_value = True
+ prerequisite = Prerequisite()
+ result = prerequisite.checker()
+ assert result is True
+ mock_darwin_checker.assert_called_once()
+
+ @mock.patch('sys.platform', 'win32')
+ def test_checker_raises_on_unsupported_platform(self):
+ """Test checker raises exception on unsupported platform."""
+ prerequisite = Prerequisite()
+ with pytest.raises(Exception, match="Unsupported platform"):
+ prerequisite.checker()
+
+
+class TestPrerequisiteInstallation:
+ """Test prerequisite installation workflow."""
+
+ @mock.patch.dict('os.environ', {'PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE': '1'})
+ @mock.patch('builtins.input')
+ def test_ask_to_install_user_accepts(self, mock_input):
+ """Test ask_to_install returns True when user enters 'y'."""
+ prerequisite = Prerequisite()
+ prerequisite.name = "TestPrerequisite"
+ mock_input.return_value = 'y'
+ result = prerequisite.ask_to_install()
+ assert result is True
+
+ @mock.patch.dict('os.environ', {'PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE': '1'})
+ @mock.patch('builtins.input')
+ def test_ask_to_install_user_declines(self, mock_input):
+ """Test ask_to_install returns False when user enters 'n'."""
+ prerequisite = Prerequisite()
+ prerequisite.name = "TestPrerequisite"
+ mock_input.return_value = 'n'
+ result = prerequisite.ask_to_install()
+ assert result is False
+
+ @mock.patch.dict('os.environ', {'PYTHONFORANDROID_PREREQUISITES_INSTALL_INTERACTIVE': '0'})
+ @mock.patch('pythonforandroid.prerequisites.info')
+ def test_ask_to_install_non_interactive(self, mock_info):
+ """Test ask_to_install returns True in non-interactive mode (CI)."""
+ prerequisite = Prerequisite()
+ prerequisite.name = "TestPrerequisite"
+ result = prerequisite.ask_to_install()
+ assert result is True
+ mock_info.assert_called()
+ assert "not interactive" in mock_info.call_args[0][0]
+
+ @mock.patch('sys.platform', 'linux')
+ @mock.patch.object(Prerequisite, 'ask_to_install')
+ @mock.patch.object(Prerequisite, 'linux_installer')
+ @mock.patch('pythonforandroid.prerequisites.info')
+ def test_install_when_user_accepts_linux(self, mock_info, mock_installer, mock_ask):
+ """Test install calls linux_installer when user accepts on Linux."""
+ prerequisite = Prerequisite()
+ prerequisite.installer_is_supported = dict(linux=True, darwin=True)
+ mock_ask.return_value = True
+ prerequisite.install()
+ mock_installer.assert_called_once()
+
+ @mock.patch('sys.platform', 'darwin')
+ @mock.patch.object(Prerequisite, 'ask_to_install')
+ @mock.patch.object(Prerequisite, 'darwin_installer')
+ def test_install_when_user_accepts_darwin(self, mock_installer, mock_ask):
+ """Test install calls darwin_installer when user accepts on macOS."""
+ prerequisite = Prerequisite()
+ prerequisite.installer_is_supported = dict(linux=True, darwin=True)
+ mock_ask.return_value = True
+ prerequisite.install()
+ mock_installer.assert_called_once()
+
+ @mock.patch.object(Prerequisite, 'ask_to_install')
+ @mock.patch('pythonforandroid.prerequisites.info')
+ def test_install_when_user_declines(self, mock_info, mock_ask):
+ """Test install skips installation when user declines."""
+ prerequisite = Prerequisite()
+ prerequisite.name = "TestPrerequisite"
+ mock_ask.return_value = False
+ prerequisite.install()
+ mock_info.assert_called()
+ assert "Skipping" in mock_info.call_args[0][0]
+
+ def test_install_is_supported(self):
+ """Test install_is_supported returns correct platform support."""
+ prerequisite = Prerequisite()
+ prerequisite.installer_is_supported = dict(linux=True, darwin=False)
+ with mock.patch('sys.platform', 'linux'):
+ assert prerequisite.install_is_supported() is True
+
+
+class TestJDKPrerequisiteVersionChecking:
+ """Test JDK version checking logic."""
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.Popen')
+ @mock.patch('os.path.exists')
+ def test_darwin_jdk_is_supported_valid_version(self, mock_exists, mock_popen):
+ """Test _darwin_jdk_is_supported returns True for valid JDK 17."""
+ prerequisite = JDKPrerequisite()
+ mock_exists.return_value = True
+
+ # Mock javac version output
+ mock_process = mock.Mock()
+ mock_process.returncode = 0
+ mock_process.communicate.return_value = (b'javac 17.0.2\n', b'')
+ mock_popen.return_value = mock_process
+
+ result = prerequisite._darwin_jdk_is_supported('/path/to/jdk')
+ assert result is True
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.Popen')
+ @mock.patch('os.path.exists')
+ def test_darwin_jdk_is_supported_invalid_version(self, mock_exists, mock_popen):
+ """Test _darwin_jdk_is_supported returns False for wrong JDK version."""
+ prerequisite = JDKPrerequisite()
+ mock_exists.return_value = True
+
+ mock_process = mock.Mock()
+ mock_process.returncode = 0
+ mock_process.communicate.return_value = (b'javac 11.0.1\n', b'')
+ mock_popen.return_value = mock_process
+
+ result = prerequisite._darwin_jdk_is_supported('/path/to/jdk')
+ assert result is False
+
+ @mock.patch('os.path.exists')
+ def test_darwin_jdk_is_supported_no_javac(self, mock_exists):
+ """Test _darwin_jdk_is_supported returns False when javac doesn't exist."""
+ prerequisite = JDKPrerequisite()
+ mock_exists.return_value = False
+ result = prerequisite._darwin_jdk_is_supported('/path/to/jdk')
+ assert result is False
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.run')
+ def test_darwin_get_libexec_jdk_path(self, mock_run):
+ """Test _darwin_get_libexec_jdk_path calls java_home correctly."""
+ prerequisite = JDKPrerequisite()
+ mock_run.return_value = mock.Mock(stdout=b'/Library/Java/JDK/17\n')
+
+ result = prerequisite._darwin_get_libexec_jdk_path(version='17')
+ assert result == '/Library/Java/JDK/17'
+ mock_run.assert_called_once()
+ assert '-v' in mock_run.call_args[0][0]
+ assert '17' in mock_run.call_args[0][0]
+
+ @mock.patch.dict('os.environ', {'JAVA_HOME': '/custom/jdk'})
+ @mock.patch.object(JDKPrerequisite, '_darwin_jdk_is_supported')
+ def test_darwin_checker_uses_java_home_env(self, mock_is_supported):
+ """Test darwin_checker uses JAVA_HOME env var if set."""
+ prerequisite = JDKPrerequisite()
+ mock_is_supported.return_value = True
+
+ result = prerequisite.darwin_checker()
+ assert result is True
+ mock_is_supported.assert_called_with('/custom/jdk')
+
+
+class TestHomebrewHelpers:
+ """Test Homebrew helper methods."""
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.Popen')
+ def test_darwin_get_brew_formula_location_prefix_success(self, mock_popen):
+ """Test _darwin_get_brew_formula_location_prefix returns path on success."""
+ prerequisite = Prerequisite()
+ mock_process = mock.Mock()
+ mock_process.returncode = 0
+ mock_process.communicate.return_value = (b'/opt/homebrew/opt/openssl@3\n', b'')
+ mock_popen.return_value = mock_process
+
+ result = prerequisite._darwin_get_brew_formula_location_prefix('openssl@3')
+ assert result == '/opt/homebrew/opt/openssl@3'
+ mock_popen.assert_called_once()
+ assert 'brew' in mock_popen.call_args[0][0]
+ assert '--prefix' in mock_popen.call_args[0][0]
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.Popen')
+ @mock.patch('pythonforandroid.prerequisites.error')
+ def test_darwin_get_brew_formula_location_prefix_failure(self, mock_error, mock_popen):
+ """Test _darwin_get_brew_formula_location_prefix returns None on failure."""
+ prerequisite = Prerequisite()
+ mock_process = mock.Mock()
+ mock_process.returncode = 1
+ mock_process.communicate.return_value = (b'', b'Formula not found\n')
+ mock_popen.return_value = mock_process
+
+ result = prerequisite._darwin_get_brew_formula_location_prefix('nonexistent')
+ assert result is None
+ mock_error.assert_called()
+
+ @mock.patch('pythonforandroid.prerequisites.subprocess.Popen')
+ def test_darwin_get_brew_formula_location_prefix_with_installed_flag(self, mock_popen):
+ """Test _darwin_get_brew_formula_location_prefix uses --installed flag."""
+ prerequisite = Prerequisite()
+ mock_process = mock.Mock()
+ mock_process.returncode = 0
+ mock_process.communicate.return_value = (b'/opt/homebrew/opt/cmake\n', b'')
+ mock_popen.return_value = mock_process
+
+ prerequisite._darwin_get_brew_formula_location_prefix('cmake', installed=True)
+ assert '--installed' in mock_popen.call_args[0][0]
+
+
+class TestCheckAndInstallPrerequisites:
+ """Test main prerequisite checking workflow."""
+
+ @mock.patch('pythonforandroid.prerequisites.get_required_prerequisites')
+ def test_check_and_install_all_met(self, mock_get_prereqs):
+ """Test check_and_install when all prerequisites are met."""
+ # Create mock prerequisites that are all valid
+ mock_prereq1 = mock.Mock()
+ mock_prereq1.is_valid.return_value = True
+ mock_prereq2 = mock.Mock()
+ mock_prereq2.is_valid.return_value = True
+
+ mock_get_prereqs.return_value = [mock_prereq1, mock_prereq2]
+
+ check_and_install_default_prerequisites()
+
+ # Verify prerequisites were checked
+ mock_prereq1.is_valid.assert_called_once()
+ mock_prereq2.is_valid.assert_called_once()
+
+ # Verify no installation attempted
+ mock_prereq1.install.assert_not_called()
+ mock_prereq2.install.assert_not_called()
+
+ @mock.patch('pythonforandroid.prerequisites.get_required_prerequisites')
+ def test_check_and_install_some_not_met(self, mock_get_prereqs):
+ """Test check_and_install when some prerequisites are not met."""
+ # First prerequisite valid, second not valid but has installer
+ mock_prereq1 = mock.Mock()
+ mock_prereq1.is_valid.return_value = True
+
+ mock_prereq2 = mock.Mock()
+ mock_prereq2.is_valid.return_value = False
+ mock_prereq2.install_is_supported.return_value = True
+
+ mock_get_prereqs.return_value = [mock_prereq1, mock_prereq2]
+
+ check_and_install_default_prerequisites()
+
+ # Verify second prerequisite triggers installation workflow
+ mock_prereq2.show_helper.assert_called_once()
+ mock_prereq2.install.assert_called_once()
diff --git a/tests/test_pythonpackage_basic.py b/tests/test_pythonpackage_basic.py
index f1eb68369c..46262472f3 100644
--- a/tests/test_pythonpackage_basic.py
+++ b/tests/test_pythonpackage_basic.py
@@ -12,6 +12,8 @@
import tempfile
import textwrap
from unittest import mock
+import pytest
+from build import BuildBackendException
from pythonforandroid.pythonpackage import (
_extract_info_from_package,
@@ -198,6 +200,93 @@ def test_parse_as_folder_reference():
assert parse_as_folder_reference("test @ https://bla") is None
+@pytest.mark.parametrize("input_ref,expected", [
+ # URL-encoded special characters
+ ("file:///path/with%40special", "/path/with@special"),
+ ("file:///path/with%23hash", "/path/with#hash"),
+ # Mixed @ syntax
+ ("pkg @ file:///path/to/pkg", "/path/to/pkg"),
+ # Empty and relative paths
+ ("", ""),
+ ("./relative", "./relative"),
+])
+def test_parse_as_folder_reference_edge_cases(input_ref, expected):
+ """Test edge cases in folder reference parsing."""
+ assert parse_as_folder_reference(input_ref) == expected
+
+
+@pytest.mark.parametrize("path,expected", [
+ # Relative paths (should be filesystem paths)
+ ("../parent", True),
+ ("~/home/path", True),
+ ("./current", True),
+ # Git URLs (should not be filesystem paths)
+ ("git+https://github.com/user/repo.git", False),
+ ("git+ssh://git@github.com/user/repo.git", False),
+ # Version specifiers (should not be filesystem paths)
+ ("package>=1.0,<2.0", False),
+ ("package[extra]>=1.0", False),
+])
+def test_is_filesystem_path_edge_cases(path, expected):
+ """Test additional edge cases for filesystem path detection."""
+ assert is_filesystem_path(path) == expected
+
+
+@pytest.mark.parametrize("input_dep,expected", [
+ # Query parameters
+ ("pkg @ https://example.com/pkg.zip?token=abc123", "https://example.com/pkg.zip?token=abc123#egg=pkg"),
+ # Fragments
+ ("pkg @ https://example.com/pkg.zip#sha256=abc", "https://example.com/pkg.zip#sha256=abc#egg=pkg"),
+])
+def test_transform_dep_for_pip_with_special_urls(input_dep, expected):
+ """Test dependency transformation with query parameters and fragments."""
+ assert transform_dep_for_pip(input_dep) == expected
+
+
+def test_transform_dep_for_pip_passthrough():
+ """Test passthrough for already-transformed URLs."""
+ url = "https://example.com/package.zip#egg=package"
+ assert transform_dep_for_pip(url) == url
+
+
+def test_get_package_name_with_error():
+ """Test get_package_name handles errors gracefully."""
+ # Test with invalid package that doesn't exist
+ with mock.patch("pythonforandroid.pythonpackage."
+ "extract_metainfo_files_from_package") as mock_extract:
+ exception_message = "Package not found"
+ mock_extract.side_effect = Exception(exception_message)
+
+ with pytest.raises(Exception, match=exception_message):
+ get_package_name("nonexistent-package-xyz-123")
+
+
+def test_get_dep_names_error_handling():
+ """Test error handling in dependency extraction."""
+ # Use context manager to ensure cleanup even if test fails
+ with tempfile.TemporaryDirectory(prefix="p4a-error-test-") as temp_d:
+ # Create a setup.py that will fail
+ with open(os.path.join(temp_d, "setup.py"), "w") as f:
+ f.write("raise RuntimeError('Invalid setup.py')")
+
+ with pytest.raises(BuildBackendException, match="Backend subprocess exited when trying to invoke get_requires_for_build_wheel"):
+ get_dep_names_of_package(temp_d, recursive=False, verbose=True)
+
+
+def test_extract_info_from_package_missing_metadata():
+ """Test _extract_info_from_package raises error when metadata is missing."""
+ def fake_empty_metadata(dep_name, output_folder, debug=False):
+ # Don't create any metadata files
+ pass
+
+ with mock.patch("pythonforandroid.pythonpackage."
+ "extract_metainfo_files_from_package",
+ fake_empty_metadata):
+ # Should raise an exception when metadata is missing
+ with pytest.raises(FileNotFoundError):
+ _extract_info_from_package("test", extract_type="name")
+
+
class TestGetSystemPythonExecutable():
""" This contains all tests for _get_system_python_executable().
From 7b24a371b9bf16a1f5bb2ecce248bd053fa03158 Mon Sep 17 00:00:00 2001
From: Dexer <73297572+DexerBR@users.noreply.github.com>
Date: Sun, 9 Nov 2025 22:44:47 -0300
Subject: [PATCH 158/158] Add math library linkage
---
pythonforandroid/recipes/Pillow/__init__.py | 3 +++
1 file changed, 3 insertions(+)
diff --git a/pythonforandroid/recipes/Pillow/__init__.py b/pythonforandroid/recipes/Pillow/__init__.py
index 447803263f..8326d25ed9 100644
--- a/pythonforandroid/recipes/Pillow/__init__.py
+++ b/pythonforandroid/recipes/Pillow/__init__.py
@@ -34,6 +34,9 @@ class PillowRecipe(PyProjectRecipe):
def get_recipe_env(self, arch, **kwargs):
env = super().get_recipe_env(arch, **kwargs)
+ # Add math library linkage
+ env["LDFLAGS"] = env.get("LDFLAGS", "") + " -lm"
+
jpeg = self.get_recipe('jpeg', self.ctx)
jpeg_inc_dir = jpeg_lib_dir = jpeg.get_build_dir(arch.arch)
env["JPEG_ROOT"] = "{}:{}".format(jpeg_lib_dir, jpeg_inc_dir)