c2cgeoportal-geoportal 2.3.5.80__py3-none-any.whl → 2.9rc1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- c2cgeoportal_geoportal/__init__.py +960 -0
- c2cgeoportal_geoportal/lib/__init__.py +256 -0
- c2cgeoportal_geoportal/lib/authentication.py +250 -0
- c2cgeoportal_geoportal/lib/bashcolor.py +46 -0
- c2cgeoportal_geoportal/lib/cacheversion.py +75 -0
- c2cgeoportal_geoportal/lib/caching.py +176 -0
- c2cgeoportal_geoportal/lib/check_collector.py +80 -0
- c2cgeoportal_geoportal/lib/checker.py +295 -0
- c2cgeoportal_geoportal/lib/common_headers.py +170 -0
- c2cgeoportal_geoportal/lib/dbreflection.py +266 -0
- c2cgeoportal_geoportal/lib/filter_capabilities.py +360 -0
- c2cgeoportal_geoportal/lib/fulltextsearch.py +50 -0
- c2cgeoportal_geoportal/lib/functionality.py +166 -0
- c2cgeoportal_geoportal/lib/headers.py +62 -0
- c2cgeoportal_geoportal/lib/i18n.py +38 -0
- c2cgeoportal_geoportal/lib/layers.py +132 -0
- c2cgeoportal_geoportal/lib/lingva_extractor.py +937 -0
- c2cgeoportal_geoportal/lib/loader.py +57 -0
- c2cgeoportal_geoportal/lib/metrics.py +117 -0
- c2cgeoportal_geoportal/lib/oauth2.py +1186 -0
- c2cgeoportal_geoportal/lib/oidc.py +302 -0
- c2cgeoportal_geoportal/lib/wmstparsing.py +353 -0
- c2cgeoportal_geoportal/lib/xsd.py +166 -0
- c2cgeoportal_geoportal/py.typed +0 -0
- c2cgeoportal_geoportal/resources.py +49 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/ci/config.yaml +26 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/cookiecutter.json +18 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.dockerignore +6 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.eslintrc.yaml +19 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.prospector.yaml +30 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/Dockerfile +75 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/Makefile +6 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/alembic.ini +58 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/alembic.yaml +19 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/development.ini +121 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/gunicorn.conf.py +139 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/language_mapping +3 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/lingva-client.cfg +5 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/lingva-server.cfg +6 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/production.ini +38 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/requirements.txt +2 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/setup.py +25 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/webpack.api.js +41 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/webpack.apps.js +64 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/webpack.commons.js +11 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/webpack.config.js +22 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/__init__.py +42 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/authentication.py +10 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/dev.py +14 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/models.py +8 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/multi_organization.py +7 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/resources.py +11 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static-ngeo/api/index.js +12 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static-ngeo/js/{{cookiecutter.package}}module.js +25 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/subscribers.py +39 -0
- c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/views/__init__.py +0 -0
- c2cgeoportal_geoportal/scaffolds/advance_update/cookiecutter.json +18 -0
- c2cgeoportal_geoportal/scaffolds/advance_update/{{cookiecutter.project}}/geoportal/CONST_Makefile +121 -0
- c2cgeoportal_geoportal/scaffolds/create/cookiecutter.json +18 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.dockerignore +14 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.editorconfig +17 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.github/workflows/main.yaml +73 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.github/workflows/rebuild.yaml +50 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.github/workflows/update_l10n.yaml +66 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.gitignore +16 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.pre-commit-config.yaml +35 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.prettierignore +1 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/.prettierrc.yaml +2 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Dockerfile +75 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Makefile +70 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/README.rst +29 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/build +179 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/ci/config.yaml +22 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/ci/docker-compose-check +25 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/ci/requirements.txt +2 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-db.yaml +24 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-lib.yaml +511 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-qgis.yaml +21 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose.override.sample.yaml +59 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose.yaml +121 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/env.default +102 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/env.project +69 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/vars.yaml +430 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/locale/en/LC_MESSAGES/{{cookiecutter.package}}_geoportal-client.po +6 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/css/desktop.css +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/css/iframe_api.css +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/css/mobile.css +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/banner_left.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/banner_right.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/blank.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/markers/marker-blue.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/markers/marker-gold.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/markers/marker-green.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/images/markers/marker.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/static/robot.txt.tmpl +3 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/data/Readme.txt +69 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/data/TM_EUROPE_BORDERS-0.3.sql +70 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/demo.map.tmpl +224 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Arial.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Arialbd.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Arialbi.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Ariali.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/NotoSans-Bold.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/NotoSans-BoldItalic.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/NotoSans-Italic.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/NotoSans-Regular.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Verdana.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Verdanab.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Verdanai.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts/Verdanaz.ttf +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/fonts.conf +12 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/mapserver.conf +15 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/mapserver.map.tmpl +87 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/mapserver/tinyows.xml.tmpl +36 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/A3_Landscape.jrxml +207 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/A3_Portrait.jrxml +185 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/A4_Landscape.jrxml +200 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/A4_Portrait.jrxml +170 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/config.yaml.tmpl +175 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/legend.jrxml +109 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/localisation.properties +4 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/localisation_fr.properties +4 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/logo.png +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/north.svg +93 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/print/print-apps/{{cookiecutter.package}}/results.jrxml +25 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/project.yaml +18 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/pyproject.toml +7 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/qgisserver/pg_service.conf.tmpl +15 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/run_alembic.sh +11 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/scripts/db-backup +126 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/scripts/db-restore +132 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/setup.cfg +7 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/spell-ignore-words.txt +5 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/tests/__init__.py +0 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/tests/test_app.py +43 -0
- c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/tilegeneration/config.yaml.tmpl +195 -0
- c2cgeoportal_geoportal/scaffolds/update/cookiecutter.json +18 -0
- c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/.upgrade.yaml +67 -0
- c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_CHANGELOG.txt +295 -0
- c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_create_template/tests/test_testapp.py +48 -0
- c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/geoportal/CONST_config-schema.yaml +922 -0
- c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/geoportal/CONST_vars.yaml +1503 -0
- c2cgeoportal_geoportal/scripts/__init__.py +64 -0
- c2cgeoportal_geoportal/scripts/c2cupgrade.py +879 -0
- c2cgeoportal_geoportal/scripts/create_demo_theme.py +80 -0
- c2cgeoportal_geoportal/scripts/manage_users.py +140 -0
- c2cgeoportal_geoportal/scripts/pcreate.py +314 -0
- c2cgeoportal_geoportal/scripts/theme2fts.py +347 -0
- c2cgeoportal_geoportal/scripts/urllogin.py +81 -0
- c2cgeoportal_geoportal/templates/login.html +90 -0
- c2cgeoportal_geoportal/templates/notlogin.html +62 -0
- c2cgeoportal_geoportal/templates/testi18n.html +12 -0
- c2cgeoportal_geoportal/views/__init__.py +59 -0
- c2cgeoportal_geoportal/views/dev.py +57 -0
- c2cgeoportal_geoportal/views/dynamic.py +208 -0
- c2cgeoportal_geoportal/views/entry.py +174 -0
- c2cgeoportal_geoportal/views/fulltextsearch.py +189 -0
- c2cgeoportal_geoportal/views/geometry_processing.py +75 -0
- c2cgeoportal_geoportal/views/i18n.py +129 -0
- c2cgeoportal_geoportal/views/layers.py +713 -0
- c2cgeoportal_geoportal/views/login.py +679 -0
- c2cgeoportal_geoportal/views/mapserverproxy.py +191 -0
- c2cgeoportal_geoportal/views/memory.py +90 -0
- c2cgeoportal_geoportal/views/ogcproxy.py +120 -0
- c2cgeoportal_geoportal/views/pdfreport.py +245 -0
- c2cgeoportal_geoportal/views/printproxy.py +143 -0
- c2cgeoportal_geoportal/views/profile.py +127 -0
- c2cgeoportal_geoportal/views/proxy.py +259 -0
- c2cgeoportal_geoportal/views/raster.py +193 -0
- c2cgeoportal_geoportal/views/resourceproxy.py +73 -0
- c2cgeoportal_geoportal/views/shortener.py +152 -0
- c2cgeoportal_geoportal/views/theme.py +1322 -0
- c2cgeoportal_geoportal/views/tinyowsproxy.py +189 -0
- c2cgeoportal_geoportal/views/vector_tiles.py +83 -0
- {c2cgeoportal_geoportal-2.3.5.80.dist-info → c2cgeoportal_geoportal-2.9rc1.dist-info}/METADATA +21 -24
- c2cgeoportal_geoportal-2.9rc1.dist-info/RECORD +192 -0
- {c2cgeoportal_geoportal-2.3.5.80.dist-info → c2cgeoportal_geoportal-2.9rc1.dist-info}/WHEEL +1 -1
- c2cgeoportal_geoportal-2.9rc1.dist-info/entry_points.txt +28 -0
- c2cgeoportal_geoportal-2.9rc1.dist-info/top_level.txt +2 -0
- tests/__init__.py +100 -0
- tests/test_cachebuster.py +71 -0
- tests/test_caching.py +275 -0
- tests/test_checker.py +85 -0
- tests/test_decimaljson.py +47 -0
- tests/test_headerstween.py +64 -0
- tests/test_i18n.py +31 -0
- tests/test_init.py +193 -0
- tests/test_locale_negociator.py +69 -0
- tests/test_mapserverproxy_route_predicate.py +64 -0
- tests/test_raster.py +267 -0
- tests/test_wmstparsing.py +238 -0
- tests/xmlstr.py +103 -0
- c2cgeoportal_geoportal-2.3.5.80.dist-info/DESCRIPTION.rst +0 -8
- c2cgeoportal_geoportal-2.3.5.80.dist-info/RECORD +0 -7
- c2cgeoportal_geoportal-2.3.5.80.dist-info/entry_points.txt +0 -22
- c2cgeoportal_geoportal-2.3.5.80.dist-info/metadata.json +0 -1
- c2cgeoportal_geoportal-2.3.5.80.dist-info/top_level.txt +0 -1
@@ -0,0 +1,80 @@
|
|
1
|
+
# Copyright (c) 2011-2024, Camptocamp SA
|
2
|
+
# All rights reserved.
|
3
|
+
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
|
7
|
+
# 1. Redistributions of source code must retain the above copyright notice, this
|
8
|
+
# list of conditions and the following disclaimer.
|
9
|
+
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
10
|
+
# this list of conditions and the following disclaimer in the documentation
|
11
|
+
# and/or other materials provided with the distribution.
|
12
|
+
|
13
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
14
|
+
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
15
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
16
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
17
|
+
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
18
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
19
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
20
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
21
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
22
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
23
|
+
|
24
|
+
# The views and conclusions contained in the software and documentation are those
|
25
|
+
# of the authors and should not be interpreted as representing official policies,
|
26
|
+
# either expressed or implied, of the FreeBSD Project.
|
27
|
+
|
28
|
+
|
29
|
+
import argparse
|
30
|
+
import logging
|
31
|
+
|
32
|
+
import transaction
|
33
|
+
|
34
|
+
from c2cgeoportal_geoportal.scripts import fill_arguments, get_appsettings, get_session
|
35
|
+
|
36
|
+
_LOG = logging.getLogger(__name__)
|
37
|
+
|
38
|
+
|
39
|
+
def main() -> None:
|
40
|
+
"""Create and populate the database tables."""
|
41
|
+
parser = argparse.ArgumentParser(description="Create and populate the database tables.")
|
42
|
+
fill_arguments(parser)
|
43
|
+
options = parser.parse_args()
|
44
|
+
settings = get_appsettings(options)
|
45
|
+
|
46
|
+
with transaction.manager:
|
47
|
+
session = get_session(settings, transaction.manager)
|
48
|
+
|
49
|
+
from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel
|
50
|
+
Interface,
|
51
|
+
LayerGroup,
|
52
|
+
LayerWMS,
|
53
|
+
OGCServer,
|
54
|
+
Theme,
|
55
|
+
)
|
56
|
+
|
57
|
+
interfaces = session.query(Interface).all()
|
58
|
+
ogc_server = session.query(OGCServer).filter(OGCServer.name == "source for image/png").one()
|
59
|
+
|
60
|
+
layer_borders = LayerWMS("Borders", "borders")
|
61
|
+
layer_borders.interfaces = interfaces
|
62
|
+
layer_borders.ogc_server = ogc_server
|
63
|
+
layer_density = LayerWMS("Density", "density")
|
64
|
+
layer_density.interfaces = interfaces
|
65
|
+
layer_density.ogc_server = ogc_server
|
66
|
+
|
67
|
+
group = LayerGroup("Demo")
|
68
|
+
group.children = [layer_borders, layer_density]
|
69
|
+
|
70
|
+
theme = Theme("Demo")
|
71
|
+
theme.children = [group]
|
72
|
+
theme.interfaces = interfaces
|
73
|
+
|
74
|
+
session.add(theme)
|
75
|
+
|
76
|
+
print("Successfully added the demo theme")
|
77
|
+
|
78
|
+
|
79
|
+
if __name__ == "__main__":
|
80
|
+
main()
|
@@ -0,0 +1,140 @@
|
|
1
|
+
# Copyright (c) 2011-2024, Camptocamp SA
|
2
|
+
# All rights reserved.
|
3
|
+
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
|
7
|
+
# 1. Redistributions of source code must retain the above copyright notice, this
|
8
|
+
# list of conditions and the following disclaimer.
|
9
|
+
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
10
|
+
# this list of conditions and the following disclaimer in the documentation
|
11
|
+
# and/or other materials provided with the distribution.
|
12
|
+
|
13
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
14
|
+
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
15
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
16
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
17
|
+
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
18
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
19
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
20
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
21
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
22
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
23
|
+
|
24
|
+
# The views and conclusions contained in the software and documentation are those
|
25
|
+
# of the authors and should not be interpreted as representing official policies,
|
26
|
+
# either expressed or implied, of the FreeBSD Project.
|
27
|
+
|
28
|
+
|
29
|
+
import argparse
|
30
|
+
import sys
|
31
|
+
from typing import cast
|
32
|
+
|
33
|
+
import transaction
|
34
|
+
|
35
|
+
from c2cgeoportal_geoportal.scripts import fill_arguments, get_appsettings, get_session
|
36
|
+
|
37
|
+
|
38
|
+
def get_argparser() -> argparse.ArgumentParser:
|
39
|
+
"""Get the argument parser for this script."""
|
40
|
+
|
41
|
+
usage = """Reset a user password.
|
42
|
+
The username is used as password if the password is not provided with the corresponding option.
|
43
|
+
User can be created if it does not exist yet."""
|
44
|
+
|
45
|
+
parser = argparse.ArgumentParser(description=usage)
|
46
|
+
fill_arguments(parser)
|
47
|
+
parser.add_argument("--password", "-p", help="Set password (if not set, username is used as password")
|
48
|
+
parser.add_argument(
|
49
|
+
"--create", "-c", action="store_true", default=False, help="Create user if it does not already exist"
|
50
|
+
)
|
51
|
+
parser.add_argument(
|
52
|
+
"--rolename", "-r", default="role_admin", help="The role name which must exist in the database"
|
53
|
+
)
|
54
|
+
parser.add_argument("--email", "-e", default=None, help="The user email")
|
55
|
+
parser.add_argument("user", help="The user")
|
56
|
+
return parser
|
57
|
+
|
58
|
+
|
59
|
+
def main() -> None:
|
60
|
+
"""
|
61
|
+
Emergency user create and password reset script example.
|
62
|
+
|
63
|
+
Reset toto password to foobar: docker compose
|
64
|
+
exec geoportal manage-users --password=foobar toto example, create user foo with password bar and role
|
65
|
+
admin: docker compose exec geoportal manage-users --create --rolename=role_admin --password=bar foo.
|
66
|
+
|
67
|
+
to get the options list, do: docker compose exec geoportal manage-users --help
|
68
|
+
"""
|
69
|
+
|
70
|
+
parser = get_argparser()
|
71
|
+
options = parser.parse_args()
|
72
|
+
username = options.user
|
73
|
+
settings = get_appsettings(options)
|
74
|
+
|
75
|
+
with transaction.manager:
|
76
|
+
session = get_session(settings, transaction.manager)
|
77
|
+
|
78
|
+
# Must be done only once we have loaded the project config
|
79
|
+
from c2cgeoportal_commons.models.main import Role # pylint: disable=import-outside-toplevel
|
80
|
+
from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel
|
81
|
+
|
82
|
+
print("\n")
|
83
|
+
|
84
|
+
# Check that user exists
|
85
|
+
query = session.query(User).filter_by(username=username)
|
86
|
+
|
87
|
+
result = query.count()
|
88
|
+
if result == 0:
|
89
|
+
if not options.create:
|
90
|
+
# If doesn't exist and no -c option, throw error
|
91
|
+
print(f"User {username} does not exist in database")
|
92
|
+
sys.exit(1)
|
93
|
+
else:
|
94
|
+
if options.password is None:
|
95
|
+
parser.error("The password is mandatory on user creation")
|
96
|
+
if options.email is None:
|
97
|
+
parser.error("The email is mandatory on user creation")
|
98
|
+
|
99
|
+
# Get roles
|
100
|
+
query_role = session.query(Role).filter(Role.name == options.rolename)
|
101
|
+
|
102
|
+
if query_role.count() == 0:
|
103
|
+
# Role not found in db?
|
104
|
+
print(f"Role matching {options.rolename} does not exist in database")
|
105
|
+
sys.exit(1)
|
106
|
+
|
107
|
+
role = query_role.first()
|
108
|
+
assert role is not None
|
109
|
+
|
110
|
+
user = User(
|
111
|
+
username=username,
|
112
|
+
password=cast(str, options.password),
|
113
|
+
email=cast(str, options.email),
|
114
|
+
settings_role=role,
|
115
|
+
roles=[role],
|
116
|
+
)
|
117
|
+
session.add(user)
|
118
|
+
|
119
|
+
print(f"User {username} created with password {options.password} and role {options.rolename}")
|
120
|
+
|
121
|
+
else:
|
122
|
+
# If user exists (assuming username are unique)
|
123
|
+
first_user = query.first()
|
124
|
+
assert first_user is not None
|
125
|
+
user = first_user
|
126
|
+
|
127
|
+
if options.password is not None:
|
128
|
+
print(f"Password set to: {options.password}")
|
129
|
+
user.password = f"{options.password}"
|
130
|
+
|
131
|
+
if options.email is not None:
|
132
|
+
user.email = options.email
|
133
|
+
|
134
|
+
session.add(user)
|
135
|
+
|
136
|
+
print(f"Password reset for user {username}")
|
137
|
+
|
138
|
+
|
139
|
+
if __name__ == "__main__":
|
140
|
+
main()
|
@@ -0,0 +1,314 @@
|
|
1
|
+
# Copyright (c) 2021-2024, Camptocamp SA
|
2
|
+
# All rights reserved.
|
3
|
+
|
4
|
+
# Redistribution and use in source and binary forms, with or without
|
5
|
+
# modification, are permitted provided that the following conditions are met:
|
6
|
+
|
7
|
+
# 1. Redistributions of source code must retain the above copyright notice, this
|
8
|
+
# list of conditions and the following disclaimer.
|
9
|
+
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
10
|
+
# this list of conditions and the following disclaimer in the documentation
|
11
|
+
# and/or other materials provided with the distribution.
|
12
|
+
|
13
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
14
|
+
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
15
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
16
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
17
|
+
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
18
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
19
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
20
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
21
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
22
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
23
|
+
|
24
|
+
# The views and conclusions contained in the software and documentation are those
|
25
|
+
# of the authors and should not be interpreted as representing official policies,
|
26
|
+
# either expressed or implied, of the FreeBSD Project.
|
27
|
+
|
28
|
+
|
29
|
+
import json
|
30
|
+
import os
|
31
|
+
import re
|
32
|
+
import subprocess
|
33
|
+
import sys
|
34
|
+
from argparse import ArgumentParser
|
35
|
+
from typing import Any, Union, cast
|
36
|
+
|
37
|
+
import pkg_resources
|
38
|
+
import requests
|
39
|
+
import yaml
|
40
|
+
from cookiecutter.log import configure_logger
|
41
|
+
from cookiecutter.main import cookiecutter
|
42
|
+
|
43
|
+
_bad_chars_re = re.compile("[^a-zA-Z0-9_]")
|
44
|
+
SCAFFOLDS_DIR = pkg_resources.resource_filename("c2cgeoportal_geoportal", "scaffolds")
|
45
|
+
|
46
|
+
|
47
|
+
def get_argparser() -> ArgumentParser:
|
48
|
+
"""Get the argument parser for this script."""
|
49
|
+
|
50
|
+
parser = ArgumentParser(
|
51
|
+
prog=sys.argv[0],
|
52
|
+
add_help=True,
|
53
|
+
description="Wrapper around cookiecutter that create appropriated context.",
|
54
|
+
)
|
55
|
+
parser.add_argument(
|
56
|
+
"-s",
|
57
|
+
"--scaffold",
|
58
|
+
dest="scaffold_names",
|
59
|
+
action="append",
|
60
|
+
help=("Add a scaffold to the create process " "(multiple -s args accepted)"),
|
61
|
+
)
|
62
|
+
parser.add_argument(
|
63
|
+
"-l",
|
64
|
+
"--list",
|
65
|
+
dest="list",
|
66
|
+
action="store_true",
|
67
|
+
help="List all available scaffold names",
|
68
|
+
)
|
69
|
+
parser.add_argument(
|
70
|
+
"--package-name",
|
71
|
+
dest="package_name",
|
72
|
+
action="store",
|
73
|
+
help="Package name to use. The name provided is "
|
74
|
+
"assumed to be a valid Python package name, and "
|
75
|
+
"will not be validated. By default the package "
|
76
|
+
"name is derived from the value of "
|
77
|
+
"output_directory.",
|
78
|
+
)
|
79
|
+
parser.add_argument(
|
80
|
+
"--overwrite",
|
81
|
+
dest="overwrite",
|
82
|
+
action="store_true",
|
83
|
+
help="Always overwrite",
|
84
|
+
)
|
85
|
+
parser.add_argument(
|
86
|
+
"output_directory",
|
87
|
+
nargs="?",
|
88
|
+
default=None,
|
89
|
+
help="The directory where the project will be " "created.",
|
90
|
+
)
|
91
|
+
return parser
|
92
|
+
|
93
|
+
|
94
|
+
def main() -> int:
|
95
|
+
"""Entry point to run PCreateCommand."""
|
96
|
+
command = PCreateCommand(sys.argv)
|
97
|
+
try:
|
98
|
+
return command.run()
|
99
|
+
except KeyboardInterrupt: # pragma: no cover
|
100
|
+
return 1
|
101
|
+
|
102
|
+
|
103
|
+
class PCreateCommand:
|
104
|
+
"""
|
105
|
+
Wrapper around cookiecutter with appropriated context creator for our scaffolds.
|
106
|
+
|
107
|
+
This is a port of Pyramid 1 PCreateCommand using cookiecutter as a backend.
|
108
|
+
"""
|
109
|
+
|
110
|
+
def __init__(self, argv: list[str], quiet: bool = False) -> None:
|
111
|
+
self.quiet = quiet
|
112
|
+
self.parser = get_argparser()
|
113
|
+
self.args = self.parser.parse_args(argv[1:])
|
114
|
+
self.scaffolds = self.all_scaffolds()
|
115
|
+
|
116
|
+
def run(self) -> int:
|
117
|
+
if self.args.list:
|
118
|
+
return self.show_scaffolds()
|
119
|
+
if not self.args.scaffold_names and not self.args.output_directory:
|
120
|
+
if not self.quiet: # pragma: no cover
|
121
|
+
self.parser.print_help()
|
122
|
+
self.out("")
|
123
|
+
self.show_scaffolds()
|
124
|
+
return 2
|
125
|
+
|
126
|
+
return self.render_scaffolds()
|
127
|
+
|
128
|
+
@property
|
129
|
+
def output_path(self) -> str:
|
130
|
+
return cast(str, os.path.abspath(os.path.normpath(self.args.output_directory)))
|
131
|
+
|
132
|
+
def render_scaffolds(self) -> int:
|
133
|
+
verbose = True
|
134
|
+
debug_file = None
|
135
|
+
configure_logger(stream_level="DEBUG" if verbose else "INFO", debug_file=debug_file)
|
136
|
+
|
137
|
+
context = self.get_context()
|
138
|
+
|
139
|
+
for scaffold_name in self.args.scaffold_names:
|
140
|
+
# Needed to be backward compatible for the `test-upgrade init` command
|
141
|
+
if scaffold_name.startswith("c2cgeoportal_"):
|
142
|
+
scaffold_name = scaffold_name[len("c2cgeoportal_") :]
|
143
|
+
self.out(f"Rendering scaffold: {scaffold_name}")
|
144
|
+
cookiecutter(
|
145
|
+
template=os.path.join(SCAFFOLDS_DIR, scaffold_name),
|
146
|
+
extra_context=context,
|
147
|
+
no_input=True,
|
148
|
+
overwrite_if_exists=self.args.overwrite,
|
149
|
+
output_dir=os.path.dirname(self.output_path),
|
150
|
+
)
|
151
|
+
return 0
|
152
|
+
|
153
|
+
def show_scaffolds(self) -> int:
|
154
|
+
scaffolds = sorted(self.scaffolds)
|
155
|
+
if scaffolds:
|
156
|
+
self.out("Available scaffolds:")
|
157
|
+
for scaffold in scaffolds:
|
158
|
+
self.out(f" {scaffold}")
|
159
|
+
else:
|
160
|
+
self.out("No scaffolds available")
|
161
|
+
return 0
|
162
|
+
|
163
|
+
@staticmethod
|
164
|
+
def all_scaffolds() -> list[str]:
|
165
|
+
return os.listdir(SCAFFOLDS_DIR)
|
166
|
+
|
167
|
+
def out(self, msg: str) -> None:
|
168
|
+
if not self.quiet:
|
169
|
+
print(msg)
|
170
|
+
|
171
|
+
def get_context(self) -> dict[str, str | int]:
|
172
|
+
output_dir = self.output_path
|
173
|
+
project_name = os.path.basename(output_dir)
|
174
|
+
if self.args.package_name is None:
|
175
|
+
pkg_name = _bad_chars_re.sub("", project_name.lower().replace("-", "_"))
|
176
|
+
else:
|
177
|
+
pkg_name = self.args.package_name
|
178
|
+
|
179
|
+
context: dict[str, str | int] = {
|
180
|
+
"project": project_name,
|
181
|
+
"package": pkg_name,
|
182
|
+
"authtkt_secret": gen_authtkt_secret(),
|
183
|
+
}
|
184
|
+
context.update(self.read_project_file())
|
185
|
+
if os.environ.get("CI") == "true":
|
186
|
+
context["authtkt_secret"] = ( # nosec
|
187
|
+
"io7heoDui8xaikie1rushaeGeiph8Bequei6ohchaequob6viejei0xooWeuvohf"
|
188
|
+
)
|
189
|
+
|
190
|
+
self.get_var(context, "srid", "Spatial Reference System Identifier (e.g. 2056): ", int)
|
191
|
+
srid = cast(int, context["srid"])
|
192
|
+
extent = self.epsg2bbox(srid)
|
193
|
+
self.get_var(
|
194
|
+
context,
|
195
|
+
"extent",
|
196
|
+
(
|
197
|
+
f"Extent (minx miny maxx maxy): in EPSG: {srid} projection, default is "
|
198
|
+
f"[{extent[0]} {extent[1]} {extent[2]} {extent[3]}]: "
|
199
|
+
if extent
|
200
|
+
else f"Extent (minx miny maxx maxy): in EPSG: {srid} projection: "
|
201
|
+
),
|
202
|
+
)
|
203
|
+
match = re.match(
|
204
|
+
r"([\d.]+)[,; ] *([\d.]+)[,; ] *([\d.]+)[,; ] *([\d.]+)",
|
205
|
+
cast(str, context["extent"]),
|
206
|
+
)
|
207
|
+
if match is not None:
|
208
|
+
extent = [match.group(n + 1) for n in range(4)]
|
209
|
+
assert extent is not None
|
210
|
+
context["extent"] = ",".join(extent)
|
211
|
+
context["extent_mapserver"] = " ".join(extent)
|
212
|
+
|
213
|
+
if context["package"] == "site":
|
214
|
+
raise ValueError(
|
215
|
+
"Sorry, you may not name your package 'site'. "
|
216
|
+
"The package name 'site' has a special meaning in "
|
217
|
+
"Python. Please name it anything except 'site'."
|
218
|
+
)
|
219
|
+
|
220
|
+
package_logger = context["package"]
|
221
|
+
if package_logger == "root":
|
222
|
+
# Rename the app logger in the rare case a project
|
223
|
+
# is named "root"
|
224
|
+
package_logger = "app"
|
225
|
+
context["package_logger"] = package_logger
|
226
|
+
context["geomapfish_version"] = os.environ["VERSION"]
|
227
|
+
# Used in the Docker files to shoos the version of the build image
|
228
|
+
context["geomapfish_version_tag"] = "GEOMAPFISH_VERSION"
|
229
|
+
context["geomapfish_version_tag_env"] = "${GEOMAPFISH_VERSION}"
|
230
|
+
geomapfish_major_version_tag = (
|
231
|
+
"GEOMAPFISH_VERSION"
|
232
|
+
if context.get("unsafe_long_version", False)
|
233
|
+
else "GEOMAPFISH_MAIN_MINOR_VERSION"
|
234
|
+
)
|
235
|
+
# Used in the Docker files to shoos the version of the run image
|
236
|
+
context["geomapfish_major_version_tag"] = geomapfish_major_version_tag
|
237
|
+
context["geomapfish_major_version_tag_env"] = "${" + geomapfish_major_version_tag + "}"
|
238
|
+
context["geomapfish_main_version"] = os.environ["MAJOR_VERSION"]
|
239
|
+
context["geomapfish_main_version_dash"] = os.environ["MAJOR_VERSION"].replace(".", "-")
|
240
|
+
context["geomapfish_main_minor_version"] = os.environ["MAJOR_MINOR_VERSION"]
|
241
|
+
|
242
|
+
return context
|
243
|
+
|
244
|
+
def read_project_file(self) -> dict[str, str | int]:
|
245
|
+
project_file = os.path.join(self.output_path, "project.yaml")
|
246
|
+
if os.path.exists(project_file):
|
247
|
+
with open(project_file, encoding="utf8") as f:
|
248
|
+
project = yaml.safe_load(f)
|
249
|
+
return cast(dict[str, Union[str, int]], project.get("template_vars", {}))
|
250
|
+
else:
|
251
|
+
return {}
|
252
|
+
|
253
|
+
@staticmethod
|
254
|
+
def get_var(
|
255
|
+
context: dict[str, Any],
|
256
|
+
name: str,
|
257
|
+
prompt: str,
|
258
|
+
type_: type[Any] | None = None,
|
259
|
+
) -> None:
|
260
|
+
if name.upper() in os.environ and os.environ[name.upper()] != "":
|
261
|
+
value = os.environ.get(name.upper())
|
262
|
+
else:
|
263
|
+
value = context.get(name)
|
264
|
+
|
265
|
+
if value is None:
|
266
|
+
value = input(prompt).strip()
|
267
|
+
|
268
|
+
if type_ is not None and not isinstance(value, type_):
|
269
|
+
try:
|
270
|
+
value = type_(value)
|
271
|
+
except ValueError:
|
272
|
+
print(f"The attribute {name}={value} is not a {type_}")
|
273
|
+
sys.exit(1)
|
274
|
+
|
275
|
+
context[name] = value
|
276
|
+
|
277
|
+
@staticmethod
|
278
|
+
def epsg2bbox(srid: int) -> list[str] | None:
|
279
|
+
try:
|
280
|
+
r = requests.get(f"https://epsg.io/?format=json&q={srid}", timeout=60)
|
281
|
+
bbox = r.json()["results"][0]["bbox"]
|
282
|
+
r = requests.get(
|
283
|
+
"https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[1]},{bbox[0]}".format(
|
284
|
+
srid=srid, bbox=bbox
|
285
|
+
),
|
286
|
+
timeout=60,
|
287
|
+
)
|
288
|
+
r1 = r.json()[0]
|
289
|
+
r = requests.get(
|
290
|
+
"https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[3]},{bbox[2]}".format(
|
291
|
+
srid=srid, bbox=bbox
|
292
|
+
),
|
293
|
+
timeout=60,
|
294
|
+
)
|
295
|
+
r2 = r.json()[0]
|
296
|
+
return [r1["x"], r2["y"], r2["x"], r1["y"]]
|
297
|
+
except requests.RequestException:
|
298
|
+
print("Failed to establish a connection to epsg.io.")
|
299
|
+
except json.JSONDecodeError:
|
300
|
+
print("epsg.io doesn't return a correct json.")
|
301
|
+
except IndexError:
|
302
|
+
print("Unable to get the bbox")
|
303
|
+
except Exception as exception: # pylint: disable=broad-exception-caught
|
304
|
+
print(f"unexpected error: {str(exception)}")
|
305
|
+
return None
|
306
|
+
|
307
|
+
|
308
|
+
def gen_authtkt_secret() -> str:
|
309
|
+
"""Generate a random authtkt secret."""
|
310
|
+
return subprocess.run(["pwgen", "64"], stdout=subprocess.PIPE, check=True).stdout.decode().strip()
|
311
|
+
|
312
|
+
|
313
|
+
if __name__ == "__main__": # pragma: no cover
|
314
|
+
sys.exit(main() or 0)
|