diff --git a/.bandit.yaml b/.bandit.yaml deleted file mode 100644 index a04a4bb1dc..0000000000 --- a/.bandit.yaml +++ /dev/null @@ -1,31 +0,0 @@ -profile: - sql: - exclude: - - /commons/c2cgeoportal_commons/alembic/main/ - - /commons/c2cgeoportal_commons/alembic/static/ - tests: - - B608 # Possible SQL injection vector through string-based query construction. - subprocess: - exclude: - - /commons/c2cgeoportal_commons/testing/ - - /commons/tests/ - - /geoportal/tests/ - - /geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py - - /admin/tests/ - tests: - - B603 # subprocess call - check for execution of untrusted input. - - B607 # Starting a process with a partial executable path - - B404 # Consider possible security implications associated with call module. - tmp: - exclude: - - /geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py - tests: - - B108 # Probable insecure usage of temp file/directory. -skips: - - B101 # Test for use of assert - - B603 # subprocess call - check for execution of untrusted input. - - B607 # Starting a process with a partial executable path - - B608 # Possible SQL injection vector through string-based query construction. - - B108 # Probable insecure usage of temp file/directory. - - B404 # Consider possible security implications associated with call module. - - B113 # Requests call without timeout. Done by c2cwsgiutils diff --git a/.github/workflows/main.yaml b/.github/workflows/main.yaml index 0688a4e1c0..41eb72df04 100644 --- a/.github/workflows/main.yaml +++ b/.github/workflows/main.yaml @@ -34,7 +34,7 @@ jobs: # When we upgrade this we should also upgrade the requirements # in the documentation: doc/integrator/requirements.rst # and the first pyupgrade pre-commit hook in .pre-commit-config.yaml - MIN_PYTHON_VERSION: '3.8' + MIN_PYTHON_VERSION: '3.10' steps: - run: '! ls BACKPORT_TODO' @@ -128,6 +128,7 @@ jobs: git add commons/c2cgeoportal_commons/alembic/main/*.py commons/c2cgeoportal_commons/alembic/static/*.py git diff --staged --patch > /tmp/alembic.patch git diff --staged --exit-code + git reset --hard - uses: actions/upload-artifact@v4 with: name: Add Alembic upgrade script.patch diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 3be1489d3b..274d5b640e 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -104,64 +104,28 @@ repos: docker/config/haproxy_dev/localhost\.pem |geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter\.project}}/mapserver/data/TM_EUROPE_BORDERS-0.3\.sql )$ - - repo: https://github.com/asottile/pyupgrade - rev: v3.19.0 + - repo: https://github.com/astral-sh/ruff-pre-commit + rev: v0.8.1 hooks: - # The script that will run on the project host - - id: pyupgrade + - id: ruff-format args: - - --py38-plus - files: |- - (?x)^( - geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter\.project}}/(build - |scripts/.*) - |scripts/(get-version - |upgrade) - )$ - # All other - - id: pyupgrade - args: - - --py310-plus - # geoportal/c2cgeoportal_geoportal/views/theme\.py is present because of issue: - # https://bugs.launchpad.net/lxml/+bug/2079018 - exclude: |- - (?x)^( - geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter\.project}}/(build - |scripts/.*) - |scripts/(get-version - |upgrade) - |geoportal/c2cgeoportal_geoportal/views/theme\.py - )$ - - repo: https://github.com/PyCQA/autoflake - rev: v2.3.1 - hooks: - - id: autoflake - - repo: https://github.com/PyCQA/isort - rev: 5.13.2 - hooks: - - id: isort - - repo: https://github.com/psf/black - rev: 24.10.0 - hooks: - - id: black + - --line-length=110 exclude: |- (?x)^( - commons/c2cgeoportal_commons/alembic/script\.py\.mako - |.*\.rst - |.*\.rst\.tmpl - |geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/__init__\.py + geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/__init__\.py )$ - repo: https://github.com/PyCQA/prospector rev: v1.13.3 hooks: - id: prospector args: - - --tool=pydocstyle + - --tool=ruff - --die-on-tool-error - --output-format=pylint additional_dependencies: - - prospector-profile-duplicated==1.8.0 # pypi + - prospector-profile-duplicated==1.8.1 # pypi - prospector-profile-utils==1.13.0 # pypi + - ruff==0.8.1 # pypi - repo: https://github.com/sbrunner/jsonschema-validator rev: 0.3.2 hooks: diff --git a/.prospector.yaml b/.prospector.yaml index 95964a60c7..c6b2c654c5 100644 --- a/.prospector.yaml +++ b/.prospector.yaml @@ -3,6 +3,8 @@ inherits: - utils:base - utils:fix - utils:no-design-checks + - utils:unsafe + - utils:c2cwsgiutils ignore-paths: - commons/setup.py @@ -10,28 +12,14 @@ ignore-paths: - admin/setup.py - geoportal/c2cgeoportal_geoportal/scaffolds - docker/qgisserver + - commons/c2cgeoportal_commons/alembic/main + - commons/c2cgeoportal_commons/alembic/static -pycodestyle: - disable: - # Buggy checks with Python 3.12 - - W604 # backticks are deprecated, use 'repr()' - - W603 # '<>' is deprecated, use '!=' - - E702 # multiple statements on one line (semicolon) - - E713 # test for membership should be 'not in' +mypy: + options: + python_version: '3.10' -pydocstyle: +ruff: disable: - D102 # Missing docstring in public method - - D104 # Missing docstring in public package - - D105 # Missing docstring in magic method - - D107 # Missing docstring in __init__ - - D200 # One-line docstring should fit on one line with quotes - - D202 # No blank lines allowed after function docstring (found 1) - - D203 # 1 blank line required before class docstring (found 0) - - D212 # Multi-line docstring summary should start at the first line - - D407 # Missing dashed underline after section ('Arguments') - - D412 # No blank lines allowed between a section header and its content ('Arguments') - -bandit: - options: - config: .bandit.yaml + - D107 # Missing docstring in `__init__` diff --git a/Makefile b/Makefile index 4aa527c073..228172e45e 100644 --- a/Makefile +++ b/Makefile @@ -37,7 +37,8 @@ prospector: build-checks ## Run the prospector checker @docker run --rm camptocamp/geomapfish-checks:$(DOCKER_TAG) mypy --version @docker run --rm camptocamp/geomapfish-checks:$(DOCKER_TAG) pylint --version --rcfile=/dev/null @docker run --rm camptocamp/geomapfish-checks:$(DOCKER_TAG) pyflakes --version - docker run --rm --volume=$(shell pwd):/opt/c2cgeoportal camptocamp/geomapfish-checks:$(DOCKER_TAG) prospector --output-format=pylint --die-on-tool-error + docker run --rm --volume=$(shell pwd):/opt/c2cgeoportal camptocamp/geomapfish-checks:$(DOCKER_TAG) \ + prospector --without=ruff --output-format=pylint --die-on-tool-error .PHONY: poetry-dev poetry-dev: @@ -45,7 +46,7 @@ poetry-dev: .PHONY: prospector-poetry prospector-poetry: poetry-dev - poetry run prospector --output-format=pylint --die-on-tool-error + poetry run prospector --without=ruff --output-format=pylint --die-on-tool-error .PHONY: additionallint additionallint: ## Check that we should replace some strings in the code @@ -100,7 +101,7 @@ build-qgisserver-tests: .PHONY: prospector-qgisserver prospector-qgisserver: build-qgisserver-tests - docker run --rm --volume=$(shell pwd)/docker/qgisserver:/src camptocamp/geomapfish-qgisserver-tests prospector --output-format=pylint --die-on-tool-error + docker run --rm --volume=$(shell pwd)/docker/qgisserver:/src camptocamp/geomapfish-qgisserver-tests prospector --without=ruff --output-format=pylint --die-on-tool-error .PHONY: build-test-db build-test-db: diff --git a/admin/c2cgeoportal_admin/__init__.py b/admin/c2cgeoportal_admin/__init__.py index 4d4227d388..2747e3c5a7 100644 --- a/admin/c2cgeoportal_admin/__init__.py +++ b/admin/c2cgeoportal_admin/__init__.py @@ -93,7 +93,9 @@ def get_tm_session( ) # Add fake user as we do not have authentication from geoportal - from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.static import ( # pylint: disable=import-outside-toplevel + User, + ) config.add_request_method( lambda request: User( diff --git a/admin/c2cgeoportal_admin/lib/ogcserver_synchronizer.py b/admin/c2cgeoportal_admin/lib/ogcserver_synchronizer.py index 64905a6f47..3ca3f03444 100644 --- a/admin/c2cgeoportal_admin/lib/ogcserver_synchronizer.py +++ b/admin/c2cgeoportal_admin/lib/ogcserver_synchronizer.py @@ -29,16 +29,15 @@ import functools import logging from io import StringIO -from typing import Any, Optional, cast +from typing import Any, cast from xml.etree.ElementTree import Element # nosec import pyramid.request import requests -from defusedxml import ElementTree -from sqlalchemy.orm.session import Session - from c2cgeoportal_commons.lib.url import get_url2 from c2cgeoportal_commons.models import main +from defusedxml import ElementTree +from sqlalchemy.orm.session import Session class dry_run_transaction: # noqa ignore=N801: class names should use CapWords convention @@ -238,7 +237,7 @@ def get_theme(self, el: ElementTree) -> main.Theme: name = name_el.text theme = cast( - Optional[main.Theme], + main.Theme | None, self._request.dbsession.query(main.Theme).filter(main.Theme.name == name).one_or_none(), ) @@ -263,7 +262,7 @@ def get_layer_group(self, el: Element, parent: main.TreeGroup) -> main.LayerGrou assert name is not None group = cast( - Optional[main.LayerGroup], + main.LayerGroup | None, ( self._request.dbsession.query(main.LayerGroup) .filter(main.LayerGroup.name == name) @@ -294,7 +293,7 @@ def get_layer_wms(self, el: Element, parent: main.TreeGroup | None = None) -> ma assert name is not None layer = cast( - Optional[main.LayerWMS], + main.LayerWMS | None, self._request.dbsession.query(main.LayerWMS).filter(main.LayerWMS.name == name).one_or_none(), ) @@ -358,7 +357,7 @@ def get_layer_wms(self, el: Element, parent: main.TreeGroup | None = None) -> ma return layer - @functools.lru_cache(maxsize=10) + @functools.lru_cache(maxsize=10) # noqa: B019 def wms_capabilities(self) -> bytes: errors: set[str] = set() url = get_url2( diff --git a/admin/c2cgeoportal_admin/schemas/dimensions.py b/admin/c2cgeoportal_admin/schemas/dimensions.py index 3326057d16..4a9fb0c27e 100644 --- a/admin/c2cgeoportal_admin/schemas/dimensions.py +++ b/admin/c2cgeoportal_admin/schemas/dimensions.py @@ -30,11 +30,10 @@ import colander from c2cgeoform.schema import GeoFormSchemaNode +from c2cgeoportal_commons.models.main import Dimension from deform.widget import MappingWidget, SequenceWidget from sqlalchemy.orm.attributes import InstrumentedAttribute -from c2cgeoportal_commons.models.main import Dimension - def dimensions_schema_node( prop: InstrumentedAttribute[Any], # pylint: disable=unsubscriptable-object diff --git a/admin/c2cgeoportal_admin/schemas/functionalities.py b/admin/c2cgeoportal_admin/schemas/functionalities.py index 327aa56b56..a05d544032 100644 --- a/admin/c2cgeoportal_admin/schemas/functionalities.py +++ b/admin/c2cgeoportal_admin/schemas/functionalities.py @@ -31,12 +31,11 @@ import colander from c2cgeoform.ext.deform_ext import RelationCheckBoxListWidget from c2cgeoform.schema import GeoFormManyToManySchemaNode, manytomany_validator +from c2cgeoportal_commons.models.main import Functionality from sqlalchemy import inspect, select from sqlalchemy.orm.attributes import InstrumentedAttribute from sqlalchemy.sql.functions import concat -from c2cgeoportal_commons.models.main import Functionality - def available_functionalities_for(settings: dict[str, Any], model: type[Any]) -> list[dict[str, Any]]: """Return filtered list of functionality definitions.""" @@ -81,7 +80,6 @@ def functionalities_schema_node( prop: InstrumentedAttribute[Any], model: type[Any] ) -> colander.SequenceSchema: """Get the schema of the functionalities.""" - return colander.SequenceSchema( GeoFormManyToManySchemaNode(Functionality, None), name=prop.key, diff --git a/admin/c2cgeoportal_admin/schemas/interfaces.py b/admin/c2cgeoportal_admin/schemas/interfaces.py index 212aefc3c1..2bf697e42a 100644 --- a/admin/c2cgeoportal_admin/schemas/interfaces.py +++ b/admin/c2cgeoportal_admin/schemas/interfaces.py @@ -31,9 +31,8 @@ import colander from c2cgeoform.ext.deform_ext import RelationCheckBoxListWidget from c2cgeoform.schema import GeoFormManyToManySchemaNode, manytomany_validator -from sqlalchemy.orm.attributes import InstrumentedAttribute - from c2cgeoportal_commons.models.main import Interface +from sqlalchemy.orm.attributes import InstrumentedAttribute def interfaces_schema_node( diff --git a/admin/c2cgeoportal_admin/schemas/metadata.py b/admin/c2cgeoportal_admin/schemas/metadata.py index 4fa216ff61..cb0c738392 100644 --- a/admin/c2cgeoportal_admin/schemas/metadata.py +++ b/admin/c2cgeoportal_admin/schemas/metadata.py @@ -31,14 +31,14 @@ import colander import pyramid.request from c2cgeoform.schema import GeoFormSchemaNode +from c2cgeoportal_commons.lib.validators import url +from c2cgeoportal_commons.models.main import Metadata from deform.widget import MappingWidget, SelectWidget, SequenceWidget, TextAreaWidget from sqlalchemy import inspect from sqlalchemy.orm.attributes import InstrumentedAttribute from sqlalchemy.orm.mapper import Mapper from c2cgeoportal_admin import _ -from c2cgeoportal_commons.lib.validators import url -from c2cgeoportal_commons.models.main import Metadata def get_relevant_for(model: type[Any] | Mapper[Any]) -> set[str]: @@ -202,7 +202,6 @@ def _translate_available_metadata( def metadata_schema_node(prop: InstrumentedAttribute[Any], model: type[Any]) -> colander.SequenceSchema: """Get the schema of a collection of metadata.""" - # Deferred which returns a dictionary with metadata name as key and metadata definition as value. # Needed to get the metadata types on UI side. metadata_definitions_dict = colander.deferred( diff --git a/admin/c2cgeoportal_admin/schemas/restriction_areas.py b/admin/c2cgeoportal_admin/schemas/restriction_areas.py index d13e70b3de..cd22bd6ea2 100644 --- a/admin/c2cgeoportal_admin/schemas/restriction_areas.py +++ b/admin/c2cgeoportal_admin/schemas/restriction_areas.py @@ -31,9 +31,8 @@ import colander from c2cgeoform.ext.deform_ext import RelationCheckBoxListWidget from c2cgeoform.schema import GeoFormManyToManySchemaNode, manytomany_validator -from sqlalchemy.orm.attributes import InstrumentedAttribute - from c2cgeoportal_commons.models.main import RestrictionArea +from sqlalchemy.orm.attributes import InstrumentedAttribute def restrictionareas_schema_node( diff --git a/admin/c2cgeoportal_admin/schemas/roles.py b/admin/c2cgeoportal_admin/schemas/roles.py index 308a322b43..8ec1357ee3 100644 --- a/admin/c2cgeoportal_admin/schemas/roles.py +++ b/admin/c2cgeoportal_admin/schemas/roles.py @@ -31,9 +31,8 @@ import colander from c2cgeoform.ext.deform_ext import RelationCheckBoxListWidget from c2cgeoform.schema import GeoFormManyToManySchemaNode, manytomany_validator -from sqlalchemy.orm.attributes import InstrumentedAttribute - from c2cgeoportal_commons.models.main import Role +from sqlalchemy.orm.attributes import InstrumentedAttribute def roles_schema_node( diff --git a/admin/c2cgeoportal_admin/schemas/treegroup.py b/admin/c2cgeoportal_admin/schemas/treegroup.py index ea47cb501d..187fb37f29 100644 --- a/admin/c2cgeoportal_admin/schemas/treegroup.py +++ b/admin/c2cgeoportal_admin/schemas/treegroup.py @@ -34,13 +34,13 @@ import pyramid.request import sqlalchemy from c2cgeoform.schema import GeoFormSchemaNode +from c2cgeoportal_commons.lib.literal import Literal +from c2cgeoportal_commons.models.main import LayergroupTreeitem, TreeGroup, TreeItem from sqlalchemy.orm import aliased from sqlalchemy.sql.expression import case, func from c2cgeoportal_admin import _ from c2cgeoportal_admin.widgets import ChildrenWidget, ChildWidget -from c2cgeoportal_commons.lib.literal import Literal -from c2cgeoportal_commons.models.main import LayergroupTreeitem, TreeGroup, TreeItem _LOG = logging.getLogger(__name__) @@ -74,7 +74,8 @@ def treeitems( assert isinstance(dbsession, sqlalchemy.orm.Session) group = case( - (func.count(LayergroupTreeitem.id) == 0, "Unlinked"), else_="Others" # pylint: disable=not-callable + (func.count(LayergroupTreeitem.id) == 0, "Unlinked"), # pylint: disable=not-callable + else_="Others", ) query = ( @@ -123,7 +124,7 @@ def treeitems( def children_validator(node, cstruct): """Get the validator on the children nodes.""" for dict_ in cstruct: - if not dict_["treeitem_id"] in [item["id"] for item in node.candidates]: + if dict_["treeitem_id"] not in [item["id"] for item in node.candidates]: raise colander.Invalid( node, _("Value {} does not exist in table {} or is not allowed to avoid cycles").format( diff --git a/admin/c2cgeoportal_admin/views/__init__.py b/admin/c2cgeoportal_admin/views/__init__.py index 611546b28b..189e935184 100644 --- a/admin/c2cgeoportal_admin/views/__init__.py +++ b/admin/c2cgeoportal_admin/views/__init__.py @@ -4,9 +4,7 @@ class IsAdminPredicate: - """ - A custom predicate that checks if the request is for the admin interface. - """ + """A custom predicate that checks if the request is for the admin interface.""" def __init__(self, val, info): del info diff --git a/admin/c2cgeoportal_admin/views/dimension_layers.py b/admin/c2cgeoportal_admin/views/dimension_layers.py index 169627eb40..51209e8891 100644 --- a/admin/c2cgeoportal_admin/views/dimension_layers.py +++ b/admin/c2cgeoportal_admin/views/dimension_layers.py @@ -32,10 +32,10 @@ import sqlalchemy.orm.query from c2cgeoform.views.abstract_views import ListField +from c2cgeoportal_commons.models.main import DimensionLayer from sqlalchemy.orm import subqueryload from c2cgeoportal_admin.views.layers import LayerViews -from c2cgeoportal_commons.models.main import DimensionLayer _list_field = partial(ListField, DimensionLayer) diff --git a/admin/c2cgeoportal_admin/views/functionalities.py b/admin/c2cgeoportal_admin/views/functionalities.py index 71de35fcfc..ea4ab57a41 100644 --- a/admin/c2cgeoportal_admin/views/functionalities.py +++ b/admin/c2cgeoportal_admin/views/functionalities.py @@ -40,12 +40,12 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import Functionality from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from c2cgeoportal_admin import _ from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.models.main import Functionality _list_field = partial(ListField, Functionality) @@ -103,7 +103,9 @@ def delete(self) -> DeleteResponse: return super().delete() @view_config( - route_name="c2cgeoform_item_duplicate", request_method="GET", renderer="../templates/edit.jinja2" # type: ignore[misc] + route_name="c2cgeoform_item_duplicate", + request_method="GET", + renderer="../templates/edit.jinja2", # type: ignore[misc] ) def duplicate(self) -> ObjectResponse: return super().duplicate() diff --git a/admin/c2cgeoportal_admin/views/interfaces.py b/admin/c2cgeoportal_admin/views/interfaces.py index 806534cc6a..2a8b6406ae 100644 --- a/admin/c2cgeoportal_admin/views/interfaces.py +++ b/admin/c2cgeoportal_admin/views/interfaces.py @@ -37,10 +37,10 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import Interface from pyramid.view import view_config, view_defaults from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.models.main import Interface _list_field = partial(ListField, Interface) diff --git a/admin/c2cgeoportal_admin/views/layer_groups.py b/admin/c2cgeoportal_admin/views/layer_groups.py index fc5a61d93a..ae68c2f2f5 100644 --- a/admin/c2cgeoportal_admin/views/layer_groups.py +++ b/admin/c2cgeoportal_admin/views/layer_groups.py @@ -39,6 +39,7 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import LayerGroup, TreeGroup from deform.widget import FormWidget from pyramid.view import view_config, view_defaults @@ -46,7 +47,6 @@ from c2cgeoportal_admin.schemas.treegroup import children_schema_node from c2cgeoportal_admin.schemas.treeitem import parent_id_node from c2cgeoportal_admin.views.treeitems import TreeItemViews -from c2cgeoportal_commons.models.main import LayerGroup, TreeGroup _list_field = partial(ListField, LayerGroup) diff --git a/admin/c2cgeoportal_admin/views/layers.py b/admin/c2cgeoportal_admin/views/layers.py index 038472b15f..b2b47521bd 100644 --- a/admin/c2cgeoportal_admin/views/layers.py +++ b/admin/c2cgeoportal_admin/views/layers.py @@ -32,10 +32,10 @@ import sqlalchemy import sqlalchemy.orm.query from c2cgeoform.views.abstract_views import ListField +from c2cgeoportal_commons.models.main import Interface, Layer from sqlalchemy.orm import subqueryload from c2cgeoportal_admin.views.treeitems import TreeItemViews -from c2cgeoportal_commons.models.main import Interface, Layer _list_field = partial(ListField, Layer) diff --git a/admin/c2cgeoportal_admin/views/layers_cog.py b/admin/c2cgeoportal_admin/views/layers_cog.py index 7ba329eba9..f655304baf 100644 --- a/admin/c2cgeoportal_admin/views/layers_cog.py +++ b/admin/c2cgeoportal_admin/views/layers_cog.py @@ -40,6 +40,8 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.lib.literal import Literal +from c2cgeoportal_commons.models.main import LayerCOG, LayerGroup from deform.widget import FormWidget from pyramid.httpexceptions import HTTPNotFound from pyramid.view import view_config, view_defaults @@ -50,8 +52,6 @@ from c2cgeoportal_admin.schemas.restriction_areas import restrictionareas_schema_node from c2cgeoportal_admin.schemas.treeitem import parent_id_node from c2cgeoportal_admin.views.layers import LayerViews -from c2cgeoportal_commons.lib.literal import Literal -from c2cgeoportal_commons.models.main import LayerCOG, LayerGroup _list_field = partial(ListField, LayerCOG) diff --git a/admin/c2cgeoportal_admin/views/layers_vectortiles.py b/admin/c2cgeoportal_admin/views/layers_vectortiles.py index c451da4128..6648c9ddf8 100644 --- a/admin/c2cgeoportal_admin/views/layers_vectortiles.py +++ b/admin/c2cgeoportal_admin/views/layers_vectortiles.py @@ -40,6 +40,8 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.lib.literal import Literal +from c2cgeoportal_commons.models.main import LayerGroup, LayerVectorTiles from deform.widget import FormWidget from pyramid.httpexceptions import HTTPNotFound from pyramid.view import view_config, view_defaults @@ -50,8 +52,6 @@ from c2cgeoportal_admin.schemas.restriction_areas import restrictionareas_schema_node from c2cgeoportal_admin.schemas.treeitem import parent_id_node from c2cgeoportal_admin.views.dimension_layers import DimensionLayerViews -from c2cgeoportal_commons.lib.literal import Literal -from c2cgeoportal_commons.models.main import LayerGroup, LayerVectorTiles _list_field = partial(ListField, LayerVectorTiles) diff --git a/admin/c2cgeoportal_admin/views/layers_wms.py b/admin/c2cgeoportal_admin/views/layers_wms.py index 8acd8bab2b..520fd2ffd8 100644 --- a/admin/c2cgeoportal_admin/views/layers_wms.py +++ b/admin/c2cgeoportal_admin/views/layers_wms.py @@ -40,6 +40,14 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import ( + LayerGroup, + LayerWMS, + LayerWMTS, + LogAction, + OGCServer, + TreeItem, +) from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from sqlalchemy import delete, insert, inspect, update @@ -52,7 +60,6 @@ from c2cgeoportal_admin.schemas.restriction_areas import restrictionareas_schema_node from c2cgeoportal_admin.schemas.treeitem import parent_id_node from c2cgeoportal_admin.views.dimension_layers import DimensionLayerViews -from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS, LayerWMTS, LogAction, OGCServer, TreeItem _list_field = partial(ListField, LayerWMS) diff --git a/admin/c2cgeoportal_admin/views/layers_wmts.py b/admin/c2cgeoportal_admin/views/layers_wmts.py index 1798d6317c..cdbf4416ba 100644 --- a/admin/c2cgeoportal_admin/views/layers_wmts.py +++ b/admin/c2cgeoportal_admin/views/layers_wmts.py @@ -40,6 +40,14 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import ( + LayerGroup, + LayerWMS, + LayerWMTS, + LogAction, + OGCServer, + TreeItem, +) from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from sqlalchemy import delete, insert, inspect, update @@ -52,7 +60,6 @@ from c2cgeoportal_admin.schemas.restriction_areas import restrictionareas_schema_node from c2cgeoportal_admin.schemas.treeitem import parent_id_node from c2cgeoportal_admin.views.dimension_layers import DimensionLayerViews -from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS, LayerWMTS, LogAction, OGCServer, TreeItem _list_field = partial(ListField, LayerWMTS) diff --git a/admin/c2cgeoportal_admin/views/layertree.py b/admin/c2cgeoportal_admin/views/layertree.py index c9a1b5ebab..e229418c34 100644 --- a/admin/c2cgeoportal_admin/views/layertree.py +++ b/admin/c2cgeoportal_admin/views/layertree.py @@ -30,12 +30,18 @@ import pyramid.request from c2cgeoform.views.abstract_views import DeleteResponse, ItemAction +from c2cgeoportal_commons.models.main import ( + Interface, + Layer, + LayergroupTreeitem, + Theme, + TreeItem, +) from pyramid.httpexceptions import HTTPNotFound from pyramid.view import view_config, view_defaults from translationstring import TranslationStringFactory from c2cgeoportal_admin import _ -from c2cgeoportal_commons.models.main import Interface, Layer, LayergroupTreeitem, Theme, TreeItem itemtypes_tables = { "theme": "themes", diff --git a/admin/c2cgeoportal_admin/views/logged_views.py b/admin/c2cgeoportal_admin/views/logged_views.py index 85f287ecbe..da1464c16f 100644 --- a/admin/c2cgeoportal_admin/views/logged_views.py +++ b/admin/c2cgeoportal_admin/views/logged_views.py @@ -29,10 +29,9 @@ from typing import Generic, TypeVar from c2cgeoform.views.abstract_views import AbstractViews, DeleteResponse, SaveResponse -from pyramid.httpexceptions import HTTPFound - from c2cgeoportal_commons.models import Base from c2cgeoportal_commons.models.main import Log, LogAction +from pyramid.httpexceptions import HTTPFound _T = TypeVar("_T", bound=Log) diff --git a/admin/c2cgeoportal_admin/views/logs.py b/admin/c2cgeoportal_admin/views/logs.py index 922df76cb0..0e20869e1c 100644 --- a/admin/c2cgeoportal_admin/views/logs.py +++ b/admin/c2cgeoportal_admin/views/logs.py @@ -28,11 +28,16 @@ from functools import partial from c2cgeoform import JSONDict -from c2cgeoform.views.abstract_views import AbstractViews, GridResponse, IndexResponse, ItemAction, ListField -from pyramid.view import view_config, view_defaults - +from c2cgeoform.views.abstract_views import ( + AbstractViews, + GridResponse, + IndexResponse, + ItemAction, + ListField, +) from c2cgeoportal_commons.models import _ from c2cgeoportal_commons.models.main import AbstractLog +from pyramid.view import view_config, view_defaults _list_field = partial(ListField, AbstractLog) diff --git a/admin/c2cgeoportal_admin/views/oauth2_clients.py b/admin/c2cgeoportal_admin/views/oauth2_clients.py index e30d5618a5..44297b2b49 100644 --- a/admin/c2cgeoportal_admin/views/oauth2_clients.py +++ b/admin/c2cgeoportal_admin/views/oauth2_clients.py @@ -37,10 +37,10 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.static import Log, OAuth2Client from pyramid.view import view_config, view_defaults from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.models.static import Log, OAuth2Client _list_field = partial(ListField, OAuth2Client) diff --git a/admin/c2cgeoportal_admin/views/ogc_servers.py b/admin/c2cgeoportal_admin/views/ogc_servers.py index 92e97c6b44..a1599cf0ed 100644 --- a/admin/c2cgeoportal_admin/views/ogc_servers.py +++ b/admin/c2cgeoportal_admin/views/ogc_servers.py @@ -45,6 +45,9 @@ SaveResponse, UserMessage, ) +from c2cgeoportal_commons.lib.literal import Literal +from c2cgeoportal_commons.models import cache_invalidate_cb +from c2cgeoportal_commons.models.main import LogAction, OGCServer from deform.widget import FormWidget from pyramid.httpexceptions import HTTPFound from pyramid.view import view_config, view_defaults @@ -53,9 +56,6 @@ from c2cgeoportal_admin import _ from c2cgeoportal_admin.lib.ogcserver_synchronizer import OGCServerSynchronizer from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.lib.literal import Literal -from c2cgeoportal_commons.models import cache_invalidate_cb -from c2cgeoportal_commons.models.main import LogAction, OGCServer _list_field = partial(ListField, OGCServer) diff --git a/admin/c2cgeoportal_admin/views/restriction_areas.py b/admin/c2cgeoportal_admin/views/restriction_areas.py index 17b63d499f..3edcb63f3f 100644 --- a/admin/c2cgeoportal_admin/views/restriction_areas.py +++ b/admin/c2cgeoportal_admin/views/restriction_areas.py @@ -39,6 +39,7 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import Layer, RestrictionArea from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from sqlalchemy.orm import subqueryload @@ -47,7 +48,6 @@ from c2cgeoportal_admin.schemas.treegroup import treeitem_edit_url from c2cgeoportal_admin.views.logged_views import LoggedViews from c2cgeoportal_admin.widgets import ChildrenWidget, ChildWidget -from c2cgeoportal_commons.models.main import Layer, RestrictionArea _list_field = partial(ListField, RestrictionArea) diff --git a/admin/c2cgeoportal_admin/views/roles.py b/admin/c2cgeoportal_admin/views/roles.py index ec8ee1706d..f26e1572d5 100644 --- a/admin/c2cgeoportal_admin/views/roles.py +++ b/admin/c2cgeoportal_admin/views/roles.py @@ -39,6 +39,8 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import Role +from c2cgeoportal_commons.models.static import User from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from sqlalchemy.orm import subqueryload @@ -47,8 +49,6 @@ from c2cgeoportal_admin.schemas.restriction_areas import restrictionareas_schema_node from c2cgeoportal_admin.views.logged_views import LoggedViews from c2cgeoportal_admin.widgets import ChildrenWidget, ChildWidget -from c2cgeoportal_commons.models.main import Role -from c2cgeoportal_commons.models.static import User _list_field = partial(ListField, Role) diff --git a/admin/c2cgeoportal_admin/views/themes.py b/admin/c2cgeoportal_admin/views/themes.py index 15fc45980d..7914988884 100644 --- a/admin/c2cgeoportal_admin/views/themes.py +++ b/admin/c2cgeoportal_admin/views/themes.py @@ -39,6 +39,7 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.models.main import Functionality, Interface, Role, Theme from deform.widget import FormWidget from pyramid.view import view_config, view_defaults from sqlalchemy.orm import subqueryload @@ -50,7 +51,6 @@ from c2cgeoportal_admin.schemas.roles import roles_schema_node from c2cgeoportal_admin.schemas.treegroup import children_schema_node from c2cgeoportal_admin.views.treeitems import TreeItemViews -from c2cgeoportal_commons.models.main import Functionality, Interface, Role, Theme _list_field = partial(ListField, Theme) diff --git a/admin/c2cgeoportal_admin/views/themes_ordering.py b/admin/c2cgeoportal_admin/views/themes_ordering.py index c9faa7175a..80f4615039 100644 --- a/admin/c2cgeoportal_admin/views/themes_ordering.py +++ b/admin/c2cgeoportal_admin/views/themes_ordering.py @@ -29,6 +29,7 @@ import colander from c2cgeoform.schema import GeoFormSchemaNode from c2cgeoform.views.abstract_views import AbstractViews, ObjectResponse, SaveResponse +from c2cgeoportal_commons.models.main import Theme, TreeItem from deform import ValidationFailure from pyramid.httpexceptions import HTTPFound from pyramid.view import view_config @@ -36,7 +37,6 @@ from c2cgeoportal_admin import _ from c2cgeoportal_admin.schemas.treegroup import treeitem_edit_url from c2cgeoportal_admin.widgets import ChildrenWidget, ChildWidget -from c2cgeoportal_commons.models.main import Theme, TreeItem class ThemeOrderSchema(GeoFormSchemaNode): # pylint: disable=abstract-method @@ -61,7 +61,7 @@ def themes(node, kw): # pylint: disable=unused-argument def themes_validator(node, cstruct): """Validate the theme.""" for dict_ in cstruct: - if not dict_["id"] in [item["id"] for item in node.candidates]: + if dict_["id"] not in [item["id"] for item in node.candidates]: raise colander.Invalid( node, _("Value {} does not exist in table {}").format(dict_["id"], Theme.__tablename__), @@ -126,7 +126,7 @@ def save(self) -> SaveResponse: self._request.dbsession.flush() return HTTPFound(self._request.route_url("layertree")) except ValidationFailure as e: - # FIXME see https://github.com/Pylons/deform/pull/243 + # FIXME see https://github.com/Pylons/deform/pull/243 # pylint: disable=fixme self._populate_widgets(form.schema) return { "title": form.title, diff --git a/admin/c2cgeoportal_admin/views/treeitems.py b/admin/c2cgeoportal_admin/views/treeitems.py index f762d9f22b..0c3d2a5196 100644 --- a/admin/c2cgeoportal_admin/views/treeitems.py +++ b/admin/c2cgeoportal_admin/views/treeitems.py @@ -31,12 +31,17 @@ import sqlalchemy from c2cgeoform.views.abstract_views import ListField, SaveResponse +from c2cgeoportal_commons.models.main import ( + LayergroupTreeitem, + Metadata, + TreeGroup, + TreeItem, +) from pyramid.view import view_config from sqlalchemy.orm import subqueryload from sqlalchemy.sql.functions import concat from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.models.main import LayergroupTreeitem, Metadata, TreeGroup, TreeItem _list_field = partial(ListField, TreeItem) diff --git a/admin/c2cgeoportal_admin/views/users.py b/admin/c2cgeoportal_admin/views/users.py index 769de9aa64..f42c4943a9 100644 --- a/admin/c2cgeoportal_admin/views/users.py +++ b/admin/c2cgeoportal_admin/views/users.py @@ -38,6 +38,9 @@ ObjectResponse, SaveResponse, ) +from c2cgeoportal_commons.lib.email_ import send_email_config +from c2cgeoportal_commons.models.main import Role +from c2cgeoportal_commons.models.static import Log, User from deform.widget import FormWidget from passwordgenerator import pwgenerator from pyramid.httpexceptions import HTTPFound @@ -46,9 +49,6 @@ from c2cgeoportal_admin.schemas.roles import roles_schema_node from c2cgeoportal_admin.views.logged_views import LoggedViews -from c2cgeoportal_commons.lib.email_ import send_email_config -from c2cgeoportal_commons.models.main import Role -from c2cgeoportal_commons.models.static import Log, User _list_field = partial(ListField, User) diff --git a/admin/c2cgeoportal_admin/widgets.py b/admin/c2cgeoportal_admin/widgets.py index bce6599c96..57c8e1a07a 100644 --- a/admin/c2cgeoportal_admin/widgets.py +++ b/admin/c2cgeoportal_admin/widgets.py @@ -29,12 +29,11 @@ import colander import pyramid.request +from c2cgeoportal_commons.models.main import TreeItem from colander import Mapping, SchemaNode from deform import widget from deform.widget import MappingWidget, SequenceWidget -from c2cgeoportal_commons.models.main import TreeItem - registry = widget.default_resource_registry registry.set_js_resources( "magicsuggest", None, "c2cgeoportal_admin:node_modules/magicsuggest-alpine/magicsuggest-min.js" diff --git a/admin/tests/__init__.py b/admin/tests/__init__.py index 2fdfcf47c8..efcfac140e 100644 --- a/admin/tests/__init__.py +++ b/admin/tests/__init__.py @@ -9,7 +9,12 @@ def get_test_default_layers(dbsession, default_ogc_server): - from c2cgeoportal_commons.models.main import LayerCOG, LayerVectorTiles, LayerWMS, LayerWMTS + from c2cgeoportal_commons.models.main import ( + LayerCOG, + LayerVectorTiles, + LayerWMS, + LayerWMTS, + ) default_wms = None if default_ogc_server: @@ -122,9 +127,9 @@ def check_grid_headers(self, resp, expected_col_headers, new="New"): "\n\n{}\n\n differs from \n\n{}", pp.pformat(expected_col_headers), pp.pformat(effective_cols) ) actions = resp.html.select_one('th[data-field="actions"]') - assert "false" == actions.attrs["data-sortable"] + assert actions.attrs["data-sortable"] == "false" if new is not False: - assert 1 == len(list(filter(lambda x: next(x.stripped_strings) == new, resp.html.findAll("a")))) + assert len(list(filter(lambda x: next(x.stripped_strings) == new, resp.html.findAll("a")))) == 1 def check_search(self, test_app, search="", offset=0, limit=10, sort="", order="", total=None): json = test_app.post( @@ -184,7 +189,7 @@ def set_first_field_named(self, form, name, value): def _check_sequence(self, sequence, expected): seq_items = sequence.select(".deform-seq-item") assert len(expected) == len(seq_items) - for seq_item, exp in zip(seq_items, expected): + for seq_item, exp in zip(seq_items, expected, strict=False): self._check_mapping(seq_item, exp) def _check_mapping(self, mapping_item, expected): @@ -203,13 +208,13 @@ def _check_mapping(self, mapping_item, expected): else: assert (exp["value"] or "") == input_tag.attrs.get("value", "") if exp.get("hidden", False): - assert "hidden" == input_tag["type"] + assert input_tag["type"] == "hidden" if "label" in exp: label_tag = mapping_item.select_one('label[for="{}"]'.format(input_tag["id"])) assert exp["label"] == label_tag.getText().strip() def _check_select(self, select, expected): - for exp, option in zip(expected, select.find_all("option")): + for exp, option in zip(expected, select.find_all("option"), strict=False): if "text" in exp: assert exp["text"] == option.text if "value" in exp: @@ -219,11 +224,11 @@ def _check_select(self, select, expected): def _check_submission_problem(self, resp, expected_msg): assert ( - "There was a problem with your submission" - == resp.html.select_one('div[class="error-msg-lbl"]').text + resp.html.select_one('div[class="error-msg-lbl"]').text + == "There was a problem with your submission" ) assert ( - "Errors have been highlighted below" == resp.html.select_one('div[class="error-msg-detail"]').text + resp.html.select_one('div[class="error-msg-detail"]').text == "Errors have been highlighted below" ) assert ( expected_msg diff --git a/admin/tests/conftest.py b/admin/tests/conftest.py index dc32b465aa..e35b3a5679 100644 --- a/admin/tests/conftest.py +++ b/admin/tests/conftest.py @@ -4,6 +4,13 @@ import pytest import sqlalchemy.exc import transaction +from c2cgeoportal_commons.testing import ( + generate_mappers, + get_engine, + get_session_factory, + get_tm_session, +) +from c2cgeoportal_commons.testing.initializedb import truncate_tables from pyramid import testing from pyramid.paster import bootstrap from pyramid.router import Router @@ -12,9 +19,6 @@ from sqlalchemy.orm import Session, SessionTransaction from webtest import TestApp as WebTestApp # Avoid warning with pytest -from c2cgeoportal_commons.testing import generate_mappers, get_engine, get_session_factory, get_tm_session -from c2cgeoportal_commons.testing.initializedb import truncate_tables - @pytest.fixture(scope="session") @pytest.mark.usefixtures("settings") diff --git a/admin/tests/lib/test_ogcserver_synchronizer.py b/admin/tests/lib/test_ogcserver_synchronizer.py index c193facbb5..c5278d4031 100644 --- a/admin/tests/lib/test_ogcserver_synchronizer.py +++ b/admin/tests/lib/test_ogcserver_synchronizer.py @@ -29,18 +29,16 @@ def wms_capabilities(content=DEFAULT_CONTENT): - return """ + return f""" OGC:WMS - {} + {content} -""".format( - content - ) +""" @pytest.fixture(scope="function") @@ -443,9 +441,7 @@ def test_synchronize_clean(self, cap_mock, web_request, dbsession): ) def test_get_layer_wms_defaut(self, web_request, dbsession): - """ - We should copy properties from default LayerWMS. - """ + """We should copy properties from default LayerWMS.""" from c2cgeoportal_commons.models import main synchronizer = self.synchronizer(web_request) @@ -490,9 +486,7 @@ def test_get_layer_wms_defaut(self, web_request, dbsession): assert layer.style == "default_style" def test_get_layer_wms_defaut_style_not_exists(self, web_request, dbsession): - """ - We should not copy style from default LayerWMS if does not exist in capabilities. - """ + """We should not copy style from default LayerWMS if does not exist in capabilities.""" from c2cgeoportal_commons.models import main synchronizer = self.synchronizer(web_request) diff --git a/admin/tests/schemas/test_metadata.py b/admin/tests/schemas/test_metadata.py index 3931872747..8e05e4ac65 100644 --- a/admin/tests/schemas/test_metadata.py +++ b/admin/tests/schemas/test_metadata.py @@ -11,7 +11,9 @@ def test_get_relevant_for(): def test_metadata_definitions(): - from c2cgeoportal_admin.schemas.metadata import get_relevant_for, metadata_definitions + from c2cgeoportal_admin.schemas.metadata import ( + metadata_definitions, + ) from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS settings = { diff --git a/admin/tests/test_edit_url.py b/admin/tests/test_edit_url.py index 24e20f0c2b..bf8f96af60 100644 --- a/admin/tests/test_edit_url.py +++ b/admin/tests/test_edit_url.py @@ -45,7 +45,7 @@ def edit_url_test_data(dbsession, transact): layer_wmts.restrictionareas = [restrictionareas[i % 5], restrictionareas[(i + 2) % 5]] if i % 10 != 1: layer_wmts.interfaces = [interfaces[i % 4], interfaces[(i + 2) % 4]] - layer_wmts.public = 1 == i % 2 + layer_wmts.public = i % 2 == 1 layer_wmts.image_type = "image/jpeg" dbsession.add(layer_wmts) layers_wmts.append(layer_wmts) diff --git a/admin/tests/test_functionalities.py b/admin/tests/test_functionalities.py index 5331d3dcad..a907b6a66a 100644 --- a/admin/tests/test_functionalities.py +++ b/admin/tests/test_functionalities.py @@ -66,7 +66,7 @@ def test_submit_new(self, dbsession, test_app): assert functionality.name == "new_name" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "functionality" assert log.element_id == functionality.id @@ -80,14 +80,14 @@ def test_edit(self, test_app, functionality_test_data, dbsession): resp = test_app.get(f"/admin/functionalities/{functionality.id}", status=200) form = resp.form assert str(functionality.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert functionality.name == form["name"].value form["description"] = "new_description" assert form.submit().status_int == 302 assert functionality.description == "new_description" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "functionality" assert log.element_id == functionality.id @@ -103,7 +103,7 @@ def test_delete(self, test_app, functionality_test_data, dbsession): assert dbsession.query(Functionality).get(deleted_id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "functionality" assert log.element_id == functionality.id diff --git a/admin/tests/test_home.py b/admin/tests/test_home.py index 5c1a8cb0d4..1445238436 100644 --- a/admin/tests/test_home.py +++ b/admin/tests/test_home.py @@ -9,6 +9,6 @@ class TestHome(AbstractViewsTests): def test_index_rendering(self, test_app): resp = self.get(test_app, status=302) - assert "http://localhost/admin/layertree" == resp.location + assert resp.location == "http://localhost/admin/layertree" to_layer_tree = resp.follow() assert to_layer_tree.html.select_one('div[id="layertree"]') is not None diff --git a/admin/tests/test_interface.py b/admin/tests/test_interface.py index 0b9dab763f..8a407fc9f3 100644 --- a/admin/tests/test_interface.py +++ b/admin/tests/test_interface.py @@ -24,7 +24,7 @@ def interface_test_data(dbsession, transact): layers = [] for i in range(0, 15): layer = LayerWMS(name=f"layer_wms_{i}") - layer.public = 1 == i % 2 + layer.public = i % 2 == 1 layer.ogc_server = servers[i % 4] dbsession.add(layer) layers.append(layer) @@ -94,7 +94,7 @@ def test_submit_new(self, dbsession, test_app): assert interface.name == "new_name" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "interface" assert log.element_id == interface.id @@ -112,13 +112,13 @@ def test_edit(self, test_app, interface_test_data, dbsession): form = resp.form form["description"] = descriptions assert str(interface.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert interface.name == self.get_first_field_named(form, "name").value assert form.submit().status_int == 302 assert len(dbsession.query(Interface).filter(Interface.description == descriptions).all()) == 1 log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "interface" assert log.element_id == interface.id @@ -133,7 +133,7 @@ def test_delete(self, test_app, interface_test_data, dbsession): assert len(dbsession.query(Interface).filter(Interface.id == interface.id).all()) == 0 log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "interface" assert log.element_id == interface.id @@ -144,6 +144,6 @@ def test_duplicate(self, interface_test_data, test_app): interface = interface_test_data["interfaces"][3] resp = test_app.get(f"/admin/interfaces/{interface.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert str(interface.description or "") == "description_3" assert form.submit().status_int == 302 diff --git a/admin/tests/test_layer_groups.py b/admin/tests/test_layer_groups.py index 804c6189c5..82c9d66559 100644 --- a/admin/tests/test_layer_groups.py +++ b/admin/tests/test_layer_groups.py @@ -12,7 +12,11 @@ def layer_groups_test_data(dbsession, transact): del transact - from c2cgeoportal_commons.models.main import LayerGroup, LayergroupTreeitem, Metadata + from c2cgeoportal_commons.models.main import ( + LayerGroup, + LayergroupTreeitem, + Metadata, + ) metadatas_protos = [ ("copyable", "true"), @@ -47,7 +51,7 @@ def flatten_tree(key, value): for val in value: add_relation(key, val) else: - for val in value.keys(): + for val in value: add_relation(key, val) flatten_tree(val, value[val]) @@ -85,8 +89,8 @@ def test_grid_complex_column_val(self, test_app, layer_groups_test_data): assert group.id == int(row["_id_"]) assert group.name == row["name"] - assert "groups_02, groups_06" == row["parents_relation"] - assert "disclaimer: © le momo, copyable: true" == row["metadatas"] + assert row["parents_relation"] == "groups_02, groups_06" + assert row["metadatas"] == "disclaimer: © le momo, copyable: true" def test_grid_search(self, test_app): # search on metadatas @@ -100,7 +104,7 @@ def test_edit(self, test_app, layer_groups_test_data, dbsession): form = self.get_item(test_app, group.id).form assert str(group.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert group.name == self.get_first_field_named(form, "name").value assert str(group.description or "") == self.get_first_field_named(form, "description").value @@ -136,7 +140,7 @@ def test_edit(self, test_app, layer_groups_test_data, dbsession): assert str(value or "") == str(getattr(group, key) or "") log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "layergroup" assert log.element_id == group.id @@ -144,9 +148,7 @@ def test_edit(self, test_app, layer_groups_test_data, dbsession): assert log.username == "test_user" def test_post_new_with_children_invalid(self, test_app, layer_groups_test_data): - """ - Check there is no rendering error when validation fails. - """ + """Check there is no rendering error when validation fails.""" groups = layer_groups_test_data["groups"] resp = test_app.post( f"{self._prefix}/new", @@ -165,7 +167,7 @@ def test_post_new_with_children_invalid(self, test_app, layer_groups_test_data): ), status=200, ) - assert "Required" == resp.html.select_one(".item-name .help-block").getText().strip() + assert resp.html.select_one(".item-name .help-block").getText().strip() == "Required" def test_post_new_with_children_success(self, test_app, dbsession, layer_groups_test_data): from c2cgeoportal_commons.models.main import Log, LogAction @@ -214,7 +216,7 @@ def test_post_new_with_children_success(self, test_app, dbsession, layer_groups_ ] log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "layergroup" assert log.element_id == group.id @@ -222,9 +224,7 @@ def test_post_new_with_children_success(self, test_app, dbsession, layer_groups_ assert log.username == "test_user" def test_post_with_ancestor(self, layer_groups_test_data, test_app): - """ - Check that ancestors are refused to avoid cycles. - """ + """Check that ancestors are refused to avoid cycles.""" groups = layer_groups_test_data["groups"] resp = test_app.post( f"{self._prefix}/{groups[3].id}", @@ -259,7 +259,7 @@ def test_duplicate(self, layer_groups_test_data, test_app, dbsession): group = dbsession.query(LayerGroup).filter(LayerGroup.id == group.id).one() - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert group.name == self.get_first_field_named(form, "name").value assert str(group.description or "") == self.get_first_field_named(form, "description").value @@ -305,13 +305,13 @@ def test_delete(self, test_app, dbsession, layer_groups_test_data): group = layer_groups_test_data["groups"][9] assert ( - 3 - == dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treegroup_id == group.id).count() + dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treegroup_id == group.id).count() + == 3 ) assert ( - 1 - == dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treeitem_id == group.id).count() + dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treeitem_id == group.id).count() + == 1 ) test_app.delete(f"/admin/layer_groups/{group.id}", status=200) @@ -323,17 +323,17 @@ def test_delete(self, test_app, dbsession, layer_groups_test_data): assert dbsession.query(TreeItem).get(group.id) is None assert ( - 0 - == dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treegroup_id == group.id).count() + dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treegroup_id == group.id).count() + == 0 ) assert ( - 0 - == dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treeitem_id == group.id).count() + dbsession.query(LayergroupTreeitem).filter(LayergroupTreeitem.treeitem_id == group.id).count() + == 0 ) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "layergroup" assert log.element_id == group.id diff --git a/admin/tests/test_layers_cog.py b/admin/tests/test_layers_cog.py index a89f013634..70d5cbaeb2 100644 --- a/admin/tests/test_layers_cog.py +++ b/admin/tests/test_layers_cog.py @@ -20,7 +20,7 @@ def layer_cog_test_data(dbsession: Session, transact: SessionTransaction) -> dic def layer_builder(i: int) -> LayerCOG: name = f"layer_cog_{i}" layer = LayerCOG(name=name) - layer.public = 1 == i % 2 + layer.public = i % 2 == 1 layer.url = "https://example.com/image.tiff" return layer @@ -73,9 +73,9 @@ def test_new(self, test_app: WebTestApp, layer_cog_test_data: dict[str, Any], db form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "url").value + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "url").value == "" def test_grid_search(self, test_app: WebTestApp) -> None: self.check_search(test_app, "layer_cog_10", total=1) @@ -85,8 +85,8 @@ def test_base_edit(self, test_app: WebTestApp, layer_cog_test_data: dict[str, An form = self.get_item(test_app, layer.id).form - assert "layer_cog_10" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "description").value + assert self.get_first_field_named(form, "name").value == "layer_cog_10" + assert self.get_first_field_named(form, "description").value == "" def test_public_checkbox_edit(self, test_app: WebTestApp, layer_cog_test_data: dict[str, Any]) -> None: layer = layer_cog_test_data["layers"][10] @@ -107,7 +107,7 @@ def test_edit( form = self.get_item(test_app, layer.id).form assert str(layer.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is False @@ -153,7 +153,7 @@ def test_edit( assert {ras[1].id, ras[3].id} == {ra.id for ra in layer.restrictionareas} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "layer_cog" assert log.element_id == layer.id @@ -182,7 +182,7 @@ def test_submit_new( ).group(1) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "layer_cog" assert log.element_id == layer.id @@ -199,7 +199,7 @@ def test_duplicate( resp = test_app.get(f"/admin/layers_cog/{layer.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is True @@ -224,7 +224,13 @@ def test_duplicate( assert layer_cog_test_data["layers"][3].metadatas[1].name == layer.metadatas[1].name def test_delete(self, test_app: WebTestApp, dbsession: Session) -> None: - from c2cgeoportal_commons.models.main import Layer, LayerCOG, Log, LogAction, TreeItem + from c2cgeoportal_commons.models.main import ( + Layer, + LayerCOG, + Log, + LogAction, + TreeItem, + ) layer = dbsession.query(LayerCOG).first() @@ -235,7 +241,7 @@ def test_delete(self, test_app: WebTestApp, dbsession: Session) -> None: assert dbsession.query(TreeItem).get(layer.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "layer_cog" assert log.element_id == layer.id diff --git a/admin/tests/test_layers_vectortiles.py b/admin/tests/test_layers_vectortiles.py index 6e1c757284..0e8280c893 100644 --- a/admin/tests/test_layers_vectortiles.py +++ b/admin/tests/test_layers_vectortiles.py @@ -18,7 +18,7 @@ def layer_builder(i): name = f"layer_vectortiles_{i}" layer = LayerVectorTiles(name=name) layer.layer = name - layer.public = 1 == i % 2 + layer.public = i % 2 == 1 layer.style = "https://vectortiles-staging.geoportail.lu/styles/roadmap/style.json" layer.xyz = "https://vectortiles-staging.geoportail.lu/styles/roadmap/{z}/{x}/{y}.png" return layer @@ -74,10 +74,10 @@ def test_new(self, test_app, layer_vectortiles_test_data, dbsession): form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "style").value - assert "" == self.get_first_field_named(form, "xyz").value + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "style").value == "" + assert self.get_first_field_named(form, "xyz").value == "" def test_grid_search(self, test_app): self.check_search(test_app, "layer_vectortiles_10", total=1) @@ -87,8 +87,8 @@ def test_base_edit(self, test_app, layer_vectortiles_test_data): form = self.get_item(test_app, layer.id).form - assert "layer_vectortiles_10" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "description").value + assert self.get_first_field_named(form, "name").value == "layer_vectortiles_10" + assert self.get_first_field_named(form, "description").value == "" def test_public_checkbox_edit(self, test_app, layer_vectortiles_test_data): layer = layer_vectortiles_test_data["layers"][10] @@ -107,7 +107,7 @@ def test_edit(self, test_app, layer_vectortiles_test_data, dbsession): form = self.get_item(test_app, layer.id).form assert str(layer.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is False @@ -155,7 +155,7 @@ def test_edit(self, test_app, layer_vectortiles_test_data, dbsession): assert {ras[1].id, ras[3].id} == {ra.id for ra in layer.restrictionareas} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "layer_vectortiles" assert log.element_id == layer.id @@ -183,7 +183,7 @@ def test_submit_new(self, dbsession, test_app, layer_vectortiles_test_data): ).group(1) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "layer_vectortiles" assert log.element_id == layer.id @@ -198,7 +198,7 @@ def test_duplicate(self, layer_vectortiles_test_data, test_app, dbsession): resp = test_app.get(f"/admin/layers_vectortiles/{layer.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is True @@ -224,7 +224,13 @@ def test_duplicate(self, layer_vectortiles_test_data, test_app, dbsession): assert layer_vectortiles_test_data["layers"][3].metadatas[1].name == layer.metadatas[1].name def test_delete(self, test_app, dbsession): - from c2cgeoportal_commons.models.main import Layer, LayerVectorTiles, Log, LogAction, TreeItem + from c2cgeoportal_commons.models.main import ( + Layer, + LayerVectorTiles, + Log, + LogAction, + TreeItem, + ) layer = dbsession.query(LayerVectorTiles).first() @@ -235,7 +241,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(TreeItem).get(layer.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "layer_vectortiles" assert log.element_id == layer.id diff --git a/admin/tests/test_layers_wms.py b/admin/tests/test_layers_wms.py index d636574442..08dcaad703 100644 --- a/admin/tests/test_layers_wms.py +++ b/admin/tests/test_layers_wms.py @@ -22,7 +22,7 @@ def layer_wms_test_data(dbsession, transact): def layer_builder(i): layer = LayerWMS(name=f"layer_wms_{i}") layer.layer = f"layer_{i}" - layer.public = 1 == i % 2 + layer.public = i % 2 == 1 layer.geo_table = f"geotable_{i}" layer.ogc_server = servers[i % 4] layer.style = "décontrasté" @@ -77,12 +77,12 @@ def test_grid_complex_column_val(self, test_app, layer_wms_test_data): assert layer.id == int(row["_id_"]) assert layer.name == row["name"] - assert "restrictionarea_0, restrictionarea_2" == row["restrictionareas"] - assert "server_0" == row["ogc_server"] - assert "desktop, edit" == row["interfaces"] - assert "Date: 2017, 1988; CLC: all" == row["dimensions"] - assert "layer_group_0, layer_group_3" == row["parents_relation"] - assert 'copyable: true, snappingConfig: {"tolerance": 50}' == row["metadatas"] + assert row["restrictionareas"] == "restrictionarea_0, restrictionarea_2" + assert row["ogc_server"] == "server_0" + assert row["interfaces"] == "desktop, edit" + assert row["dimensions"] == "Date: 2017, 1988; CLC: all" + assert row["parents_relation"] == "layer_group_0, layer_group_3" + assert row["metadatas"] == 'copyable: true, snappingConfig: {"tolerance": 50}' def test_grid_sort_on_ogc_server(self, test_app, layer_wms_test_data): json = self.check_search(test_app, sort="ogc_server") @@ -116,21 +116,21 @@ def test_new_no_default(self, test_app, layer_wms_test_data, dbsession): form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "layer").value - assert "" == self.get_first_field_named(form, "ogc_server_id").value - assert "disabled" == self.get_first_field_named(form, "time_mode").value - assert "slider" == self.get_first_field_named(form, "time_widget").value + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "layer").value == "" + assert self.get_first_field_named(form, "ogc_server_id").value == "" + assert self.get_first_field_named(form, "time_mode").value == "disabled" + assert self.get_first_field_named(form, "time_widget").value == "slider" def test_new_default(self, test_app, layer_wms_test_data): default_wms = layer_wms_test_data["default"]["wms"] form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "layer").value + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "layer").value == "" assert str(default_wms.ogc_server.id) == self.get_first_field_named(form, "ogc_server_id").value assert default_wms.time_mode == self.get_first_field_named(form, "time_mode").value assert default_wms.time_widget == self.get_first_field_named(form, "time_widget").value @@ -140,8 +140,8 @@ def test_base_edit(self, test_app, layer_wms_test_data): form = self.get_item(test_app, layer.id).form - assert "layer_wms_10" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "description").value + assert self.get_first_field_named(form, "name").value == "layer_wms_10" + assert self.get_first_field_named(form, "description").value == "" def test_public_checkbox_edit(self, test_app, layer_wms_test_data): layer = layer_wms_test_data["layers"][10] @@ -160,7 +160,7 @@ def test_edit(self, test_app, layer_wms_test_data, dbsession): form = self.get_item(test_app, layer.id).form assert str(layer.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is False @@ -214,7 +214,7 @@ def test_edit(self, test_app, layer_wms_test_data, dbsession): assert {ras[1].id, ras[3].id} == {ra.id for ra in layer.restrictionareas} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "layer_wms" assert log.element_id == layer.id @@ -247,7 +247,7 @@ def test_submit_new(self, dbsession, test_app, layer_wms_test_data): ).group(1) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "layer_wms" assert log.element_id == layer.id @@ -262,7 +262,7 @@ def test_duplicate(self, layer_wms_test_data, test_app, dbsession): resp = test_app.get(f"/admin/layers_wms/{layer.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is True @@ -301,15 +301,15 @@ def test_convert_common_fields_copied(self, layer_wms_test_data, test_app, dbses layer = layer_wms_test_data["layers"][3] - assert 0 == dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() - assert 1 == dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() + assert dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() == 0 + assert dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() == 1 resp = test_app.post(f"/admin/layers_wms/{layer.id}/convert_to_wmts", status=200) assert resp.json["success"] assert f"http://localhost/admin/layers_wmts/{layer.id}?msg_col=submit_ok" == resp.json["redirect"] - assert 1 == dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() - assert 0 == dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() + assert dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() == 1 + assert dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() == 0 resp = test_app.get(resp.json["redirect"], status=200) form = resp.form @@ -333,8 +333,8 @@ def test_convert_common_fields_copied(self, layer_wms_test_data, test_app, dbses self._check_dimensions(resp.html, layer.dimensions) assert ( - "Your submission has been taken into account." - == resp.html.find("div", {"class": "msg-lbl"}).getText() + resp.html.find("div", {"class": "msg-lbl"}).getText() + == "Your submission has been taken into account." ) def test_convert_image_type_from_ogcserver(self, layer_wms_test_data, test_app): @@ -345,7 +345,7 @@ def test_convert_image_type_from_ogcserver(self, layer_wms_test_data, test_app): assert f"http://localhost/admin/layers_wmts/{layer.id}?msg_col=submit_ok" == resp.json["redirect"] resp = test_app.get(resp.json["redirect"], status=200) - assert "image/png" == resp.form["image_type"].value + assert resp.form["image_type"].value == "image/png" layer = layer_wms_test_data["layers"][2] resp = test_app.post(f"/admin/layers_wms/{layer.id}/convert_to_wmts", status=200) @@ -353,7 +353,7 @@ def test_convert_image_type_from_ogcserver(self, layer_wms_test_data, test_app): assert f"http://localhost/admin/layers_wmts/{layer.id}?msg_col=submit_ok" == resp.json["redirect"] resp = test_app.get(resp.json["redirect"], status=200) - assert "image/jpeg" == resp.form["image_type"].value + assert resp.form["image_type"].value == "image/jpeg" def test_convert_without_wmts_defaults(self, test_app, layer_wms_test_data, dbsession): from c2cgeoportal_commons.models.main import LayerWMTS, Log, LogAction @@ -363,7 +363,7 @@ def test_convert_without_wmts_defaults(self, test_app, layer_wms_test_data, dbse test_app.post(f"/admin/layers_wms/{layer.id}/convert_to_wmts", status=200) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.CONVERT_TO_WMTS assert log.element_type == "layer_wms" assert log.element_id == layer.id @@ -381,7 +381,7 @@ def test_unicity_validator(self, layer_wms_test_data, test_app): def test_unicity_validator_does_not_matter_amongst_cousin(self, layer_wms_test_data, test_app, dbsession): from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS - assert 1 == dbsession.query(LayerGroup).filter(LayerGroup.name == "layer_group_0").count() + assert dbsession.query(LayerGroup).filter(LayerGroup.name == "layer_group_0").count() == 1 assert dbsession.query(LayerWMS).filter(LayerWMS.name == "layer_group_0").one_or_none() is None @@ -396,7 +396,13 @@ def test_unicity_validator_does_not_matter_amongst_cousin(self, layer_wms_test_d # assert str(layer.id) == re.match('http://localhost/admin/layers_wms/(.*)', resp.location).group(1) def test_delete(self, test_app, dbsession): - from c2cgeoportal_commons.models.main import Layer, LayerWMS, Log, LogAction, TreeItem + from c2cgeoportal_commons.models.main import ( + Layer, + LayerWMS, + Log, + LogAction, + TreeItem, + ) layer = dbsession.query(LayerWMS).first() @@ -407,7 +413,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(TreeItem).get(layer.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "layer_wms" assert log.element_id == layer.id @@ -432,12 +438,12 @@ def test_submit_new_no_layer_name(self, test_app, layer_wms_test_data, dbsession ) assert ( - "There was a problem with your submission" - == resp.html.select_one('div[class="error-msg-lbl"]').text + resp.html.select_one('div[class="error-msg-lbl"]').text + == "There was a problem with your submission" ) assert ( - "Errors have been highlighted below" == resp.html.select_one('div[class="error-msg-detail"]').text + resp.html.select_one('div[class="error-msg-detail"]').text == "Errors have been highlighted below" ) - assert ["WMS layer name"] == sorted( + assert sorted( (x.select_one("label").text.strip()) for x in resp.html.select("[class~='has-error']") - ) + ) == ["WMS layer name"] diff --git a/admin/tests/test_layers_wmts.py b/admin/tests/test_layers_wmts.py index 5eed5e5f99..c47a612af9 100644 --- a/admin/tests/test_layers_wmts.py +++ b/admin/tests/test_layers_wmts.py @@ -23,7 +23,7 @@ def layer_builder(i): layer = LayerWMTS(name=name) layer.layer = name layer.url = f"https:///wms.geo.admin.ch_{i}.org?service=wms&request=GetCapabilities" - layer.public = 1 == i % 2 + layer.public = i % 2 == 1 layer.geo_table = f"geotable_{i}" layer.image_type = "image/jpeg" layer.style = "décontrasté" @@ -83,20 +83,20 @@ def test_new_no_default(self, test_app, layer_wmts_test_data, dbsession): form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "layer").value - assert "" == self.get_first_field_named(form, "url").value - assert "" == self.get_first_field_named(form, "matrix_set").value + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "layer").value == "" + assert self.get_first_field_named(form, "url").value == "" + assert self.get_first_field_named(form, "matrix_set").value == "" def test_new_default(self, test_app, layer_wmts_test_data): default_wmts = layer_wmts_test_data["default"]["wmts"] form = self.get_item(test_app, "new").form - assert "" == self.get_first_field_named(form, "id").value - assert "" == self.get_first_field_named(form, "name").value - assert "" == self.get_first_field_named(form, "layer").value + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "name").value == "" + assert self.get_first_field_named(form, "layer").value == "" assert default_wmts.url == self.get_first_field_named(form, "url").value assert default_wmts.matrix_set == self.get_first_field_named(form, "matrix_set").value @@ -108,7 +108,7 @@ def test_edit(self, test_app, layer_wmts_test_data, dbsession): form = self.get_item(test_app, layer.id).form assert str(layer.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is False @@ -162,7 +162,7 @@ def test_edit(self, test_app, layer_wmts_test_data, dbsession): assert {ras[1].id, ras[3].id} == {ra.id for ra in layer.restrictionareas} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "layer_wmts" assert log.element_id == layer.id @@ -177,7 +177,7 @@ def test_duplicate(self, layer_wmts_test_data, test_app, dbsession): resp = test_app.get(f"/admin/layers_wmts/{layer.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert layer.name == self.get_first_field_named(form, "name").value assert str(layer.description or "") == self.get_first_field_named(form, "description").value assert layer.public is True @@ -199,7 +199,13 @@ def test_duplicate(self, layer_wmts_test_data, test_app, dbsession): ).group(1) def test_delete(self, test_app, dbsession): - from c2cgeoportal_commons.models.main import Layer, LayerWMTS, Log, LogAction, TreeItem + from c2cgeoportal_commons.models.main import ( + Layer, + LayerWMTS, + Log, + LogAction, + TreeItem, + ) layer = dbsession.query(LayerWMTS).first() @@ -210,7 +216,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(TreeItem).get(layer.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "layer_wmts" assert log.element_id == layer.id @@ -230,14 +236,14 @@ def test_convert_common_fields_copied(self, layer_wmts_test_data, test_app, dbse layer = layer_wmts_test_data["layers"][3] - assert 0 == dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() - assert 1 == dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() + assert dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() == 0 + assert dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() == 1 resp = test_app.post(f"/admin/layers_wmts/{layer.id}/convert_to_wms", status=200) assert f"http://localhost/admin/layers_wms/{layer.id}?msg_col=submit_ok" == resp.json["redirect"] - assert 1 == dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() - assert 0 == dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() + assert dbsession.query(LayerWMS).filter(LayerWMS.name == layer.name).count() == 1 + assert dbsession.query(LayerWMTS).filter(LayerWMTS.name == layer.name).count() == 0 resp = test_app.get(resp.json["redirect"], status=200) form = resp.form @@ -260,8 +266,8 @@ def test_convert_common_fields_copied(self, layer_wmts_test_data, test_app, dbse self._check_restrictionsareas(form, ras, layer) self._check_dimensions(resp.html, layer.dimensions) assert ( - "Your submission has been taken into account." - == resp.html.find("div", {"class": "msg-lbl"}).getText() + resp.html.find("div", {"class": "msg-lbl"}).getText() + == "Your submission has been taken into account." ) def test_convert_without_wms_defaults(self, test_app, layer_wmts_test_data, dbsession): @@ -272,7 +278,7 @@ def test_convert_without_wms_defaults(self, test_app, layer_wmts_test_data, dbse test_app.post(f"/admin/layers_wmts/{layer.id}/convert_to_wms", status=200) log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.CONVERT_TO_WMS assert log.element_type == "layer_wmts" assert log.element_id == layer.id diff --git a/admin/tests/test_layertree.py b/admin/tests/test_layertree.py index e135321470..857d5dbf84 100644 --- a/admin/tests/test_layertree.py +++ b/admin/tests/test_layertree.py @@ -140,7 +140,7 @@ def check_new_action(self, test_app, nodes, parent_id, action_name, label, route def test_themes(self, test_app, layertree_test_data): resp = self.get(test_app, "/children", status=200) nodes = resp.json - assert 5 == len(nodes) + assert len(nodes) == 5 # check themes are sorted by ordering expected = [ @@ -154,7 +154,7 @@ def test_themes(self, test_app, layertree_test_data): # no unlink on theme theme_node = next(n for n in nodes if n["id"] == theme.id) - assert 0 == len([a for a in theme_node["actions"] if a["name"] == "unlink"]) + assert len([a for a in theme_node["actions"] if a["name"] == "unlink"]) == 0 self.check_translation(nodes, theme) @@ -176,9 +176,9 @@ def test_groups(self, test_app, layertree_test_data, dbsession): theme.children_relation[1].ordering = 0 dbsession.flush() - resp = self.get(test_app, "/children?group_id={0}&path=_{0}".format(theme.id), status=200) + resp = self.get(test_app, f"/children?group_id={theme.id}&path=_{theme.id}", status=200) nodes = resp.json - assert 2 == len(nodes) + assert len(nodes) == 2 # check groups are sorted by ordering expected = [ @@ -227,9 +227,7 @@ def test_layers(self, test_app, layertree_test_data, dbsession): group.children_relation[1].ordering = 0 dbsession.flush() - resp = self.get( - test_app, "/children?group_id={0}&path=_{1}_{0}".format(group.id, theme.id), status=200 - ) + resp = self.get(test_app, f"/children?group_id={group.id}&path=_{theme.id}_{group.id}", status=200) nodes = resp.json assert len(nodes) == 2 diff --git a/admin/tests/test_lingva_extractor_config.py b/admin/tests/test_lingva_extractor_config.py index 89b9035d7a..7e7a16a638 100644 --- a/admin/tests/test_lingva_extractor_config.py +++ b/admin/tests/test_lingva_extractor_config.py @@ -29,10 +29,6 @@ from unittest.mock import Mock, mock_open, patch -import pytest -import yaml -from c2c.template.config import config as configuration - from c2cgeoportal_admin.lib.lingva_extractor import GeomapfishConfigExtractor GMF_CONFIG = """ diff --git a/admin/tests/test_logs.py b/admin/tests/test_logs.py index 48c4439b77..875098b9e1 100644 --- a/admin/tests/test_logs.py +++ b/admin/tests/test_logs.py @@ -13,7 +13,6 @@ def logs_test_data(dbsession, transact): del transact - from c2cgeoportal_commons.models.main import AbstractLog from c2cgeoportal_commons.models.main import Log as MainLog from c2cgeoportal_commons.models.main import LogAction from c2cgeoportal_commons.models.static import Log as StaticLog diff --git a/admin/tests/test_main.py b/admin/tests/test_main.py index fb9c0287d1..10bc80e8b9 100644 --- a/admin/tests/test_main.py +++ b/admin/tests/test_main.py @@ -5,9 +5,7 @@ @pytest.mark.usefixtures("app_env") def test_main(app_env): - """ - Test dev environment. - """ + """Test dev environment.""" config = testing.setUp(registry=app_env["registry"]) app = config.make_wsgi_app() testapp = WebTestApp(app) diff --git a/admin/tests/test_metadatas.py b/admin/tests/test_metadatas.py index cfd8600c86..8f9d7bc6e4 100644 --- a/admin/tests/test_metadatas.py +++ b/admin/tests/test_metadatas.py @@ -10,7 +10,14 @@ def metadatas_test_data(dbsession, transact): del transact - from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS, LayerWMTS, Metadata, OGCServer, Theme + from c2cgeoportal_commons.models.main import ( + LayerGroup, + LayerWMS, + LayerWMTS, + Metadata, + OGCServer, + Theme, + ) ogc_server = OGCServer(name="ogc_server") @@ -333,9 +340,7 @@ def test_group_metadatas(self, metadatas_test_data, test_app): self._test_edit_treeitem("layer_groups", metadatas_test_data["group"], test_app, LayerGroup) def test_undefined_metadata(self, metadatas_test_data, test_app): - """ - Undefined metadata must be kept intact across submissions. - """ + """Undefined metadata must be kept intact across submissions.""" from c2cgeoportal_commons.models.main import Metadata layer = metadatas_test_data["layer_wms"] diff --git a/admin/tests/test_oauth2_clients.py b/admin/tests/test_oauth2_clients.py index b1febe74b4..6289f1d18e 100644 --- a/admin/tests/test_oauth2_clients.py +++ b/admin/tests/test_oauth2_clients.py @@ -18,7 +18,7 @@ def oauth2_clients_test_data(dbsession, transact): from c2cgeoportal_commons.models.static import OAuth2Client clients = [] - for i in range(23): + for _i in range(23): client = OAuth2Client() client.client_id = str(uuid4()) client.secret = "1234" @@ -86,7 +86,7 @@ def test_submit_new(self, dbsession, test_app, oauth2_clients_test_data): assert oauth2_client.redirect_uri == "http://127.0.0.1:7070/bis" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "oauth2_client" assert log.element_id == oauth2_client.id @@ -120,12 +120,12 @@ def test_edit_then_save(self, dbsession, test_app, oauth2_clients_test_data): dbsession.expire(oauth2_client) - assert "New client ID" == oauth2_client.client_id - assert "New secret" == oauth2_client.secret - assert "New redirect URI" == oauth2_client.redirect_uri + assert oauth2_client.client_id == "New client ID" + assert oauth2_client.secret == "New secret" + assert oauth2_client.redirect_uri == "New redirect URI" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "oauth2_client" assert log.element_id == oauth2_client.id @@ -140,7 +140,7 @@ def test_duplicate(self, oauth2_clients_test_data, test_app, dbsession): resp = test_app.get(f"/admin/oauth2_clients/{oauth2_client_proto.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert oauth2_client_proto.client_id == form["client_id"].value assert oauth2_client_proto.secret == form["secret"].value assert oauth2_client_proto.redirect_uri == form["redirect_uri"].value @@ -162,7 +162,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(OAuth2Client).get(oauth2_client.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "oauth2_client" assert log.element_id == oauth2_client.id diff --git a/admin/tests/test_ogc_servers.py b/admin/tests/test_ogc_servers.py index c072a25707..16a0c7aec8 100644 --- a/admin/tests/test_ogc_servers.py +++ b/admin/tests/test_ogc_servers.py @@ -81,7 +81,7 @@ def test_submit_new(self, dbsession, test_app): assert ogc_server.name == "new_name" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "ogc_server" assert log.element_id == ogc_server.id @@ -95,7 +95,7 @@ def test_edit(self, test_app, ogc_server_test_data, dbsession): resp = test_app.get(f"/admin/ogc_servers/{ogc_server.id}", status=200) form = resp.form assert str(ogc_server.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert ogc_server.name == form["name"].value form["description"] = "new_description" with patch("c2cgeoportal_admin.views.ogc_servers.OGCServerViews._update_cache"): @@ -103,7 +103,7 @@ def test_edit(self, test_app, ogc_server_test_data, dbsession): assert ogc_server.description == "new_description" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "ogc_server" assert log.element_id == ogc_server.id @@ -118,7 +118,7 @@ def test_delete(self, test_app, ogc_server_test_data, dbsession): assert dbsession.query(OGCServer).get(ogc_server.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "ogc_server" assert log.element_id == ogc_server.id @@ -131,7 +131,7 @@ def test_duplicate(self, ogc_server_test_data, test_app, dbsession): ogc_server = ogc_server_test_data["ogc_servers"][3] resp = test_app.get(f"/admin/ogc_servers/{ogc_server.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" self.set_first_field_named(form, "name", "clone") with patch("c2cgeoportal_admin.views.ogc_servers.OGCServerViews._update_cache"): resp = form.submit("submit") @@ -181,7 +181,7 @@ def test_synchronize_success(self, ogc_server_test_data, test_app, dbsession): resp = resp.forms["form-synchronize"].submit("synchronize") log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.SYNCHRONIZE assert log.element_type == "ogc_server" assert log.element_id == ogc_server.id diff --git a/admin/tests/test_restriction_areas.py b/admin/tests/test_restriction_areas.py index 71edeae5c5..be5e85832c 100644 --- a/admin/tests/test_restriction_areas.py +++ b/admin/tests/test_restriction_areas.py @@ -15,7 +15,12 @@ @pytest.mark.usefixtures("dbsession", "transact") def restriction_area_test_data(dbsession, transact): del transact - from c2cgeoportal_commons.models.main import LayerWMS, OGCServer, RestrictionArea, Role + from c2cgeoportal_commons.models.main import ( + LayerWMS, + OGCServer, + RestrictionArea, + Role, + ) roles = [] for i in range(0, 4): @@ -116,7 +121,7 @@ def test_submit_new(self, dbsession, test_app, restriction_area_test_data): assert set(restriction_area.layers) == {layers[0], layers[1]} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "restrictionarea" assert log.element_id == restriction_area.id @@ -143,7 +148,7 @@ def test_edit(self, test_app, restriction_area_test_data, dbsession): form = self.get_item(test_app, restriction_area.id).form assert str(restriction_area.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert restriction_area.name == form["name"].value expected = Polygon( [ @@ -174,7 +179,7 @@ def test_edit(self, test_app, restriction_area_test_data, dbsession): assert set(restriction_area.roles) == {roles[i] for i in range(0, 3)} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "restrictionarea" assert log.element_id == restriction_area.id @@ -190,7 +195,7 @@ def test_delete(self, test_app, restriction_area_test_data, dbsession): assert dbsession.query(RestrictionArea).get(deleted_id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "restrictionarea" assert log.element_id == restriction_area.id @@ -208,7 +213,7 @@ def test_duplicate(self, restriction_area_test_data, test_app, dbsession): form = test_app.get(f"/admin/restriction_areas/{restriction_area.id}/duplicate", status=200).form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" self._check_roles(form, roles, restriction_area) self.check_children( form, diff --git a/admin/tests/test_role.py b/admin/tests/test_role.py index ed0248e4f3..94b65e4057 100644 --- a/admin/tests/test_role.py +++ b/admin/tests/test_role.py @@ -155,7 +155,7 @@ def test_submit_new(self, dbsession, test_app, roles_test_data): assert set(role.users) == {users[0], users[1]} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "role" assert log.element_id == role.id @@ -172,7 +172,7 @@ def test_edit(self, dbsession, test_app, roles_test_data): form = self.get_item(test_app, role.id).form - assert "secretary_10" == form["name"].value + assert form["name"].value == "secretary_10" expected = Polygon( [ @@ -266,8 +266,8 @@ def test_edit(self, dbsession, test_app, roles_test_data): dbsession.expire(role) - assert "New name" == role.name - assert "New description" == role.description + assert role.name == "New name" + assert role.description == "New description" expected = Polygon( [ @@ -284,7 +284,7 @@ def test_edit(self, dbsession, test_app, roles_test_data): assert set(ra_ids) == {f.id for f in role.restrictionareas} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "role" assert log.element_id == role.id @@ -299,7 +299,7 @@ def test_duplicate(self, roles_test_data, test_app, dbsession): resp = test_app.get(f"/admin/roles/{role_proto.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value + assert self.get_first_field_named(form, "id").value == "" assert role_proto.name == form["name"].value assert role_proto.description == form["description"].value form["name"].value = "clone" @@ -322,7 +322,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(Role).get(role.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "role" assert log.element_id == role.id diff --git a/admin/tests/test_themes.py b/admin/tests/test_themes.py index 20418f77a0..ee93610f45 100644 --- a/admin/tests/test_themes.py +++ b/admin/tests/test_themes.py @@ -54,7 +54,7 @@ def theme_test_data(dbsession, transact): themes = [] for i in range(0, 25): theme = Theme(name=f"theme_{i}", ordering=1, icon=f"icon_{i}") - theme.public = 1 == i % 2 + theme.public = i % 2 == 1 theme.interfaces = [interfaces[i % 4], interfaces[(i + 2) % 4]] theme.metadatas = [ Metadata(name=metadatas_protos[id][0], value=metadatas_protos[id][1]) @@ -121,10 +121,10 @@ def test_grid_complex_column_val(self, test_app, theme_test_data): assert first_theme.id == int(first_row["_id_"]) assert first_theme.name == first_row["name"] - assert "default_basemap=value_0" == first_row["functionalities"] - assert "secretary_0, secretary_2" == first_row["restricted_roles"] - assert "desktop, edit" == first_row["interfaces"] - assert 'copyable: true, snappingConfig: {"tolerance": 50}' == first_row["metadatas"] + assert first_row["functionalities"] == "default_basemap=value_0" + assert first_row["restricted_roles"] == "secretary_0, secretary_2" + assert first_row["interfaces"] == "desktop, edit" + assert first_row["metadatas"] == 'copyable: true, snappingConfig: {"tolerance": 50}' def test_grid_search(self, test_app): # search on metadatas key and value parts @@ -165,7 +165,7 @@ def test_edit(self, test_app, theme_test_data, dbsession): form = resp.form assert str(theme.id) == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert theme.name == self.get_first_field_named(form, "name").value assert str(theme.description or "") == self.get_first_field_named(form, "description").value assert str(theme.ordering or "") == self.get_first_field_named(form, "ordering").value @@ -231,10 +231,10 @@ def test_edit(self, test_app, theme_test_data, dbsession): assert str(value or "") == str(getattr(theme, key) or "") assert {interfaces[1].id, interfaces[3].id} == {interface.id for interface in theme.interfaces} assert {functionalities[2].id} == {functionality.id for functionality in theme.functionalities} - assert 0 == len(theme.restricted_roles) + assert len(theme.restricted_roles) == 0 log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "theme" assert log.element_id == theme.id @@ -242,9 +242,7 @@ def test_edit(self, test_app, theme_test_data, dbsession): assert log.username == "test_user" def test_post_new_with_children_invalid(self, test_app, theme_test_data): - """ - Check there is no rendering error when validation fails. - """ + """Check there is no rendering error when validation fails.""" groups = theme_test_data["groups"] resp = test_app.post( f"{self._prefix}/new", @@ -314,7 +312,7 @@ def test_post_new_with_children_success(self, test_app, dbsession, theme_test_da ] log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "theme" assert log.element_id == theme.id @@ -322,9 +320,7 @@ def test_post_new_with_children_success(self, test_app, dbsession, theme_test_da assert log.username == "test_user" def test_post_new_with_child_layer(self, theme_test_data, test_app): - """ - Check layers are rejected by the validator (also means that they are not proposed to the user). - """ + """Check layers are rejected by the validator (also means that they are not proposed to the user).""" layers = theme_test_data["layers"] resp = test_app.post( f"{self._prefix}/new", @@ -359,8 +355,8 @@ def test_duplicate(self, theme_test_data, test_app, dbsession): resp = test_app.get(f"{self._prefix}/{theme.id}/duplicate", status=200) form = resp.form - assert "" == self.get_first_field_named(form, "id").value - assert "hidden" == self.get_first_field_named(form, "id").attrs["type"] + assert self.get_first_field_named(form, "id").value == "" + assert self.get_first_field_named(form, "id").attrs["type"] == "hidden" assert theme.name == self.get_first_field_named(form, "name").value assert str(theme.description or "") == self.get_first_field_named(form, "description").value assert str(theme.ordering or "") == self.get_first_field_named(form, "ordering").value @@ -421,7 +417,7 @@ def test_delete(self, test_app, dbsession): assert dbsession.query(Theme).get(theme.id) is None log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "theme" assert log.element_id == theme.id diff --git a/admin/tests/test_treegroup.py b/admin/tests/test_treegroup.py index b2a4aa2c5d..ac063d188f 100644 --- a/admin/tests/test_treegroup.py +++ b/admin/tests/test_treegroup.py @@ -8,7 +8,7 @@ def check_children(self, form, group, expected): form_group = form.html.select_one(f".item-{group}") items = form_group.select(".deform-seq-item") assert len(expected) == len(items) - for item, exp in zip(items, expected): + for item, exp in zip(items, expected, strict=False): assert exp["label"] == item.select_one(".well").getText().strip() for key, value in exp["values"].items(): assert value == item.select_one(f'input[name="{key}"]')["value"] diff --git a/admin/tests/test_user.py b/admin/tests/test_user.py index 956a7660ac..d16a36fc4c 100644 --- a/admin/tests/test_user.py +++ b/admin/tests/test_user.py @@ -127,7 +127,7 @@ def test_view_edit(self, test_app, users_test_data, dbsession): assert {roles[2].id, roles[3].id} == {role.id for role in user.roles} log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.UPDATE assert log.element_type == "user" assert log.element_id == user.id @@ -145,7 +145,7 @@ def test_delete(self, test_app, users_test_data, dbsession): assert dbsession.query(user_role).filter(user_role.c.user_id == user.id).count() == 0 log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.DELETE assert log.element_type == "user" assert log.element_id == user.id @@ -224,7 +224,7 @@ def test_duplicate(self, pw_gen_mock, smtp_mock, users_test_data, test_app, dbse resp = test_app.get(f"/admin/users/{user.id}/duplicate", status=200) form = resp.form - assert "" == form["id"].value + assert form["id"].value == "" assert user.username == form["username"].value assert user.email == form["email"].value assert str(user.settings_role_id) == form["settings_role_id"].value @@ -249,7 +249,7 @@ def test_duplicate(self, pw_gen_mock, smtp_mock, users_test_data, test_app, dbse assert EXPECTED_WELCOME_MAIL.format("clone", "clone", "basile") == parts[1].get_payload( decode=True ).decode("utf8") - assert "mail7@valid.net" == parts[0].items()[3][1] + assert parts[0].items()[3][1] == "mail7@valid.net" @patch("c2cgeoportal_commons.lib.email_.smtplib.SMTP") @patch("c2cgeoportal_admin.views.users.pwgenerator.generate") @@ -299,10 +299,10 @@ def test_submit_new(self, pw_gen_mock, smtp_mock, dbsession, test_app, users_tes assert EXPECTED_WELCOME_MAIL.format("new_user", "new_user", "basile") == parts[1].get_payload( decode=True ).decode("utf8") - assert "valid@email.net" == parts[0].items()[3][1] + assert parts[0].items()[3][1] == "valid@email.net" log = dbsession.query(Log).one() - assert log.date != None + assert log.date is not None assert log.action == LogAction.INSERT assert log.element_type == "user" assert log.element_id == user.id @@ -339,9 +339,7 @@ def test_grid_dberror(self, dbsession): UserViews(request).grid() def test_grid_settings_role_none(self, dbsession, test_app): - """ - Grid view must work even if a user's settings_role is None. - """ + """Grid view must work even if a user's settings_role is None.""" from c2cgeoportal_commons.models.static import User dbsession.add(User("test", email="test@valid.net")) diff --git a/admin/tests/themes_ordering.py b/admin/tests/themes_ordering.py index edba54a859..afba47dcdc 100644 --- a/admin/tests/themes_ordering.py +++ b/admin/tests/themes_ordering.py @@ -41,7 +41,7 @@ def test_edit(self, test_app, themes_ordering_test_data): ) resp = form.submit("submit", status=302) - assert "http://localhost/admin/layertree" == resp.location + assert resp.location == "http://localhost/admin/layertree" for i, theme in enumerate(sorted(themes_ordering_test_data["themes"], key=lambda t: t.name)): assert i == theme.ordering diff --git a/bin/azure b/bin/azure index dd09b06725..24c4cea98f 100755 --- a/bin/azure +++ b/bin/azure @@ -25,7 +25,7 @@ import argparse import glob import os -import subprocess +import subprocess # nosec import yaml from azure.storage.blob import BlobServiceClient, ContainerClient, __version__ diff --git a/bin/build-l10n b/bin/build-l10n index ce85daa550..ea64b9a151 100755 --- a/bin/build-l10n +++ b/bin/build-l10n @@ -4,7 +4,7 @@ import argparse import glob import os import shutil -import subprocess +import subprocess # nosec def main() -> None: diff --git a/ci/changelog b/ci/changelog index 661004c953..665b61d1cc 100755 --- a/ci/changelog +++ b/ci/changelog @@ -31,7 +31,7 @@ import json import os import re -import subprocess +import subprocess # nosec import sys from typing import Any diff --git a/ci/requirements-project.txt b/ci/requirements-project.txt index 5f07c9c4a5..a8c6fdf453 100644 --- a/ci/requirements-project.txt +++ b/ci/requirements-project.txt @@ -1,4 +1,4 @@ # When we upgrade this we should also upgrade the requirements # in the documentation: doc/integrator/requirements.rst -PyYAML==3.13 +PyYAML==6.0 requests==2.31.0 diff --git a/commons/c2cgeoportal_commons/alembic/env.py b/commons/c2cgeoportal_commons/alembic/env.py index 6c05079fe3..99667969b5 100755 --- a/commons/c2cgeoportal_commons/alembic/env.py +++ b/commons/c2cgeoportal_commons/alembic/env.py @@ -103,7 +103,11 @@ def run_migrations_online() -> None: # Autogenerate config alembic_name = context.config.get_main_option("type") - from c2cgeoportal_commons.models import Base, main, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + Base, + main, + static, + ) _schema = main._schema if alembic_name == "main" else static._schema # pylint: disable=protected-access diff --git a/commons/c2cgeoportal_commons/alembic/main/166ff2dcc48d_create_database.py b/commons/c2cgeoportal_commons/alembic/main/166ff2dcc48d_create_database.py index 3b40b6fddb..d745fb9b7f 100644 --- a/commons/c2cgeoportal_commons/alembic/main/166ff2dcc48d_create_database.py +++ b/commons/c2cgeoportal_commons/alembic/main/166ff2dcc48d_create_database.py @@ -40,7 +40,15 @@ from alembic import op from c2c.template.config import config from sqlalchemy import Column, ForeignKey, MetaData, Table -from sqlalchemy.types import Boolean, DateTime, Float, Integer, String, Unicode, UserDefinedType +from sqlalchemy.types import ( + Boolean, + DateTime, + Float, + Integer, + String, + Unicode, + UserDefinedType, +) # revision identifiers, used by Alembic. revision = "166ff2dcc48d" @@ -93,10 +101,7 @@ def upgrade() -> None: Column("readwrite", Boolean, default=False), schema=schema, ) - op.execute( - "SELECT AddGeometryColumn('%(schema)s', 'restrictionarea', " - "'area', %(srid)s, 'POLYGON', 2)" % {"schema": schema, "srid": srid} - ) + op.execute(f"SELECT AddGeometryColumn('{schema}', 'restrictionarea', " f"'area', {srid}, 'POLYGON', 2)") op.create_table( "shorturl", Column("id", Integer, primary_key=True), @@ -116,10 +121,7 @@ def upgrade() -> None: Column("description", Unicode), schema=schema, ) - op.execute( - "SELECT AddGeometryColumn('%(schema)s', 'role', " - "'extent', %(srid)s, 'POLYGON', 2)" % {"schema": schema, "srid": srid} - ) + op.execute(f"SELECT AddGeometryColumn('{schema}', 'role', " f"'extent', {srid}, 'POLYGON', 2)") role = Table("role", MetaData(), Column("name", Unicode, unique=True, nullable=False), schema=schema) op.bulk_insert(role, [{"name": "role_admin"}]) @@ -172,10 +174,7 @@ def upgrade() -> None: Column("params", Unicode, nullable=True), schema=schema, ) - op.execute( - "SELECT AddGeometryColumn('%(schema)s', 'tsearch', 'the_geom', " - "%(srid)s, 'GEOMETRY', 2)" % {"schema": schema, "srid": srid} - ) + op.execute(f"SELECT AddGeometryColumn('{schema}', 'tsearch', 'the_geom', " f"{srid}, 'GEOMETRY', 2)") op.create_index("tsearch_ts_idx", "tsearch", ["ts"], schema=schema, postgresql_using="gin") op.create_table( "treegroup", diff --git a/commons/c2cgeoportal_commons/alembic/main/21f11066f8ec_trigger_on_role_updates_user_in_static.py b/commons/c2cgeoportal_commons/alembic/main/21f11066f8ec_trigger_on_role_updates_user_in_static.py index 777bdf654f..b2ad419528 100644 --- a/commons/c2cgeoportal_commons/alembic/main/21f11066f8ec_trigger_on_role_updates_user_in_static.py +++ b/commons/c2cgeoportal_commons/alembic/main/21f11066f8ec_trigger_on_role_updates_user_in_static.py @@ -71,7 +71,7 @@ def downgrade() -> None: schema = config["schema"] op.execute( - """ + f""" CREATE OR REPLACE FUNCTION {schema}.on_role_name_change() RETURNS trigger AS $$ @@ -82,7 +82,5 @@ def downgrade() -> None: RETURN NEW; END; $$ -LANGUAGE plpgsql""".format( - schema=schema - ) +LANGUAGE plpgsql""" ) diff --git a/commons/c2cgeoportal_commons/alembic/main/56dc90838d90_fix_removing_layerv1.py b/commons/c2cgeoportal_commons/alembic/main/56dc90838d90_fix_removing_layerv1.py index c6298f2232..cbacc5cece 100644 --- a/commons/c2cgeoportal_commons/alembic/main/56dc90838d90_fix_removing_layerv1.py +++ b/commons/c2cgeoportal_commons/alembic/main/56dc90838d90_fix_removing_layerv1.py @@ -50,11 +50,9 @@ def upgrade() -> None: schema = config["schema"] op.execute( - ( - "DELETE from {schema}.layer_restrictionarea WHERE layer_id IN (" - "SELECT id from {schema}.treeitem WHERE type = 'layerv1'" - ");" - ).format(schema=schema) + f"DELETE from {schema}.layer_restrictionarea WHERE layer_id IN (" + f"SELECT id from {schema}.treeitem WHERE type = 'layerv1'" + ");" ) op.execute(f"DELETE from {schema}.treeitem WHERE type = 'layerv1';") diff --git a/commons/c2cgeoportal_commons/alembic/main/7530011a66a7_trigger_on_role_updates_user_in_static.py b/commons/c2cgeoportal_commons/alembic/main/7530011a66a7_trigger_on_role_updates_user_in_static.py index 4107a44967..0a8278fe29 100644 --- a/commons/c2cgeoportal_commons/alembic/main/7530011a66a7_trigger_on_role_updates_user_in_static.py +++ b/commons/c2cgeoportal_commons/alembic/main/7530011a66a7_trigger_on_role_updates_user_in_static.py @@ -71,7 +71,7 @@ def downgrade() -> None: schema = config["schema"] op.execute( - """ + f""" CREATE OR REPLACE FUNCTION {schema}.on_role_name_change() RETURNS trigger AS $$ @@ -82,7 +82,5 @@ def downgrade() -> None: RETURN NEW; END; $$ -LANGUAGE plpgsql""".format( - schema=schema - ) +LANGUAGE plpgsql""" ) diff --git a/commons/c2cgeoportal_commons/alembic/main/7d271f4527cd_add_layer_column_in_layerv1_table.py b/commons/c2cgeoportal_commons/alembic/main/7d271f4527cd_add_layer_column_in_layerv1_table.py index 1129d0194a..ed914b6974 100644 --- a/commons/c2cgeoportal_commons/alembic/main/7d271f4527cd_add_layer_column_in_layerv1_table.py +++ b/commons/c2cgeoportal_commons/alembic/main/7d271f4527cd_add_layer_column_in_layerv1_table.py @@ -53,10 +53,10 @@ def upgrade() -> None: op.add_column("layerv1", Column("layer", Unicode), schema=schema) op.execute( - "UPDATE {schema}.layerv1 AS l1 " + f"UPDATE {schema}.layerv1 AS l1 " "SET layer = name " - "FROM {schema}.treeitem AS ti " - "WHERE l1.id = ti.id".format(schema=schema) + f"FROM {schema}.treeitem AS ti " + "WHERE l1.id = ti.id" ) diff --git a/commons/c2cgeoportal_commons/alembic/main/9268a1dffac0_add_trigger_to_be_able_to_correctly_.py b/commons/c2cgeoportal_commons/alembic/main/9268a1dffac0_add_trigger_to_be_able_to_correctly_.py index 143943fe90..de7fef0297 100644 --- a/commons/c2cgeoportal_commons/alembic/main/9268a1dffac0_add_trigger_to_be_able_to_correctly_.py +++ b/commons/c2cgeoportal_commons/alembic/main/9268a1dffac0_add_trigger_to_be_able_to_correctly_.py @@ -50,7 +50,7 @@ def upgrade() -> None: schema = config["schema"] op.execute( - """ + f""" CREATE FUNCTION {schema}.on_role_name_change() RETURNS trigger AS $$ @@ -61,14 +61,12 @@ def upgrade() -> None: RETURN NEW; END; $$ -LANGUAGE plpgsql""".format( - schema=schema - ) +LANGUAGE plpgsql""" ) op.execute( - "CREATE TRIGGER on_role_name_change AFTER UPDATE ON {schema}.role FOR EACH ROW " - "EXECUTE PROCEDURE {schema}.on_role_name_change()".format(schema=schema) + f"CREATE TRIGGER on_role_name_change AFTER UPDATE ON {schema}.role FOR EACH ROW " + f"EXECUTE PROCEDURE {schema}.on_role_name_change()" ) diff --git a/commons/c2cgeoportal_commons/alembic/main/eeb345672454_merge_2_4_and_master_branches.py b/commons/c2cgeoportal_commons/alembic/main/eeb345672454_merge_2_4_and_master_branches.py index 50ba1783b8..190e3b3380 100644 --- a/commons/c2cgeoportal_commons/alembic/main/eeb345672454_merge_2_4_and_master_branches.py +++ b/commons/c2cgeoportal_commons/alembic/main/eeb345672454_merge_2_4_and_master_branches.py @@ -35,7 +35,6 @@ Create Date: 2019-09-03 09:11:57.786920 """ - # revision identifiers, used by Alembic. revision = "eeb345672454" down_revision = ("78fd093c8393", "56dc90838d90") diff --git a/commons/c2cgeoportal_commons/alembic/static/1da396a88908_move_user_table_to_static_schema.py b/commons/c2cgeoportal_commons/alembic/static/1da396a88908_move_user_table_to_static_schema.py index 68f7f3294a..b21d92d217 100644 --- a/commons/c2cgeoportal_commons/alembic/static/1da396a88908_move_user_table_to_static_schema.py +++ b/commons/c2cgeoportal_commons/alembic/static/1da396a88908_move_user_table_to_static_schema.py @@ -82,20 +82,13 @@ def upgrade() -> None: try: op.execute( - "INSERT INTO %(staticschema)s.user " - "(type, username, password, email, is_password_changed, role_name%(parent_column)s) (" + f"INSERT INTO {staticschema}.user " + f"(type, username, password, email, is_password_changed, role_name{parent_column}) (" "SELECT u.type, u.username, u.password, u.email, " - "u.is_password_changed, r.name%(parent_select)s " - "FROM %(schema)s.user AS u " - "LEFT OUTER JOIN %(schema)s.role AS r ON (r.id = u.role_id) %(parent_join)s" + f"u.is_password_changed, r.name{parent_select} " + f"FROM {schema}.user AS u " + f"LEFT OUTER JOIN {schema}.role AS r ON (r.id = u.role_id) {parent_join}" ")" - % { - "staticschema": staticschema, - "schema": schema, - "parent_select": parent_select, - "parent_column": parent_column, - "parent_join": parent_join, - } ) op.drop_table("user", schema=schema) except Exception: # pylint: disable=broad-exception-caught @@ -135,20 +128,13 @@ def downgrade() -> None: parent_join = f"LEFT OUTER JOIN {parentschema}.role AS pr ON (pr.name = u.parent_role_name)" op.execute( - "INSERT INTO %(schema)s.user " - "(type, username, password, email, is_password_changed, role_id%(parent_column)s) (" + f"INSERT INTO {schema}.user " + f"(type, username, password, email, is_password_changed, role_id{parent_column}) (" "SELECT u.type, u.username, u.password, u.email, " - "u.is_password_changed, r.id%(parent_select)s " - "FROM %(staticschema)s.user AS u " - "LEFT OUTER JOIN %(schema)s.role AS r ON (r.name = u.role_name) %(parent_join)s" + f"u.is_password_changed, r.id{parent_select} " + f"FROM {staticschema}.user AS u " + f"LEFT OUTER JOIN {schema}.role AS r ON (r.name = u.role_name) {parent_join}" ")" - % { - "staticschema": staticschema, - "schema": schema, - "parent_select": parent_select, - "parent_column": parent_column, - "parent_join": parent_join, - } ) op.drop_table("user", schema=staticschema) diff --git a/commons/c2cgeoportal_commons/alembic/static/53d671b17b20_add_timezone_on_datetime_fields.py b/commons/c2cgeoportal_commons/alembic/static/53d671b17b20_add_timezone_on_datetime_fields.py index 99d9d4ee71..1d5917363c 100644 --- a/commons/c2cgeoportal_commons/alembic/static/53d671b17b20_add_timezone_on_datetime_fields.py +++ b/commons/c2cgeoportal_commons/alembic/static/53d671b17b20_add_timezone_on_datetime_fields.py @@ -50,14 +50,12 @@ def upgrade() -> None: staticschema = config["schema_static"] op.execute( - """ + f""" SET TIME ZONE 'UTC'; ALTER TABLE {staticschema}.user ALTER COLUMN last_login TYPE timestamp with time zone; SET TIME ZONE LOCAL; ALTER TABLE {staticschema}.user ALTER COLUMN expire_on TYPE timestamp with time zone; -""".format( - staticschema=staticschema - ) +""" ) @@ -66,12 +64,10 @@ def downgrade() -> None: staticschema = config["schema_static"] op.execute( - """ + f""" SET TIME ZONE 'UTC'; ALTER TABLE {staticschema}.user ALTER COLUMN last_login TYPE timestamp without time zone; SET TIME ZONE LOCAL; ALTER TABLE {staticschema}.user ALTER COLUMN expire_on TYPE timestamp without time zone; -""".format( - staticschema=staticschema - ) +""" ) diff --git a/commons/c2cgeoportal_commons/alembic/static/aa41e9613256_wip_add_openid_connect_support.py b/commons/c2cgeoportal_commons/alembic/static/aa41e9613256_wip_add_openid_connect_support.py index 59b750d800..5f3625a956 100644 --- a/commons/c2cgeoportal_commons/alembic/static/aa41e9613256_wip_add_openid_connect_support.py +++ b/commons/c2cgeoportal_commons/alembic/static/aa41e9613256_wip_add_openid_connect_support.py @@ -35,7 +35,6 @@ Create Date: 2024-08-30 15:56:31.163378 """ - from alembic import op from c2c.template.config import config diff --git a/commons/c2cgeoportal_commons/models/__init__.py b/commons/c2cgeoportal_commons/models/__init__.py index d12489f7f4..275aaefe16 100644 --- a/commons/c2cgeoportal_commons/models/__init__.py +++ b/commons/c2cgeoportal_commons/models/__init__.py @@ -47,7 +47,7 @@ class BaseType(sqlalchemy.ext.declarative.DeclarativeMeta, type): - pass + """Base type for all the models class.""" Base: BaseType = sqlalchemy.orm.declarative_base() diff --git a/commons/c2cgeoportal_commons/models/main.py b/commons/c2cgeoportal_commons/models/main.py index ffc87dd9f4..0e008f48cb 100644 --- a/commons/c2cgeoportal_commons/models/main.py +++ b/commons/c2cgeoportal_commons/models/main.py @@ -56,7 +56,13 @@ from c2cgeoform.ext.colander_ext import Geometry as ColanderGeometry from c2cgeoform.ext.deform_ext import MapWidget, RelationSelect2Widget from colander import drop - from deform.widget import CheckboxWidget, HiddenWidget, SelectWidget, TextAreaWidget, TextInputWidget + from deform.widget import ( + CheckboxWidget, + HiddenWidget, + SelectWidget, + TextAreaWidget, + TextInputWidget, + ) colander_null = colander.null except ModuleNotFoundError: @@ -84,7 +90,6 @@ def __init__(self, *args: Any, **kwargs: Any): def state_str(state: Any) -> str: """Return a string describing an instance via its InstanceState.""" - return "None" if state is None else f"<{state.class_.__name__} {state.obj()}>" # In the original function sqlalchemy use the id of the object that don't allow us to give some useful @@ -308,7 +313,7 @@ def __str__(self) -> str: return f"{self.name}[{self.id}]>" @property - def bounds(self) -> tuple[float, float, float, float] | None: # TODO + def bounds(self) -> tuple[float, float, float, float] | None: if self.extent is None: return None return cast(tuple[float, float, float, float], to_shape(self.extent).bounds) @@ -367,11 +372,7 @@ def is_in_interface(self, name: str) -> bool: if not hasattr(self, "interfaces"): return False - for interface in self.interfaces: - if interface.name == name: - return True - - return False + return any(interface.name == name for interface in self.interfaces) def get_metadata(self, name: str) -> list["Metadata"]: return [metadata for metadata in self.metadatas if metadata.name == name] @@ -1035,7 +1036,7 @@ def __init__( @staticmethod def get_default(dbsession: Session) -> DimensionLayer | None: return cast( - Optional[DimensionLayer], + DimensionLayer | None, dbsession.query(LayerWMS).filter(LayerWMS.name == "wms-defaults").one_or_none(), ) @@ -1179,7 +1180,7 @@ def __init__(self, name: str = "", public: bool = True, image_type: ImageType = @staticmethod def get_default(dbsession: Session) -> DimensionLayer | None: return cast( - Optional[DimensionLayer], + DimensionLayer | None, dbsession.query(LayerWMTS).filter(LayerWMTS.name == "wmts-defaults").one_or_none(), ) @@ -1374,7 +1375,7 @@ def __init__(self, name: str = "", public: bool = True, style: str = "", sql: st @staticmethod def get_default(dbsession: Session) -> DimensionLayer | None: return cast( - Optional[DimensionLayer], + DimensionLayer | None, dbsession.query(LayerVectorTiles) .filter(LayerVectorTiles.name == "vector-tiles-defaults") .one_or_none(), diff --git a/commons/c2cgeoportal_commons/models/static.py b/commons/c2cgeoportal_commons/models/static.py index 0e60d1f80d..3327f93962 100644 --- a/commons/c2cgeoportal_commons/models/static.py +++ b/commons/c2cgeoportal_commons/models/static.py @@ -324,7 +324,7 @@ def set_temp_password(self, password: str) -> None: @staticmethod def __encrypt_password_legacy(password: str) -> str: """Hash the given password with SHA1.""" - return sha1(password.encode("utf8")).hexdigest() # nosec + return sha1(password.encode("utf8")).hexdigest() # noqa: S324 @staticmethod def __encrypt_password(password: str) -> str: diff --git a/commons/c2cgeoportal_commons/testing/__init__.py b/commons/c2cgeoportal_commons/testing/__init__.py index 14a6ff17af..debb5ae292 100644 --- a/commons/c2cgeoportal_commons/testing/__init__.py +++ b/commons/c2cgeoportal_commons/testing/__init__.py @@ -78,11 +78,10 @@ def get_tm_session( def generate_mappers() -> None: """Initialize the model for a Pyramid app.""" - # import or define all models here to ensure they are attached to the # Base.metadata prior to any initialization routines - import c2cgeoportal_commons.models.main # pylint: disable=unused-import,import-outside-toplevel - import c2cgeoportal_commons.models.static # pylint: disable=import-outside-toplevel + import c2cgeoportal_commons.models.main # pylint: disable=unused-import,import-outside-toplevel # noqa: F401 + import c2cgeoportal_commons.models.static # pylint: disable=import-outside-toplevel # noqa: F401 # run configure_mappers after defining all of the models to ensure # all relationships can be setup diff --git a/commons/c2cgeoportal_commons/testing/initializedb.py b/commons/c2cgeoportal_commons/testing/initializedb.py index e3951d17c6..e9167ce9dc 100644 --- a/commons/c2cgeoportal_commons/testing/initializedb.py +++ b/commons/c2cgeoportal_commons/testing/initializedb.py @@ -46,11 +46,15 @@ def usage(argv: list[str]) -> None: def schema_exists(connection: Connection, schema_name: str) -> bool: """Get if the schema exist.""" - sql = f""" -SELECT count(*) AS count -FROM information_schema.schemata -WHERE schema_name = '{schema_name}'; -""" + del schema_name + + sql = " ".join( + [ # noqa: S608 + "SELECT count(*) AS count", + "FROM information_schema.schemata", + "WHERE schema_name = '{schema_name}';", + ] + ) result = connection.execute(sqlalchemy.text(sql)) row = result.first() return cast(bool, row[0] == 1) # type: ignore[index] @@ -64,8 +68,12 @@ def truncate_tables(connection: Connection) -> None: def setup_test_data(dbsession: Session) -> None: """Initialize the testing data.""" - from c2cgeoportal_commons.models.main import Role # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Role, + ) + from c2cgeoportal_commons.models.static import ( # pylint: disable=import-outside-toplevel + User, + ) role_admin = dbsession.merge(Role(name="role_admin")) role_user = dbsession.merge(Role(name="role_user")) diff --git a/commons/tests/conftest.py b/commons/tests/conftest.py index dd7d28ccf5..1ce9c4c3ac 100644 --- a/commons/tests/conftest.py +++ b/commons/tests/conftest.py @@ -5,10 +5,14 @@ import pytest import transaction from c2c.template.config import config -from sqlalchemy.exc import DBAPIError - -from c2cgeoportal_commons.testing import generate_mappers, get_engine, get_session_factory, get_tm_session +from c2cgeoportal_commons.testing import ( + generate_mappers, + get_engine, + get_session_factory, + get_tm_session, +) from c2cgeoportal_commons.testing.initializedb import truncate_tables +from sqlalchemy.exc import DBAPIError @pytest.fixture(scope="session") diff --git a/commons/tests/lib/test_url.py b/commons/tests/lib/test_url.py index 7477789912..506e8e3bf4 100644 --- a/commons/tests/lib/test_url.py +++ b/commons/tests/lib/test_url.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -29,9 +29,8 @@ from unittest import TestCase -from pyramid.testing import DummyRequest - from c2cgeoportal_commons.lib.url import Url, get_url2 +from pyramid.testing import DummyRequest class TestUrl(TestCase): diff --git a/commons/tests/lib/test_validators.py b/commons/tests/lib/test_validators.py index 3f28cf13f1..a176740e21 100644 --- a/commons/tests/lib/test_validators.py +++ b/commons/tests/lib/test_validators.py @@ -1,7 +1,6 @@ # pylint: disable=no-self-use import pytest - from c2cgeoportal_commons.lib.validators import url diff --git a/commons/tests/models/test_interface.py b/commons/tests/models/test_interface.py index b64d7b1b0c..ad7afbcd22 100644 --- a/commons/tests/models/test_interface.py +++ b/commons/tests/models/test_interface.py @@ -6,9 +6,8 @@ @pytest.mark.usefixtures("transact") class TestInterface: def test_delete_cascade_to_tsearch(self, dbsession): - from sqlalchemy import func - from c2cgeoportal_commons.models.main import FullTextSearch, Interface + from sqlalchemy import func interface = Interface("desktop", "Desktop interface") interface_id = interface.id @@ -24,5 +23,5 @@ def test_delete_cascade_to_tsearch(self, dbsession): dbsession.flush() assert ( - 0 == dbsession.query(FullTextSearch).filter(FullTextSearch.interface_id == interface_id).count() + dbsession.query(FullTextSearch).filter(FullTextSearch.interface_id == interface_id).count() == 0 ) diff --git a/commons/tests/models/test_roles.py b/commons/tests/models/test_roles.py index b4a227807b..ac68c3537a 100644 --- a/commons/tests/models/test_roles.py +++ b/commons/tests/models/test_roles.py @@ -41,12 +41,11 @@ def test_delete(self, dbsession): dbsession.delete(roles[0]) roles = dbsession.query(Role).all() assert len(roles) == 0, "removed a role" - assert 0 == dbsession.query(user_role).count() + assert dbsession.query(user_role).count() == 0 def test_delete_cascade_to_tsearch(self, dbsession): - from sqlalchemy import func - from c2cgeoportal_commons.models.main import FullTextSearch, Role + from sqlalchemy import func role = dbsession.query(Role).filter(Role.name == "secretary").one() role_id = role.id @@ -61,4 +60,4 @@ def test_delete_cascade_to_tsearch(self, dbsession): dbsession.delete(role) dbsession.flush() - assert 0 == dbsession.query(FullTextSearch).filter(FullTextSearch.role_id == role_id).count() + assert dbsession.query(FullTextSearch).filter(FullTextSearch.role_id == role_id).count() == 0 diff --git a/commons/tests/models/test_users.py b/commons/tests/models/test_users.py index 43086621fb..2908ab8d00 100644 --- a/commons/tests/models/test_users.py +++ b/commons/tests/models/test_users.py @@ -30,7 +30,7 @@ def test_select(self, dbsession): users = dbsession.query(User).all() assert len(users) == 1, "querying for users" assert users[0].username == "babar", "user from test data is babar" - assert 2 == len(users[0].roles) + assert len(users[0].roles) == 2 def test_remove(self, dbsession): from c2cgeoportal_commons.models.static import User, user_role @@ -39,7 +39,7 @@ def test_remove(self, dbsession): dbsession.delete(users[0]) users = dbsession.query(User).all() assert len(users) == 0, "removed a user" - assert 0 == dbsession.query(user_role).count() + assert dbsession.query(user_role).count() == 0 def test_add(self, dbsession): from c2cgeoportal_commons.models.main import Role @@ -52,9 +52,9 @@ def test_add(self, dbsession): assert dbsession.query(User).count() == 2, "added a user" dbsession.expire(user) assert user.username == "momo", "added user is momo" - assert 1 == len(user.roles) + assert len(user.roles) == 1 assert user.roles[0].name == "Role3" - assert 1 == dbsession.query(user_role).filter(user_role.c.user_id == user.id).count() + assert dbsession.query(user_role).filter(user_role.c.user_id == user.id).count() == 1 @staticmethod def test_edit(dbsession): @@ -62,10 +62,10 @@ def test_edit(dbsession): from c2cgeoportal_commons.models.static import User, user_role user = dbsession.query(User).first() - assert 2 == len(user.roles) + assert len(user.roles) == 2 user.roles = [Role(name="Role4")] dbsession.flush() dbsession.expire(user) - assert 1 == len(user.roles) + assert len(user.roles) == 1 assert user.roles[0].name == "Role4" - assert 1 == dbsession.query(user_role).filter(user_role.c.user_id == user.id).count() + assert dbsession.query(user_role).filter(user_role.c.user_id == user.id).count() == 1 diff --git a/doc/import_ngeo_config.py b/doc/import_ngeo_config.py index ffb417e34b..83c47c20e9 100755 --- a/doc/import_ngeo_config.py +++ b/doc/import_ngeo_config.py @@ -30,7 +30,7 @@ def _format_type(type_definition: dict[str, Any]) -> str: f"{_format_type(type_definition['declaration']['indexSignature']['type'])}}}" ) - assert False, f"Unknown type '{type_definition['type']}':\n{type_definition}" + raise AssertionError(f"Unknown type '{type_definition['type']}':\n{type_definition}") def _get_type(type_definition: dict[str, Any]) -> list[str]: diff --git a/doc/integrator/requirements.rst b/doc/integrator/requirements.rst index 7181204e53..3f3cad91c2 100644 --- a/doc/integrator/requirements.rst +++ b/doc/integrator/requirements.rst @@ -11,7 +11,7 @@ components installed on your system: * **Git** * **Docker** >= 17.05 -* **Python** >= 3.8, with ``pip`` +* **Python** >= 3.10, with ``pip`` * **Apache** >= 2.4 (optional, can be used as a front for SSL) * **PostgreSQL** >= 9.1/**PostGIS** >= 2.1 diff --git a/doc/poetry.lock b/doc/poetry.lock index ba216a820d..1c14eda591 100644 --- a/doc/poetry.lock +++ b/doc/poetry.lock @@ -2257,13 +2257,13 @@ wsgi = ["pyramid"] [[package]] name = "tilecloud-chain" -version = "1.21.0" +version = "1.22.0" description = "Tools to generate tiles from WMS or Mapnik, to S3, Berkeley DB, MBTiles, or local filesystem in WMTS layout using Amazon cloud services." optional = false -python-versions = ">=3.9" +python-versions = ">=3.10" files = [ - {file = "tilecloud_chain-1.21.0-py3-none-any.whl", hash = "sha256:a73914cbc8742a1e6cc5b1a9804a78e1af2fcdc62993950f42e2949737faa739"}, - {file = "tilecloud_chain-1.21.0.tar.gz", hash = "sha256:7cfb46c09d3ff604a46f3d8dfc7deafac8f0d771c68e55726cdd1d88a206cec8"}, + {file = "tilecloud_chain-1.22.0-py3-none-any.whl", hash = "sha256:32dabc07c98287220d914c59baf4d6c7802a01bb855e363744a564a2f0e1704d"}, + {file = "tilecloud_chain-1.22.0.tar.gz", hash = "sha256:c61cd96c2fc6b568e173399dd8171fa43f4e29dd4c62b584f1892ce66890c21c"}, ] [package.dependencies] @@ -2610,4 +2610,4 @@ test = ["zope.testing"] [metadata] lock-version = "2.0" python-versions = ">=3.10,<3.13" -content-hash = "28c7611638e048ab7a91df45fc0ab246375795ffc471f0b0cccc961a82e84743" +content-hash = "215b13d0de25b366213f6b79ea957b73aa24917bdd9ececa56512e08a49a66bc" diff --git a/doc/pyproject.toml b/doc/pyproject.toml index e44cd81bf6..d894991c3d 100644 --- a/doc/pyproject.toml +++ b/doc/pyproject.toml @@ -1,11 +1,3 @@ -[tool.black] -line-length = 110 -target-version = ['py39'] - -[tool.isort] -profile = "black" -line_length = 110 - [tool.poetry] name = 'c2cgeoportal' version = '0.0.0' @@ -23,5 +15,5 @@ sphinx-argparse = "0.5.2" Sphinx-Substitution-Extensions = { extras = ["prompt"], version = "2024.10.17" } sphinxcontrib-mermaid = "1.0.0" c2cwsgiutils = ">=6.0.0.dev148" -tilecloud-chain = "1.21.0" +tilecloud-chain = "1.22.0" boto3 = { version = "1.35.72", optional = true } diff --git a/docker-compose.yaml b/docker-compose.yaml index f43052214c..913a6c4f4c 100644 --- a/docker-compose.yaml +++ b/docker-compose.yaml @@ -48,7 +48,7 @@ services: lra.restrictionarea_id = ra.id AND lra.layer_id = la.id AND la.name = ' tilecloudchain: - image: camptocamp/tilecloud-chain:1.21 + image: camptocamp/tilecloud-chain:1.22 user: www-data restart: unless-stopped environment: diff --git a/docker/config/bin/eval-templates b/docker/config/bin/eval-templates index 3680ed0321..05d9bfa97c 100755 --- a/docker/config/bin/eval-templates +++ b/docker/config/bin/eval-templates @@ -5,7 +5,7 @@ import glob import os import random import re -import subprocess +import subprocess # nosec import urllib.parse import zipfile from typing import cast @@ -115,14 +115,14 @@ def _main() -> None: if os.environ.get("VISIBLE_ENTRY_POINT"): os.environ["VISIBLE_ENTRY_POINT_RE_ESCAPED"] = re.escape(os.environ["VISIBLE_ENTRY_POINT"]) os.environ["MAPSERVER_DATA_SUBSELECT"] = ( - "SELECT {ST_JOIN}(ra.area) " + "SELECT {ST_JOIN}(ra.area) " # nosec "FROM {PGSCHEMA}.restrictionarea AS ra, {PGSCHEMA}.role_restrictionarea AS rra, " "{PGSCHEMA}.layer_restrictionarea AS lra, {PGSCHEMA}.treeitem AS la " "WHERE rra.role_id in (%role_ids%) AND rra.restrictionarea_id = ra.id " "AND lra.restrictionarea_id = ra.id AND lra.layer_id = la.id AND la.name = " ).format(PGSCHEMA=os.environ["PGSCHEMA"], ST_JOIN=os.environ.get("ST_JOIN", "ST_Collect")) os.environ["MAPSERVER_DATA_NOAREA_SUBSELECT"] = ( - "SELECT rra.role_id " + "SELECT rra.role_id " # nosec "FROM {PGSCHEMA}.restrictionarea AS ra, {PGSCHEMA}.role_restrictionarea AS rra, " "{PGSCHEMA}.layer_restrictionarea AS lra, {PGSCHEMA}.treeitem AS la " "WHERE rra.restrictionarea_id = ra.id AND lra.restrictionarea_id = ra.id " diff --git a/docker/qgisserver/.bandit.yaml b/docker/qgisserver/.bandit.yaml deleted file mode 100644 index 1bf9b483e1..0000000000 --- a/docker/qgisserver/.bandit.yaml +++ /dev/null @@ -1,2 +0,0 @@ -skips: - - B101 # Use of assert detected. diff --git a/docker/qgisserver/.prospector.yaml b/docker/qgisserver/.prospector.yaml index 7c072ce7a2..98eacd5fdb 100644 --- a/docker/qgisserver/.prospector.yaml +++ b/docker/qgisserver/.prospector.yaml @@ -3,10 +3,11 @@ inherits: - utils:base - utils:fix - utils:no-design-checks + - utils:unsafe pyroma: run: False -bandit: +mypy: options: - config: .bandit.yaml + python_version: '3.10' diff --git a/docker/qgisserver/geomapfish_qgisserver/__init__.py b/docker/qgisserver/geomapfish_qgisserver/__init__.py index f0e30ef830..d8da365e3b 100644 --- a/docker/qgisserver/geomapfish_qgisserver/__init__.py +++ b/docker/qgisserver/geomapfish_qgisserver/__init__.py @@ -20,6 +20,7 @@ def serverClassFactory( # pylint: disable=invalid-name serverIface: qgis.server.QgsServerInterface, # pylint: disable=invalid-name ) -> qgis.server.QgsAccessControlFilter | None: + """Create a new instance of the access control filter.""" QgsMessageLog.logMessage("Configure logging...", "GeoMapFish-init", level=Qgis.Info) try: @@ -34,7 +35,9 @@ def serverClassFactory( # pylint: disable=invalid-name _LOG.info("Starting GeoMapFish access restriction...") try: - from .accesscontrol import GeoMapFishAccessControl # pylint: disable=import-outside-toplevel + from .accesscontrol import ( # pylint: disable=import-outside-toplevel + GeoMapFishAccessControl, + ) return GeoMapFishAccessControl(serverIface) except Exception: # pylint: disable=broad-exception-caught diff --git a/docker/qgisserver/geomapfish_qgisserver/accesscontrol.py b/docker/qgisserver/geomapfish_qgisserver/accesscontrol.py index 7e421bceef..eb9793b064 100644 --- a/docker/qgisserver/geomapfish_qgisserver/accesscontrol.py +++ b/docker/qgisserver/geomapfish_qgisserver/accesscontrol.py @@ -20,6 +20,7 @@ import yaml import zope.event.classhandler from c2c.template.config import config +from c2cgeoportal_commons.lib.url import Url, get_url2 from qgis.core import ( QgsDataSourceUri, QgsFeature, @@ -35,15 +36,16 @@ from sqlalchemy.orm import configure_mappers, joinedload, sessionmaker, subqueryload from sqlalchemy.orm.session import Session -from c2cgeoportal_commons.lib.url import Url, get_url2 - if TYPE_CHECKING: - from c2cgeoportal_commons.models import main # pylint: disable=ungrouped-imports,useless-suppression + from c2cgeoportal_commons.models import ( + main, # pylint: disable=ungrouped-imports,useless-suppression + ) _LOG = logging.getLogger(__name__) def create_session_factory(url: str, configuration: dict[str, Any]) -> sessionmaker: + """Create a SQLAlchemy session factory.""" configure_mappers() db_match = re.match(".*(@[^@]+)$", url) _LOG.info( @@ -58,17 +60,22 @@ def create_session_factory(url: str, configuration: dict[str, Any]) -> sessionma class GMFException(Exception): - """Standard exception""" + """Standard exception.""" class Access(Enum): + """Access mode.""" + NO = 1 AREA = 2 FULL = 3 class GeoMapFishAccessControl(QgsAccessControlFilter): + """Implements GeoMapFish access restriction.""" + def __init__(self, server_iface: qgis.server.QgsServerInterface): + """Initialize the plugin.""" super().__init__(server_iface) self.server_iface = server_iface @@ -225,12 +232,14 @@ def __init__(self, server_iface: qgis.server.QgsServerInterface): server_iface.registerAccessControl(self, int(os.environ.get("GEOMAPFISH_POSITION", 100))) def get_ogcserver_accesscontrol_config(self) -> None: + """Get the config.""" if self.single: raise GMFException( "The method 'get_ogcserver_accesscontrol_config' can't be called on 'single' server" ) def get_ogcserver_accesscontrol(self) -> "OGCServerAccessControl": + """Get the OGCServerAccessControl instance.""" parameters = self.serverInterface().requestHandler().parameterMap() if self.single: @@ -309,10 +318,11 @@ def allowToEdit(self, layer: QgsVectorLayer, feature: QgsFeature) -> bool: # py raise def cacheKey(self) -> str: # pylint: disable=invalid-name + """Get the cache key.""" try: if not self.initialized: _LOG.error("Call on uninitialized plugin") - return str(random.randrange(1000000)) # nosec + return str(random.randrange(1000000)) # noqa: S311 return self.get_ogcserver_accesscontrol().cacheKey() except Exception: _LOG.exception("Unhandled error") @@ -320,7 +330,7 @@ def cacheKey(self) -> str: # pylint: disable=invalid-name class OGCServerAccessControl(QgsAccessControlFilter): - """Implements GeoMapFish access restriction.""" + """Implements GeoMapFish access restriction for one project.""" SUBSETSTRING_TYPE = ["PostgreSQL database with PostGIS extension"] @@ -333,6 +343,7 @@ def __init__( DBSession: sessionmaker, # pylint: disable=invalid-name ogcserver: Optional["main.OGCServer"] = None, ): + """Initialize the plugin.""" super().__init__(server_iface) self.server_iface = server_iface @@ -341,7 +352,7 @@ def __init__( self.DBSession = DBSession # pylint: disable=invalid-name self.area_cache: dict[Any, tuple[Access, BaseGeometry]] = {} - self.layers: dict[str, list["main.LayerWMS"]] | None = None + self.layers: dict[str, list[main.LayerWMS]] | None = None self.lock = Lock() self.srid = srid self.ogcserver = ogcserver @@ -358,6 +369,7 @@ def handle(_: InvalidateCacheEvent) -> None: self._init(ogcserver_name) def project(self) -> QgsProject: + """Get the project.""" return QgsConfigCache.instance().project(self.map_file) def _init(self, ogcserver_name: str) -> None: @@ -387,6 +399,7 @@ def _init(self, ogcserver_name: str) -> None: _LOG.exception("Cannot setup OGCServerAccessControl") def ogc_layer_name(self, layer: QgsVectorLayer) -> str: + """Get the OGC layer name.""" use_layer_id, _ = self.project().readBoolEntry("WMSUseLayerIDs", "/", False) if use_layer_id: return layer.id() @@ -394,9 +407,9 @@ def ogc_layer_name(self, layer: QgsVectorLayer) -> str: def get_layers(self, session: Session) -> dict[str, list["main.Layer"]]: """ - Get the list of GMF WMS layers that can give access to each QGIS layer or group. That is, for each - QGIS layer tree node, the list of GMF WMS layers that: + Get the list of GMF WMS layers that can give access to each QGIS layer or group. + That is, for each QGIS layer tree node, the list of GMF WMS layers that: - correspond to this ogc_server - contains QGIS node name in the layer_wms.layer field. Returns a dict with: @@ -435,7 +448,7 @@ def browse(path: list[str], node: QgsLayerTreeNode) -> None: browse([], self.project().layerTreeRoot()) - for ogc_layer_name, ancestors in nodes.items(): + for ogc_layer_name, _ in nodes.items(): _LOG.debug("QGIS layer: %s", ogc_layer_name) # Transform ancestor names in LayerWMS instances @@ -473,10 +486,14 @@ def browse(path: list[str], node: QgsLayerTreeNode) -> None: def get_roles(self, session: Session) -> str | list["main.Role"]: """ Get the current user's available roles based on request parameter USER_ID. + Returns: - List of c2cgeoportal_commons.models.main.Role instances. + """ - from c2cgeoportal_commons.models.main import Role # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Role, + ) parameters = self.serverInterface().requestHandler().parameterMap() @@ -514,8 +531,8 @@ def get_restriction_areas( Returns: - Access mode (NO | AREA | FULL) - List of access areas as shapely geometric objects - """ + """ # Root... if roles == "ROOT": return Access.FULL, None @@ -528,7 +545,9 @@ def get_restriction_areas( if not roles: return Access.NO, None - from c2cgeoportal_commons.models.main import Role # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Role, + ) restriction_areas = set() for layer in gmf_layers: @@ -554,9 +573,11 @@ def get_restriction_areas( def get_area(self, layer: str, session: Session, read_write: bool = False) -> tuple[Access, BaseGeometry]: """ Calculate access area for a QgsMapLayer and an access mode. + Returns: - Access mode (NO | AREA | FULL) - Access area as WKT or None + """ roles = self.get_roles(session) if roles == "ROOT": @@ -588,10 +609,7 @@ def get_area(self, layer: str, session: Session, read_write: bool = False) -> tu return (Access.AREA, area) def layerFilterSubsetString(self, layer: QgsVectorLayer) -> str | None: # pylint: disable=invalid-name - """ - Returns an additional subset string (typically SQL) filter. - """ - + """Get an additional subset string (typically SQL) filter.""" _LOG.debug("layerFilterSubsetString %s %s", layer.name(), layer.dataProvider().storageType()) if self.ogcserver is None: @@ -633,10 +651,7 @@ def layerFilterSubsetString(self, layer: QgsVectorLayer) -> str | None: # pylin raise def layerFilterExpression(self, layer: QgsVectorLayer) -> str | None: # pylint: disable=invalid-name - """ - Returns an additional expression filter. - """ - + """Get an additional expression filter.""" _LOG.debug("layerFilterExpression %s %s", layer.name(), layer.dataProvider().storageType()) if self.ogcserver is None: @@ -678,10 +693,7 @@ def layerFilterExpression(self, layer: QgsVectorLayer) -> str | None: # pylint: def layerPermissions( # pylint: disable=invalid-name self, layer: QgsVectorLayer ) -> qgis.server.QgsAccessControlFilter.LayerPermissions: - """ - Returns the layer rights. - """ - + """Get the layer rights.""" _LOG.debug("layerPermissions %s", layer.name()) try: @@ -725,10 +737,7 @@ def layerPermissions( # pylint: disable=invalid-name def authorizedLayerAttributes( # pylint: disable=invalid-name self, layer: QgsVectorLayer, attributes: list[str] ) -> list[str]: - """ - Returns the authorized layer attributes. - """ - + """Get the authorized layer attributes.""" roles = self.get_roles(self.DBSession()) if roles == "ROOT": return attributes @@ -803,6 +812,7 @@ def allowToEdit(self, layer: QgsVectorLayer, feature: QgsFeature) -> bool: # py raise def cacheKey(self) -> str: # pylint: disable=invalid-name + """Get the cache key.""" # Root... session = self.DBSession() try: diff --git a/docker/qgisserver/geomapfish_qgisserver/gmf_logging.py b/docker/qgisserver/geomapfish_qgisserver/gmf_logging.py index f8f82d5794..ee7a9bef14 100644 --- a/docker/qgisserver/geomapfish_qgisserver/gmf_logging.py +++ b/docker/qgisserver/geomapfish_qgisserver/gmf_logging.py @@ -30,6 +30,7 @@ class LogHandler(logging.Handler): """Python logging handle for QGIS.""" def emit(self, record: logging.LogRecord) -> None: + """Emit the record.""" # To be visible in the CI print(self.format(record)) @@ -37,9 +38,7 @@ def emit(self, record: logging.LogRecord) -> None: class _RequestFilter(logging.Filter): - """ - A logging filter that adds request information to CEE logs. - """ + """A logging filter that adds request information to CEE logs.""" def filter(self, record: Any) -> bool: request_handler = SERVER_IFACE.requestHandler() if SERVER_IFACE else None @@ -66,13 +65,13 @@ def filter(self, record: Any) -> bool: class JsonLogHandler(c2cwsgiutils.pyramid_logging.JsonLogHandler): - """ - Log to stdout in JSON. - """ + """Log to stdout in JSON.""" def __init__(self, stream: TextIO | None) -> None: + """Initialize the handler.""" super().__init__(stream) self.addFilter(_REQUEST_FILTER) def emit(self, record: logging.LogRecord) -> None: + """Emit the record.""" QgsMessageLog.logMessage(self.format(record), record.name, level=Qgis.Critical) diff --git a/docker/qgisserver/poetry.lock b/docker/qgisserver/poetry.lock index c3c6e2afb2..b2da6902eb 100644 --- a/docker/qgisserver/poetry.lock +++ b/docker/qgisserver/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. [[package]] name = "astroid" @@ -367,13 +367,13 @@ tqdm = ["tqdm"] [[package]] name = "geoalchemy2" -version = "0.15.2" +version = "0.16.0" description = "Using SQLAlchemy with Spatial Databases" optional = false python-versions = ">=3.7" files = [ - {file = "GeoAlchemy2-0.15.2-py3-none-any.whl", hash = "sha256:546455dc39f5bcdfc5b871e57d3f7546c8a6f798eb364c474200f488ace6fd32"}, - {file = "geoalchemy2-0.15.2.tar.gz", hash = "sha256:3af0272db927373e74ee3b064cdc9464ba08defdb945c51745db1b841482f5dc"}, + {file = "GeoAlchemy2-0.16.0-py3-none-any.whl", hash = "sha256:b0f27d5500ee757af4654c6262e0f834b7a843504d193653ec747ef1128d2ab5"}, + {file = "geoalchemy2-0.16.0.tar.gz", hash = "sha256:df64bb72af70daafaac3f359492c96501c37ab85ed20f9510c99cc6d02881100"}, ] [package.dependencies] @@ -1007,6 +1007,7 @@ pylint-django = ">=2.6.1" pylint-flask = "0.6" PyYAML = "*" requirements-detector = ">=1.3.2" +ruff = {version = "*", optional = true, markers = "extra == \"with-ruff\" or extra == \"with_everything\""} setoptconf-tmp = ">=0.3.1,<0.4.0" toml = ">=0.10.2,<0.11.0" @@ -1021,26 +1022,29 @@ with-vulture = ["vulture (>=1.5)"] [[package]] name = "prospector-profile-duplicated" -version = "1.6.0" +version = "1.9.0" description = "Profile that can be used to disable the duplicated or conflict rules between Prospector and other tools" optional = false python-versions = "*" files = [ - {file = "prospector_profile_duplicated-1.6.0-py2.py3-none-any.whl", hash = "sha256:bf6a6aae0c7de48043b95e4d42e23ccd090c6c7115b6ee8c8ca472ffb1a2022b"}, - {file = "prospector_profile_duplicated-1.6.0.tar.gz", hash = "sha256:9c2d541076537405e8b2484cb6222276a2df17492391b6af1b192695770aab83"}, + {file = "prospector_profile_duplicated-1.9.0-py2.py3-none-any.whl", hash = "sha256:7b7a665e6fa8b44fac597d2bbef1a91de5b6f090b8d64890a5f38ee60e8473c2"}, + {file = "prospector_profile_duplicated-1.9.0.tar.gz", hash = "sha256:bb0b0d0946232d570ada944ee4a180fd142529db0978579b58e766d28f3c2a27"}, ] [[package]] name = "prospector-profile-utils" -version = "1.9.1" +version = "1.14.1" description = "Some utility Prospector profiles." optional = false -python-versions = "*" +python-versions = "<4.0,>=3.9" files = [ - {file = "prospector_profile_utils-1.9.1-py2.py3-none-any.whl", hash = "sha256:b458d8c4d59bdb1547e4630a2c6de4971946c4f0999443db6a9eef6d216b26b8"}, - {file = "prospector_profile_utils-1.9.1.tar.gz", hash = "sha256:008efa6797a85233fd8093dcb9d86f5fa5d89673e431c15cb1496a91c9b2c601"}, + {file = "prospector_profile_utils-1.14.1-py3-none-any.whl", hash = "sha256:1b7d79e4293c76f9ea5107b691c888e76933aa29d33e8f8b8750dc0aeea3b657"}, + {file = "prospector_profile_utils-1.14.1.tar.gz", hash = "sha256:5447086f9a7ddba02d8ee7322baa30c80c1376328677a6593165fab23e2e4bf2"}, ] +[package.dependencies] +prospector = ">=1.13.0" + [[package]] name = "psycopg2" version = "2.9.10" @@ -1413,6 +1417,33 @@ typing-extensions = {version = ">=4.0.0,<5.0", markers = "python_version < \"3.1 [package.extras] jupyter = ["ipywidgets (>=7.5.1,<9)"] +[[package]] +name = "ruff" +version = "0.8.1" +description = "An extremely fast Python linter and code formatter, written in Rust." +optional = false +python-versions = ">=3.7" +files = [ + {file = "ruff-0.8.1-py3-none-linux_armv6l.whl", hash = "sha256:fae0805bd514066f20309f6742f6ee7904a773eb9e6c17c45d6b1600ca65c9b5"}, + {file = "ruff-0.8.1-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:b8a4f7385c2285c30f34b200ca5511fcc865f17578383db154e098150ce0a087"}, + {file = "ruff-0.8.1-py3-none-macosx_11_0_arm64.whl", hash = "sha256:cd054486da0c53e41e0086e1730eb77d1f698154f910e0cd9e0d64274979a209"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2029b8c22da147c50ae577e621a5bfbc5d1fed75d86af53643d7a7aee1d23871"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2666520828dee7dfc7e47ee4ea0d928f40de72056d929a7c5292d95071d881d1"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:333c57013ef8c97a53892aa56042831c372e0bb1785ab7026187b7abd0135ad5"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:288326162804f34088ac007139488dcb43de590a5ccfec3166396530b58fb89d"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b12c39b9448632284561cbf4191aa1b005882acbc81900ffa9f9f471c8ff7e26"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:364e6674450cbac8e998f7b30639040c99d81dfb5bbc6dfad69bc7a8f916b3d1"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b22346f845fec132aa39cd29acb94451d030c10874408dbf776af3aaeb53284c"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:b2f2f7a7e7648a2bfe6ead4e0a16745db956da0e3a231ad443d2a66a105c04fa"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:adf314fc458374c25c5c4a4a9270c3e8a6a807b1bec018cfa2813d6546215540"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_i686.whl", hash = "sha256:a885d68342a231b5ba4d30b8c6e1b1ee3a65cf37e3d29b3c74069cdf1ee1e3c9"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:d2c16e3508c8cc73e96aa5127d0df8913d2290098f776416a4b157657bee44c5"}, + {file = "ruff-0.8.1-py3-none-win32.whl", hash = "sha256:93335cd7c0eaedb44882d75a7acb7df4b77cd7cd0d2255c93b28791716e81790"}, + {file = "ruff-0.8.1-py3-none-win_amd64.whl", hash = "sha256:2954cdbe8dfd8ab359d4a30cd971b589d335a44d444b6ca2cb3d1da21b75e4b6"}, + {file = "ruff-0.8.1-py3-none-win_arm64.whl", hash = "sha256:55873cc1a473e5ac129d15eccb3c008c096b94809d693fc7053f588b67822737"}, + {file = "ruff-0.8.1.tar.gz", hash = "sha256:3583db9a6450364ed5ca3f3b4225958b24f78178908d5c4bc0f46251ccca898f"}, +] + [[package]] name = "semver" version = "3.0.2" @@ -2012,4 +2043,4 @@ test = ["zope.testing"] [metadata] lock-version = "2.0" python-versions = ">=3.10,<3.13" -content-hash = "5e211dc9d324e8a046a70cbc4009f1cfce1d806f52a2ed7a0df3049a951b8a11" +content-hash = "dfddb80abb0d577ba46d1a7f52171f21ffcfcd5f133a3494ec85080d20e2a207" diff --git a/docker/qgisserver/pyproject.toml b/docker/qgisserver/pyproject.toml index 14bbfdfbf0..d10c29cf23 100644 --- a/docker/qgisserver/pyproject.toml +++ b/docker/qgisserver/pyproject.toml @@ -1,20 +1,5 @@ -[tool.mypy] -python_version = 3.10 -warn_redundant_casts = true -warn_unused_ignores = true -check_untyped_defs = true -skip_version_check = true -ignore_missing_imports = true - -[tool.black] -line-length = 110 -target-version = ['py310'] - -[tool.isort] -profile = "black" -line_length = 110 -known_third_party = "c2cwsgiutils,c2cgeoform,qgis,pytest" -known_first_party = ["geomapfish_qgisserver", "c2cgeoportal_commons"] +[tool.ruff] +target-version = 'py310' [tool.poetry] name = 'c2cgeoportal' @@ -28,7 +13,7 @@ c2cwsgiutils = { extras = ["broadcast"], version = "6.1.5" } papyrus = "2.6.2" # commons transaction = "5.0" # commons "c2c.template" = "2.4.2" -GeoAlchemy2 = "0.15.2" +GeoAlchemy2 = "0.16.0" SQLAlchemy = "2.0.36" # commons "zope.event" = "5.0" # commons "zope.sqlalchemy" = "3.1" # commons @@ -38,8 +23,8 @@ psycopg2 = "2.9.10" pytz = "2024.2" [tool.poetry.dev-dependencies] -prospector = { extras = ["with_mypy", "with_bandit"], version = "1.13.3" } -prospector-profile-duplicated = "1.6.0" -prospector-profile-utils = "1.9.1" +prospector = { version = "1.13.3", extras = ["with_mypy", "with_bandit", "with_ruff"] } +prospector-profile-duplicated = "1.9.0" +prospector-profile-utils = "1.14.1" types-pyyaml = "6.0.12.20240917" pytest = "8.3.4" diff --git a/docker/qgisserver/tests/functional/accesscontrol_allow_to_edit_test.py b/docker/qgisserver/tests/functional/accesscontrol_allow_to_edit_test.py index dba24217fd..739a333560 100644 --- a/docker/qgisserver/tests/functional/accesscontrol_allow_to_edit_test.py +++ b/docker/qgisserver/tests/functional/accesscontrol_allow_to_edit_test.py @@ -1,4 +1,4 @@ -# Copyright (c) 2018-2023, Camptocamp SA +# Copyright (c) 2018-2024, Camptocamp SA # All rights reserved. # This program is free software; you can redistribute it and/or modify it under the terms of the @@ -12,9 +12,10 @@ from qgis.core import QgsFeature, QgsGeometry, QgsProject from shapely.geometry import LineString, box -from .accesscontrol_test import add_node_in_qgis_project, set_request_parameters from geomapfish_qgisserver.accesscontrol import OGCServerAccessControl +from .accesscontrol_test import add_node_in_qgis_project, set_request_parameters + area1 = box(400000, 70000, 800000, 100000) geom_in = LineString([[500000, 80000], [500000, 90000]]) geom_intersects = LineString([[500000, 50000], [500000, 90000]]) @@ -155,6 +156,4 @@ def test_allow_to_edit(self, server_iface, DBSession, test_data2): # noqa: igno geom.fromWkb(geometry.wkb) feature.setGeometry(geom) result = ogcserver_accesscontrol.allowToEdit(layer, feature) - assert expected == result, "allowToEdit with '{}', should return '{}'.".format( - user_name, expected - ) + assert expected == result, f"allowToEdit with '{user_name}', should return '{expected}'." diff --git a/docker/qgisserver/tests/functional/accesscontrol_protected_attributes_test.py b/docker/qgisserver/tests/functional/accesscontrol_protected_attributes_test.py index 7cb3d899fe..8e8a7d30d0 100644 --- a/docker/qgisserver/tests/functional/accesscontrol_protected_attributes_test.py +++ b/docker/qgisserver/tests/functional/accesscontrol_protected_attributes_test.py @@ -8,11 +8,15 @@ from unittest.mock import patch import pytest -from qgis.core import QgsProject, QgsVectorLayer +from qgis.core import QgsProject from shapely.geometry import box +from geomapfish_qgisserver.accesscontrol import ( + GeoMapFishAccessControl, + OGCServerAccessControl, +) + from .accesscontrol_test import add_node_in_qgis_project, set_request_parameters -from geomapfish_qgisserver.accesscontrol import GeoMapFishAccessControl, OGCServerAccessControl area1 = box(485869.5728, 76443.1884, 837076.5648, 299941.7864) @@ -124,7 +128,13 @@ def _test_data_protected(clean_dbsession, protected: bool): @pytest.fixture(scope="module") def test_data_not_protected(clean_dbsession): - from c2cgeoportal_commons.models.main import Functionality, LayerWMS, Metadata, OGCServer, Role + from c2cgeoportal_commons.models.main import ( + Functionality, + LayerWMS, + Metadata, + OGCServer, + Role, + ) from c2cgeoportal_commons.models.static import User test_data = _test_data_protected(clean_dbsession, False) @@ -140,7 +150,13 @@ def test_data_not_protected(clean_dbsession): @pytest.fixture(scope="module") def test_data_protected(clean_dbsession): - from c2cgeoportal_commons.models.main import Functionality, LayerWMS, Metadata, OGCServer, Role + from c2cgeoportal_commons.models.main import ( + Functionality, + LayerWMS, + Metadata, + OGCServer, + Role, + ) from c2cgeoportal_commons.models.static import User test_data = _test_data_protected(clean_dbsession, True) diff --git a/docker/qgisserver/tests/functional/accesscontrol_test.py b/docker/qgisserver/tests/functional/accesscontrol_test.py index c210f311b0..6d6d161be5 100644 --- a/docker/qgisserver/tests/functional/accesscontrol_test.py +++ b/docker/qgisserver/tests/functional/accesscontrol_test.py @@ -22,7 +22,9 @@ area1 = box(485869.5728, 76443.1884, 837076.5648, 299941.7864) -def set_request_parameters(server_iface, params, env={}): +def set_request_parameters(server_iface, params, env=None): + if env is None: + env = {} server_iface.configure_mock( **{ "configFilePath.return_value": params.get("MAP", None), @@ -199,9 +201,7 @@ def test_data(clean_dbsession): @pytest.fixture(scope="function") def wms_use_layer_ids(test_data): - """ - Activate WMSUseLayerIDs. - """ + """Activate WMSUseLayerIDs.""" project = test_data["project"] try: project.writeEntry("WMSUseLayerIDs", "/", True) @@ -267,7 +267,7 @@ def test_get_layers(self, server_iface, DBSession, test_data): # noqa: ignore=N layers = ogcserver_accesscontrol.get_layers(dbsession) assert set(expected.keys()) == set(layers.keys()) - for key in expected.keys(): + for key in expected: assert set(expected[key]) == {layer.name for layer in layers[key]} def test_get_roles(self, server_iface, DBSession, test_data): # noqa: ignore=N803 @@ -277,7 +277,7 @@ def test_get_roles(self, server_iface, DBSession, test_data): # noqa: ignore=N8 ) set_request_parameters(server_iface, {"USER_ID": "0"}) - assert "ROOT" == ogcserver_accesscontrol.get_roles(dbsession) + assert ogcserver_accesscontrol.get_roles(dbsession) == "ROOT" test_users = test_data["users"] test_roles = test_data["roles"] @@ -306,11 +306,11 @@ def test_get_restriction_areas(self, server_iface, DBSession, test_data): # noq server_iface, "qgisserver1", "no_project", 21781, lambda: dbsession ) - assert (Access.FULL, None) == ogcserver_accesscontrol.get_restriction_areas( + assert ogcserver_accesscontrol.get_restriction_areas( dbsession.query(LayerWMS).filter(LayerWMS.name == "private_layer1").one(), read_write=True, roles="ROOT", - ) + ) == (Access.FULL, None) for layer_names, rw, role_names, expected in ( (("private_layer1",), False, ("role1",), (Access.AREA, [area1])), @@ -322,9 +322,9 @@ def test_get_restriction_areas(self, server_iface, DBSession, test_data): # noq ] roles = [dbsession.query(Role).filter(Role.name == role_name).one() for role_name in role_names] ras = ogcserver_accesscontrol.get_restriction_areas(layers, rw, roles) - assert expected == ras, "get_restriction_areas with {} should return {}".format( - (layer_names, rw, role_names), expected - ) + assert ( + expected == ras + ), f"get_restriction_areas with {(layer_names, rw, role_names)} should return {expected}" def test_get_area(self, server_iface, DBSession, test_data): # noqa: ignore=N803 dbsession = DBSession() @@ -395,8 +395,8 @@ def test_cache_key(server_iface, DBSession, test_data): # noqa: ignore=N803 ) set_request_parameters(server_iface, {"USER_ID": "0"}, {"HTTP_HOST": "example.com"}) - assert "0" == server_iface.requestHandler().parameter("USER_ID") - assert "example.com-ROOT" == ogcserver_accesscontrol.cacheKey() + assert server_iface.requestHandler().parameter("USER_ID") == "0" + assert ogcserver_accesscontrol.cacheKey() == "example.com-ROOT" user = test_data["users"]["user12"] role1 = test_data["roles"]["role1"] diff --git a/docker/qgisserver/tests/functional/conftest.py b/docker/qgisserver/tests/functional/conftest.py index 9988539040..340a3280ce 100644 --- a/docker/qgisserver/tests/functional/conftest.py +++ b/docker/qgisserver/tests/functional/conftest.py @@ -3,12 +3,11 @@ import pytest from c2c.template.config import config +from c2cgeoportal_commons.testing import generate_mappers from qgis.server import QgsServerInterface from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker -from c2cgeoportal_commons.testing import generate_mappers - @pytest.fixture(scope="session") def settings(): @@ -33,7 +32,14 @@ def DBSession(settings): # noqa: ignore=N802 @pytest.fixture(scope="module") @pytest.mark.usefixtures("DBSession") def clean_dbsession(DBSession): # noqa: ignore=N803 - from c2cgeoportal_commons.models.main import OGCServer, RestrictionArea, Role, TreeItem, layer_ra, role_ra + from c2cgeoportal_commons.models.main import ( + OGCServer, + RestrictionArea, + Role, + TreeItem, + layer_ra, + role_ra, + ) from c2cgeoportal_commons.models.static import User, user_role def clean(): diff --git a/geoportal/c2cgeoportal_geoportal/__init__.py b/geoportal/c2cgeoportal_geoportal/__init__.py index 2e38add0fe..857f3c3c30 100644 --- a/geoportal/c2cgeoportal_geoportal/__init__.py +++ b/geoportal/c2cgeoportal_geoportal/__init__.py @@ -37,6 +37,7 @@ from typing import TYPE_CHECKING, Any, Optional, cast import c2cgeoform +import c2cgeoportal_commons.models import c2cwsgiutils import c2cwsgiutils.db import c2cwsgiutils.index @@ -50,6 +51,7 @@ import sqlalchemy.orm import zope.event.classhandler from c2cgeoform import translator +from c2cgeoportal_commons.models import InvalidateCacheEvent from c2cwsgiutils.health_check import HealthCheck from c2cwsgiutils.prometheus import MemoryMapCollector from deform import Form @@ -62,10 +64,14 @@ from pyramid_mako import add_mako_renderer from sqlalchemy.orm import joinedload -import c2cgeoportal_commons.models import c2cgeoportal_geoportal.views -from c2cgeoportal_commons.models import InvalidateCacheEvent -from c2cgeoportal_geoportal.lib import C2CPregenerator, caching, check_collector, checker, oidc +from c2cgeoportal_geoportal.lib import ( + C2CPregenerator, + caching, + check_collector, + checker, + oidc, +) from c2cgeoportal_geoportal.lib.cacheversion import version_cache_buster from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers @@ -79,7 +85,9 @@ from c2cgeoportal_geoportal.views.entry import Entry, canvas_view, custom_view if TYPE_CHECKING: - from c2cgeoportal_commons.models import static # pylint: disable=ungrouped-imports,useless-suppression + from c2cgeoportal_commons.models import ( + static, # pylint: disable=ungrouped-imports,useless-suppression + ) _LOG = logging.getLogger(__name__) @@ -174,7 +182,6 @@ def add_interface_ngeo( permission: str | None = None, ) -> None: """Add the ngeo interfaces views and routes.""" - config.add_route(route_name, route, request_method="GET") # Permalink theme: recover the theme for generating custom viewer.js url config.add_route( @@ -202,7 +209,6 @@ def add_interface_canvas( permission: str | None = None, ) -> None: """Add the ngeo interfaces views and routes.""" - renderer = f"/etc/geomapfish/interfaces/{route_name}.html.mako" config.add_route(route_name, route, request_method="GET") # Permalink theme: recover the theme for generating custom viewer.js URL @@ -235,7 +241,6 @@ def add_interface_custom( permission: str | None = None, ) -> None: """Add custom interfaces views and routes.""" - config.add_route(route_name, route, request_method="GET") # Permalink theme: recover the theme for generating custom viewer.js URL config.add_route( @@ -259,7 +264,6 @@ def add_interface_custom( def add_admin_interface(config: pyramid.config.Configurator) -> None: """Add the administration interface views and routes.""" - assert c2cgeoportal_commons.models.DBSession is not None config.add_request_method( @@ -318,7 +322,6 @@ def is_allowed_url( Allowed if URL netloc is request host or is found in allowed hosts. """ - url_netloc = urllib.parse.urlparse(url).netloc return url_netloc, url_netloc == request.host or url_netloc in _get_netlocs(allowed_hosts) @@ -329,7 +332,6 @@ def is_allowed_host(request: pyramid.request.Request) -> bool: Allowed if URL netloc is request host or is found in allowed hosts. """ - return request.host in request.registry.settings.get("allowed_hosts", []) @@ -354,7 +356,7 @@ def is_valid_referrer(request: pyramid.request.Request, settings: dict[str, Any] def create_get_user_from_request( - settings: dict[str, Any] + settings: dict[str, Any], ) -> Callable[[pyramid.request.Request, str | None], Optional["static.User"]]: """Get the get_user_from_request function.""" @@ -370,8 +372,12 @@ def get_user_from_request( * it has been deactivated * the referrer is invalid """ - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.static import ( # pylint: disable=import-outside-toplevel + User, + ) assert DBSession is not None @@ -467,8 +473,12 @@ def default_user_validator(request: pyramid.request.Request, username: str, pass otherwise. """ del request # unused - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.static import ( # pylint: disable=import-outside-toplevel + User, + ) assert DBSession is not None @@ -930,7 +940,9 @@ def init_db_sessions( ) c2cgeoportal_commons.models.Base.metadata.clear() - from c2cgeoportal_commons.models import main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + main, + ) if health_check is not None: for name, session in c2cgeoportal_commons.models.DBSessions.items(): diff --git a/geoportal/c2cgeoportal_geoportal/lib/__init__.py b/geoportal/c2cgeoportal_geoportal/lib/__init__.py index 3590649165..b531a7dbee 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/__init__.py +++ b/geoportal/c2cgeoportal_geoportal/lib/__init__.py @@ -38,10 +38,10 @@ import dateutil import pyramid.request import pyramid.response +from c2cgeoportal_commons.lib.url import get_url2 from pyramid.interfaces import IRoutePregenerator from zope.interface import implementer -from c2cgeoportal_commons.lib.url import get_url2 from c2cgeoportal_geoportal.lib.cacheversion import get_cache_version from c2cgeoportal_geoportal.lib.caching import get_region @@ -145,8 +145,12 @@ def get_setting(settings: Any, path: Iterable[str], default: Any = None) -> Any: @_CACHE_REGION_OBJ.cache_on_arguments() def get_ogc_server_wms_url_ids(request: pyramid.request.Request, host: str) -> dict[str, list[int]]: """Get the OGCServer ids mapped on the WMS URL.""" - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import OGCServer # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + OGCServer, + ) del host # used for cache assert DBSession is not None @@ -163,8 +167,12 @@ def get_ogc_server_wms_url_ids(request: pyramid.request.Request, host: str) -> d @_CACHE_REGION_OBJ.cache_on_arguments() def get_ogc_server_wfs_url_ids(request: pyramid.request.Request, host: str) -> dict[str, list[int]]: """Get the OGCServer ids mapped on the WFS URL.""" - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import OGCServer # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + OGCServer, + ) del host # used for cache assert DBSession is not None @@ -218,7 +226,10 @@ def _get_intranet_networks( @_CACHE_REGION.cache_on_arguments() def get_role_id(name: str) -> int: """Get the role ID.""" - from c2cgeoportal_commons.models import DBSession, main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + main, + ) assert DBSession is not None @@ -250,7 +261,4 @@ def get_roles_name(request: pyramid.request.Request) -> pyramid.response.Respons def is_intranet(request: pyramid.request.Request) -> bool: """Get if it's an intranet user.""" address = ipaddress.ip_address(request.client_addr) - for network in _get_intranet_networks(request): - if address in network: - return True - return False + return any(address in network for network in _get_intranet_networks(request)) diff --git a/geoportal/c2cgeoportal_geoportal/lib/authentication.py b/geoportal/c2cgeoportal_geoportal/lib/authentication.py index b8feb9dd2e..e311524f3a 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/authentication.py +++ b/geoportal/c2cgeoportal_geoportal/lib/authentication.py @@ -67,7 +67,7 @@ def __init__( self.debug = debug def unauthenticated_userid(self, request: pyramid.request.Request) -> str | None: - if not request.method == "GET" or "auth" not in request.params: + if request.method != "GET" or "auth" not in request.params: return None auth_enc = request.params.get("auth") if auth_enc is None: diff --git a/geoportal/c2cgeoportal_geoportal/lib/caching.py b/geoportal/c2cgeoportal_geoportal/lib/caching.py index 826c5ef254..f887b7c728 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/caching.py +++ b/geoportal/c2cgeoportal_geoportal/lib/caching.py @@ -33,6 +33,7 @@ import pyramid.interfaces import zope.interface +from c2cgeoportal_commons.models import Base from dogpile.cache.api import NO_VALUE, CacheBackend, CachedValue, NoValue from dogpile.cache.backends.memory import MemoryBackend from dogpile.cache.backends.redis import RedisBackend, RedisSentinelBackend @@ -40,8 +41,6 @@ from dogpile.cache.util import sha1_mangle_key from sqlalchemy.orm.util import identity_key -from c2cgeoportal_commons.models import Base - if TYPE_CHECKING: from dogpile.cache.api import SerializedReturnType else: @@ -65,7 +64,6 @@ def keygen_function(namespace: Any, function: Callable[..., Any]) -> Callable[.. This is used by :meth:`.CacheRegion.cache_on_arguments` to generate a cache key from a decorated function. """ - if namespace is None: namespace = (function.__module__, function.__name__) else: @@ -140,7 +138,7 @@ def get(self, key: str) -> CachedValue | bytes | NoValue: assert isinstance(val, bytes) value = self._redis.deserializer(val) # type: ignore[misc] if value != NO_VALUE and self._use_memory_cache: - assert isinstance(value, (CachedValue, bytes)) + assert isinstance(value, CachedValue | bytes) self._memory.set(key, value) return value diff --git a/geoportal/c2cgeoportal_geoportal/lib/checker.py b/geoportal/c2cgeoportal_geoportal/lib/checker.py index 710ba551ae..9b9abdd5a0 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/checker.py +++ b/geoportal/c2cgeoportal_geoportal/lib/checker.py @@ -147,8 +147,8 @@ def check(_request: pyramid.request.Request, response: pyramid.response.Response health_check.add_url_check( name="checker_fulltextsearch", - url=lambda r: get_both(r)["url"], # type: ignore - headers=lambda r: get_both(r)["headers"], # type: ignore + url=lambda r: get_both(r)["url"], # type: ignore[misc] + headers=lambda r: get_both(r)["headers"], # type: ignore[misc] params={"query": fts_settings["search"], "limit": "1"}, check_cb=check, level=fts_settings["level"], @@ -156,8 +156,12 @@ def check(_request: pyramid.request.Request, response: pyramid.response.Response def _themes_errors(settings: dict[str, Any], health_check: c2cwsgiutils.health_check.HealthCheck) -> None: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import Interface # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Interface, + ) assert DBSession is not None @@ -256,7 +260,7 @@ def __call__(self, request: pyramid.request.Request) -> None: cmd: list[str] = ["check-example", url] env = dict(os.environ) for name, value in self.route.get("environment", {}).items(): - if isinstance(value, (list, dict)): + if isinstance(value, list | dict): value = json.dumps(value) elif not isinstance(value, str): value = str(value) diff --git a/geoportal/c2cgeoportal_geoportal/lib/common_headers.py b/geoportal/c2cgeoportal_geoportal/lib/common_headers.py index c80a59c271..3d276abfcc 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/common_headers.py +++ b/geoportal/c2cgeoportal_geoportal/lib/common_headers.py @@ -123,7 +123,6 @@ def _set_common_headers( content_type: str | None, ) -> pyramid.response.Response: """Set the common headers.""" - response.headers.update(service_headers_settings.get("headers", {})) if cache in (Cache.PRIVATE, Cache.PRIVATE_NO): diff --git a/geoportal/c2cgeoportal_geoportal/lib/dbreflection.py b/geoportal/c2cgeoportal_geoportal/lib/dbreflection.py index 6fcaaaa8b5..feb0c0b989 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/dbreflection.py +++ b/geoportal/c2cgeoportal_geoportal/lib/dbreflection.py @@ -76,7 +76,9 @@ def __get__( return getattr(target, self.value_attr) if target else None def __set__(self, obj: str, val: str) -> None: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) assert DBSession is not None @@ -118,8 +120,10 @@ def get_table( engine = session.bind.engine metadata = MetaData() else: - from c2cgeoportal_commons.models import Base # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + Base, + DBSession, + ) assert DBSession is not None assert DBSession.bind is not None @@ -162,7 +166,6 @@ def get_class( valid string. If there is no table identified by tablename in the database a NoSuchTableError SQLAlchemy exception is raised. """ - tablename, schema = _get_schema(tablename) table = get_table(tablename, schema, None, primary_key=primary_key) @@ -187,7 +190,9 @@ def _create_class( readonly_attributes: list[str] | None = None, pk_name: str | None = None, ) -> type: - from c2cgeoportal_commons.models import Base # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + Base, + ) exclude_properties = exclude_properties or () attributes = { @@ -201,7 +206,7 @@ def _create_class( # The randint is to fix the SAWarning: This declarative base already contains a class with the same # class name and module name cls = type( - f"{table.name.capitalize()}_{random.randint(0, 9999999)}", # nosec + f"{table.name.capitalize()}_{random.randint(0, 9999999)}", # noqa: S311 (GeoInterface, Base), attributes, ) diff --git a/geoportal/c2cgeoportal_geoportal/lib/filter_capabilities.py b/geoportal/c2cgeoportal_geoportal/lib/filter_capabilities.py index c1f2f566b8..ea5946a499 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/filter_capabilities.py +++ b/geoportal/c2cgeoportal_geoportal/lib/filter_capabilities.py @@ -32,25 +32,33 @@ import xml.sax.xmlreader # nosec from collections.abc import Callable from io import StringIO -from typing import Any, Union +from typing import Any from xml.sax.saxutils import XMLFilterBase, XMLGenerator # nosec import defusedxml.expatreader import pyramid.httpexceptions import pyramid.request import requests +from c2cgeoportal_commons.lib.url import Url from owslib.map.wms111 import ContentMetadata as ContentMetadata111 from owslib.map.wms130 import ContentMetadata as ContentMetadata130 from owslib.wms import WebMapService from pyramid.httpexceptions import HTTPBadGateway -from c2cgeoportal_commons.lib.url import Url -from c2cgeoportal_geoportal.lib import caching, get_ogc_server_wfs_url_ids, get_ogc_server_wms_url_ids -from c2cgeoportal_geoportal.lib.layers import get_private_layers, get_protected_layers, get_writable_layers +from c2cgeoportal_geoportal.lib import ( + caching, + get_ogc_server_wfs_url_ids, + get_ogc_server_wms_url_ids, +) +from c2cgeoportal_geoportal.lib.layers import ( + get_private_layers, + get_protected_layers, + get_writable_layers, +) _CACHE_REGION = caching.get_region("std") _LOG = logging.getLogger(__name__) -ContentMetadata = Union[ContentMetadata111, ContentMetadata130] +ContentMetadata = ContentMetadata111 | ContentMetadata130 @_CACHE_REGION.cache_on_arguments() @@ -111,7 +119,6 @@ def filter_capabilities( request: pyramid.request.Request, content: str, wms: bool, url: Url, headers: dict[str, str] ) -> str: """Filter the WMS/WFS capabilities.""" - wms_structure_ = wms_structure(request, url, headers.get("Host")) ogc_server_ids = ( @@ -153,7 +160,6 @@ def filter_capabilities( def filter_wfst_capabilities(content: str, wfs_url: Url, request: pyramid.request.Request) -> str: """Filter the WTS capabilities.""" - writable_layers: set[str] = set() ogc_server_ids = get_ogc_server_wfs_url_ids(request, request.host).get(wfs_url.url()) if ogc_server_ids is None: @@ -312,7 +318,7 @@ def _keep_layer(self, layer_name: str) -> bool: ) def characters(self, content: str) -> None: - if self.in_name and self.layers_path and not self.layers_path[-1].self_hidden is True: + if self.in_name and self.layers_path and self.layers_path[-1].self_hidden is not True: layer_name = normalize_typename(content) if self._keep_layer(layer_name): for layer in self.layers_path: @@ -342,9 +348,8 @@ def normalize_tag(tag: str) -> str: e.g. '{https://....}TypeName' -> 'TypeName' """ normalized = tag - if len(tag) >= 3: - if tag[0] == "{": - normalized = tag[1:].split("}")[1] + if len(tag) >= 3 and tag[0] == "{": + normalized = tag[1:].split("}")[1] return normalized.lower() diff --git a/geoportal/c2cgeoportal_geoportal/lib/functionality.py b/geoportal/c2cgeoportal_geoportal/lib/functionality.py index bcc2179f94..74563b3ae1 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/functionality.py +++ b/geoportal/c2cgeoportal_geoportal/lib/functionality.py @@ -30,9 +30,9 @@ from typing import Any, cast import pyramid.request +from c2cgeoportal_commons.models import main, static from sqlalchemy.orm import joinedload -from c2cgeoportal_commons.models import main, static from c2cgeoportal_geoportal.lib import get_typed, get_types_map, is_intranet from c2cgeoportal_geoportal.lib.caching import get_region @@ -43,7 +43,9 @@ @_CACHE_REGION_OBJ.cache_on_arguments() def _get_role(name: str) -> dict[str, Any]: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) assert DBSession is not None diff --git a/geoportal/c2cgeoportal_geoportal/lib/layers.py b/geoportal/c2cgeoportal_geoportal/lib/layers.py index 99fdb22165..115a44fb79 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/layers.py +++ b/geoportal/c2cgeoportal_geoportal/lib/layers.py @@ -42,7 +42,10 @@ def _get_layers_query(request: Request, what: sqlalchemy.orm.Mapper[Any] | type[Any]) -> Query[Any]: - from c2cgeoportal_commons.models import DBSession, main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + main, + ) assert DBSession is not None @@ -64,7 +67,9 @@ def get_protected_layers_query( Private layers but accessible to the user. """ - from c2cgeoportal_commons.models import main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + main, + ) q = _get_layers_query(request, what) q = q.filter(main.Layer.public.is_(False)) @@ -76,7 +81,9 @@ def get_protected_layers_query( def get_writable_layers_query(request: Request, ogc_server_ids: Iterable[int]) -> Query["main.LayerWMS"]: """Get the writable layers query.""" - from c2cgeoportal_commons.models import main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + main, + ) q = _get_layers_query(request, main.LayerWMS) return ( @@ -92,7 +99,10 @@ def get_protected_layers(request: Request, ogc_server_ids: Iterable[int]) -> dic Private layers but accessible to the user. """ - from c2cgeoportal_commons.models import DBSession, main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + main, + ) assert DBSession is not None @@ -104,7 +114,9 @@ def get_protected_layers(request: Request, ogc_server_ids: Iterable[int]) -> dic def get_writable_layers(request: Request, ogc_server_ids: Iterable[int]) -> dict[int, "main.LayerWMS"]: """Get the writable layers.""" - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) assert DBSession is not None @@ -117,7 +129,10 @@ def get_writable_layers(request: Request, ogc_server_ids: Iterable[int]) -> dict @CACHE_REGION.cache_on_arguments() def get_private_layers(ogc_server_ids: Iterable[int]) -> dict[int, "main.LayerWMS"]: """Get the private layers.""" - from c2cgeoportal_commons.models import DBSession, main # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + main, + ) assert DBSession is not None diff --git a/geoportal/c2cgeoportal_geoportal/lib/lingva_extractor.py b/geoportal/c2cgeoportal_geoportal/lib/lingva_extractor.py index 253606f9fa..59a72a6210 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/lingva_extractor.py +++ b/geoportal/c2cgeoportal_geoportal/lib/lingva_extractor.py @@ -29,10 +29,10 @@ import json import os import re -import subprocess +import subprocess # nosec import traceback from collections.abc import Callable -from typing import TYPE_CHECKING, Any, Optional, cast +from typing import TYPE_CHECKING, Any, cast from xml.dom import Node from xml.parsers.expat import ExpatError @@ -42,6 +42,7 @@ import yaml from bottle import MakoTemplate, template from c2c.template.config import config +from c2cgeoportal_commons.lib.url import Url, get_url2 from defusedxml.minidom import parseString from geoalchemy2.types import Geometry from lingva.extractors import Extractor, Message @@ -54,13 +55,14 @@ from sqlalchemy.orm.util import class_mapper import c2cgeoportal_geoportal -from c2cgeoportal_commons.lib.url import Url, get_url2 from c2cgeoportal_geoportal.lib.bashcolor import Color, colorize from c2cgeoportal_geoportal.lib.caching import init_region from c2cgeoportal_geoportal.views.layers import Layers, get_layer_class if TYPE_CHECKING: - from c2cgeoportal_commons.models import main # pylint: disable=ungrouped-imports,useless-suppression + from c2cgeoportal_commons.models import ( + main, # pylint: disable=ungrouped-imports,useless-suppression + ) class LinguaExtractorException(Exception): @@ -76,7 +78,7 @@ def _get_config(key: str, default: str | None = None) -> str | None: """ request = pyramid.threadlocal.get_current_request() if request is not None: - return cast(Optional[str], request.params.get(key.lower(), default)) + return cast(str | None, request.params.get(key.lower(), default)) return os.environ.get(key, default) @@ -174,23 +176,23 @@ def __call__( int_filename = filename if re.match("^" + re.escape(f"./{self.config['package']}/templates"), filename): try: - empty_template = Template("") # nosec + empty_template = Template("") # noqa: S702 - class Lookup(TemplateLookup): # type: ignore + class Lookup(TemplateLookup): # type: ignore[misc] def get_template(self, uri: str) -> Template: del uri # unused return empty_template - class MyTemplate(MakoTemplate): # type: ignore + class MyTemplate(MakoTemplate): # type: ignore[misc] tpl = None def prepare(self, **kwargs: Any) -> None: options.update({"input_encoding": self.encoding}) lookup = Lookup(**kwargs) if self.source: - self.tpl = Template(self.source, lookup=lookup, **kwargs) # nosec + self.tpl = Template(self.source, lookup=lookup, **kwargs) # noqa: S702 else: - self.tpl = Template( # nosec + self.tpl = Template( # noqa: S702 uri=self.name, filename=self.filename, lookup=lookup, **kwargs ) @@ -252,10 +254,13 @@ def init_db(settings: dict[str, Any]) -> None: First test the connection, on when environment it should be OK, with the command line we should get an exception ind initialize the connection. """ - try: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import Theme # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Theme, + ) assert DBSession is not None @@ -302,7 +307,7 @@ def __call__( init_region({"backend": "dogpile.cache.memory"}, "obj") with open(filename, encoding="utf8") as config_file: - gmf_config = yaml.load(config_file, Loader=yaml.BaseLoader) # nosec + gmf_config = yaml.load(config_file, Loader=yaml.BaseLoader) # noqa: S506 # For application config (config.yaml) if "vars" in gmf_config: return self._collect_app_config(filename) @@ -332,7 +337,9 @@ def _collect_app_config(self, filename: str) -> list[Message]: DBSession, DBSessions, ) - from c2cgeoportal_commons.models.main import Metadata # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Metadata, + ) assert DBSession is not None @@ -385,7 +392,6 @@ def _collect_app_config(self, filename: str) -> list[Message]: interface_config.get("constants", {}) .get("gmfDisplayQueryGridOptions", {}) .get("mergeTabs", {}) - .keys() ): location = ( f"interfaces_config/{interface}/constants/gmfDisplayQueryGridOptions/" @@ -479,7 +485,9 @@ def __call__( try: init_db(self.config) - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) assert DBSession is not None @@ -588,8 +596,12 @@ def _import( has_interfaces: bool = True, name_regex: str = ".*", ) -> None: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import Interface # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Interface, + ) assert DBSession is not None @@ -667,8 +679,12 @@ def _import_layer_wms(self, layer: "main.Layer", messages: list[str]) -> None: raise def _import_layer_wmts(self, layer: "main.Layer", messages: list[str]) -> None: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.main import OGCServer # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + OGCServer, + ) assert DBSession is not None diff --git a/geoportal/c2cgeoportal_geoportal/lib/oauth2.py b/geoportal/c2cgeoportal_geoportal/lib/oauth2.py index a43daed74f..9becfd4aa2 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/oauth2.py +++ b/geoportal/c2cgeoportal_geoportal/lib/oauth2.py @@ -30,12 +30,12 @@ from typing import Any, TypedDict import basicauth +import c2cgeoportal_commons import oauthlib.common import oauthlib.oauth2 import pyramid.threadlocal from pyramid.httpexceptions import HTTPBadRequest -import c2cgeoportal_commons from c2cgeoportal_geoportal.lib.caching import get_region _LOG = logging.getLogger(__name__) @@ -73,9 +73,11 @@ def authenticate_client( both body and query can be obtained by direct attribute access, i.e. request.client_id for client_id in the URL query. - Keyword Arguments: - - request: oauthlib.common.Request + Arguments: + --------- + request: oauthlib.common.Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: True or False @@ -86,6 +88,7 @@ def authenticate_client( - Refresh Token Grant .. _`HTTP Basic Authentication Scheme`: https://tools.ietf.org/html/rfc1945#section-11.1 + """ del args, kwargs @@ -117,7 +120,10 @@ def authenticate_client_id( _LOG.debug("authenticate_client_id %s", client_id) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -163,9 +169,11 @@ def client_authentication_required( client credentials or whenever Client provided client authentication, see `Section 6`_ - Keyword Arguments: - - request: oauthlib.common.Request + Arguments: + --------- + request: oauthlib.common.Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: True or False @@ -177,6 +185,7 @@ def client_authentication_required( .. _`Section 4.3.2`: https://tools.ietf.org/html/rfc6749#section-4.3.2 .. _`Section 4.1.3`: https://tools.ietf.org/html/rfc6749#section-4.1.3 .. _`Section 6`: https://tools.ietf.org/html/rfc6749#section-6 + """ del request, args, kwargs @@ -205,24 +214,30 @@ def confirm_redirect_uri( the client's allowed redirect URIs, but against the URI used when the code was saved. - Keyword Arguments: - - client_id: Unicode client identifier - code: Unicode authorization_code. - redirect_uri: Unicode absolute URI - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + code: Unicode authorization_code. + redirect_uri: Unicode absolute URI + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: True or False Method is used by: - Authorization Code Grant (during token request) + """ del args, kwargs _LOG.debug("confirm_redirect_uri %s %s", client_id, redirect_uri) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -248,16 +263,19 @@ def get_default_redirect_uri( """ Get the default redirect URI for the client. - Keyword Arguments: - - client_id: Unicode client identifier - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: The default redirect URI for the client Method is used by: - Authorization Code Grant - Implicit Grant + """ del request, args, kwargs @@ -275,10 +293,12 @@ def get_default_scopes( """ Get the default scopes for the client. - Keyword Arguments: - - client_id: Unicode client identifier - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: List of default scopes @@ -287,6 +307,7 @@ def get_default_scopes( - Implicit Grant - Resource Owner Password Credentials Grant - Client Credentials grant + """ del request, args, kwargs @@ -304,15 +325,18 @@ def get_original_scopes( """ Get the list of scopes associated with the refresh token. - Keyword Arguments: - - refresh_token: Unicode refresh token - request: The HTTP Request + Arguments: + --------- + refresh_token: Unicode refresh token + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: List of scopes. Method is used by: - Refresh token grant + """ del refresh_token, request, args, kwargs @@ -352,17 +376,20 @@ def introspect_token( efficiency, but must fallback to other types to be compliant with RFC. The dict of claims is added to request.token after this method. - Keyword Arguments: - - token: The token string. - token_type_hint: access_token or refresh_token. - request: OAuthlib request. + Arguments: + --------- + token: The token string. + token_type_hint: access_token or refresh_token. + request: OAuthlib request. + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Introspect Endpoint (all grants are compatible) .. _`Introspect Claims`: https://tools.ietf.org/html/rfc7662#section-2.2 .. _`JWT Claims`: https://tools.ietf.org/html/rfc7519#section-4 + """ del token, request, args, kwargs @@ -381,20 +408,26 @@ def invalidate_authorization_code( """ Invalidate an authorization code after use. - Keyword Arguments: - - client_id: Unicode client identifier - code: The authorization code grant (request.code). - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + code: The authorization code grant (request.code). + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant + """ del args, kwargs _LOG.debug("invalidate_authorization_code %s", client_id) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -426,14 +459,17 @@ def is_within_original_scope( used in situations where returning all valid scopes from the get_original_scopes is not practical. - Keyword Arguments: - - request_scopes: A list of scopes that were requested by client - refresh_token: Unicode refresh_token - request: The HTTP Request + Arguments: + --------- + request_scopes: A list of scopes that were requested by client + refresh_token: Unicode refresh_token + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Refresh token grant + """ del request, args, kwargs @@ -452,14 +488,17 @@ def revoke_token( """ Revoke an access or refresh token. - Keyword Arguments: - - token: The token string. - token_type_hint: access_token or refresh_token. - request: The HTTP Request + Arguments: + --------- + token: The token string. + token_type_hint: access_token or refresh_token. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Revocation Endpoint + """ del token, request, args, kwargs @@ -475,12 +514,13 @@ def rotate_refresh_token(self, request: oauthlib.common.Request) -> bool: or replaced with a new one (rotated). Return True to rotate and and False for keeping original. - Keyword Arguments: - - request: oauthlib.common.Request + Arguments: + --------- + request: oauthlib.common.Request Method is used by: - Refresh Token Grant + """ del request @@ -515,11 +555,13 @@ def save_authorization_code( chose to send one. That value should be saved and used in 'validate_code'. - Keyword Arguments: - - client_id: Unicode client identifier - code: A dict of the authorization code grant and, optionally, state. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + code: A dict of the authorization code grant and, optionally, state. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant @@ -528,12 +570,16 @@ def save_authorization_code( Code Challenge (request.code_challenge) and Code Challenge Method (request.code_challenge_method) + """ del args, kwargs _LOG.debug("save_authorization_code %s", client_id) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -604,11 +650,13 @@ def save_bearer_token( Note that while "scope" is a string-separated list of authorized scopes, the original list is still available in request.scopes - Keyword Arguments: - - client_id: Unicode client identifier - token: A Bearer token dict - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + token: A Bearer token dict + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Returns: The default redirect URI for the client @@ -617,12 +665,16 @@ def save_bearer_token( - Implicit Grant - Resource Owner Password Credentials Grant (might not associate a client) - Client Credentials grant + """ del args, kwargs _LOG.debug("save_bearer_token") - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -654,11 +706,11 @@ def validate_bearer_token( """ Ensure the Bearer token is valid and authorized access to scopes. - Keyword Arguments: - - token: A string of random characters. - scopes: A list of scopes associated with the protected resource. - request: The HTTP Request + Arguments: + --------- + token: A string of random characters. + scopes: A list of scopes associated with the protected resource. + request: The HTTP Request A key to OAuth 2 security and restricting impact of leaked tokens is the short expiration time of tokens, *always ensure the token has not @@ -690,22 +742,25 @@ def validate_bearer_token( one provided for django these attributes will be made available in all protected views as keyword arguments. - Keyword Arguments: - - token: Unicode Bearer token - scopes: List of scopes (defined by you) - request: The HTTP Request + Arguments: + --------- + token: Unicode Bearer token + scopes: List of scopes (defined by you) + request: The HTTP Request Method is indirectly used by all core Bearer token issuing grant types: - Authorization Code Grant - Implicit Grant - Resource Owner Password Credentials Grant - Client Credentials Grant - """ + """ _LOG.debug("validate_bearer_token %s", scopes) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -736,20 +791,26 @@ def validate_client_id( to set request.client to the client object associated with the given client_id. - Keyword Arguments: - - client_id: Unicode client identifier - request: oauthlib.common.Request + Arguments: + --------- + client_id: Unicode client identifier + request: oauthlib.common.Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant - Implicit Grant + """ del args, kwargs _LOG.debug("validate_client_id") - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -784,21 +845,27 @@ def validate_code( associated with this authorization code. Similarly request.scopes must also be set. - Keyword Arguments: - - client_id: Unicode client identifier - code: Unicode authorization code - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + code: Unicode authorization code + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant + """ del args, kwargs _LOG.debug("validate_code %s", client_id) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -839,18 +906,21 @@ def validate_grant_type( """ Ensure client is authorized to use the grant_type requested. - Keyword Arguments: - - client_id: Unicode client identifier - grant_type: Unicode grant type, i.e. authorization_code, password. - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + grant_type: Unicode grant type, i.e. authorization_code, password. + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant - Resource Owner Password Credentials Grant - Client Credentials Grant - Refresh Token Grant + """ del client, request, args, kwargs @@ -877,21 +947,27 @@ def validate_redirect_uri( All clients should register the absolute URIs of all URIs they intend to redirect to. The registration is outside of the scope of oauthlib. - Keyword Arguments: - - client_id: Unicode client identifier - redirect_uri: Unicode absolute URI - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + redirect_uri: Unicode absolute URI + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant - Implicit Grant + """ del request, args, kwargs _LOG.debug("validate_redirect_uri %s %s", client_id, redirect_uri) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -918,22 +994,28 @@ def validate_refresh_token( OBS! The request.user attribute should be set to the resource owner associated with this refresh token. - Keyword Arguments: - - refresh_token: Unicode refresh token - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + refresh_token: Unicode refresh token + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant (indirectly by issuing refresh tokens) - Resource Owner Password Credentials Grant (also indirectly) - Refresh Token Grant + """ del args, kwargs _LOG.debug("validate_refresh_token %s", client.client_id if client else None) - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -961,16 +1043,19 @@ def validate_response_type( """ Ensure client is authorized to use the response_type requested. - Keyword Arguments: - - client_id: Unicode client identifier - response_type: Unicode response type, i.e. code, token. - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + response_type: Unicode response type, i.e. code, token. + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Authorization Code Grant - Implicit Grant + """ del client, request, args, kwargs @@ -990,18 +1075,21 @@ def validate_scopes( """ Ensure the client is authorized access to requested scopes. - Keyword Arguments: - - client_id: Unicode client identifier - scopes: List of scopes (defined by you) - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Arguments: + --------- + client_id: Unicode client identifier + scopes: List of scopes (defined by you) + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by all core grant types: - Authorization Code Grant - Implicit Grant - Resource Owner Password Credentials Grant - Client Credentials Grant + """ del client, request, args, kwargs @@ -1026,15 +1114,18 @@ def validate_user( not set you will be unable to associate a token with a user in the persistence method used (commonly, save_bearer_token). - Keyword Arguments: - - username: Unicode username - password: Unicode password - client: Client object set by you, see authenticate_client. - request: The HTTP Request + Argument: + --------- + username: Unicode username + password: Unicode password + client: Client object set by you, see authenticate_client. + request: The HTTP Request + args: Additional arguments, ignored + kwargs: Additional keyword arguments, ignored Method is used by: - Resource Owner Password Credentials Grant + """ del password, client, request, args, kwargs @@ -1056,10 +1147,10 @@ def is_pkce_required(self, client_id: int, request: oauthlib.common.Request) -> a technique to mitigate against the threat through the use of Proof Key for Code Exchange (PKCE, pronounced “pixy”). See RFC7636. - Keyword Arguments: - - client_id: Client identifier. - request (oauthlib.common.Request): OAuthlib request. + Arguments: + --------- + client_id: Client identifier. + request (oauthlib.common.Request): OAuthlib request. Method is used by: @@ -1068,7 +1159,10 @@ def is_pkce_required(self, client_id: int, request: oauthlib.common.Request) -> """ - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -1096,20 +1190,23 @@ def get_code_challenge(self, code: str, request: oauthlib.common.Request) -> str Return the code_challenge associated to the code. If None is returned, code is considered to not be associated to any challenges. - Keyword Arguments: - - code: Authorization code. - request: OAuthlib request. + Arguments: + --------- + code: Authorization code. + request: OAuthlib request. Return: - code_challenge string Method is used by: Authorization Code Grant - when PKCE is active + """ - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None @@ -1136,10 +1233,10 @@ def get_code_challenge_method(self, code: str, request: oauthlib.common.Request) Must return plain or S256. You can return a custom value if you have implemented your own AuthorizationCodeGrant class. - Keyword Arguments: - - code: Authorization code. - request: OAuthlib request. + Arguments: + --------- + code: Authorization code. + request: OAuthlib request. Return type: @@ -1148,8 +1245,12 @@ def get_code_challenge_method(self, code: str, request: oauthlib.common.Request) Method is used by: Authorization Code Grant - when PKCE is active + """ - from c2cgeoportal_commons.models import DBSession, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + static, + ) assert DBSession is not None diff --git a/geoportal/c2cgeoportal_geoportal/lib/oidc.py b/geoportal/c2cgeoportal_geoportal/lib/oidc.py index 083aa2b116..71964b445c 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/oidc.py +++ b/geoportal/c2cgeoportal_geoportal/lib/oidc.py @@ -34,7 +34,11 @@ import pyramid.response import simple_openid_connect.client import simple_openid_connect.data -from pyramid.httpexceptions import HTTPBadRequest, HTTPInternalServerError, HTTPUnauthorized +from pyramid.httpexceptions import ( + HTTPBadRequest, + HTTPInternalServerError, + HTTPUnauthorized, +) from pyramid.security import remember from c2cgeoportal_geoportal.lib.caching import get_region @@ -48,9 +52,7 @@ # User create on demand class DynamicUser(NamedTuple): - """ - User created dynamically. - """ + """User created dynamically.""" id: int username: str @@ -62,10 +64,7 @@ class DynamicUser(NamedTuple): @_CACHE_REGION_OBJ.cache_on_arguments() def get_oidc_client(request: pyramid.request.Request, host: str) -> simple_openid_connect.client.OpenidClient: - """ - Get the OpenID Connect client from the request settings. - """ - + """Get the OpenID Connect client from the request settings.""" del host # used for cache key authentication_settings = request.registry.settings.get("authentication", {}) @@ -83,9 +82,7 @@ def get_oidc_client(request: pyramid.request.Request, host: str) -> simple_openi class OidcRememberObject(TypedDict): - """ - The JSON object that is stored in a cookie to remember the user. - """ + """The JSON object that is stored in a cookie to remember the user.""" access_token: str access_token_expires: str @@ -146,11 +143,13 @@ def get_user_from_remember( :param settings: The OpenID Connect configuration. :param update_create_user: If the user should be updated or created if it does not exist. """ - # Those imports are here to avoid initializing the models module before the database schema are # correctly initialized. from c2cgeoportal_commons import models # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models import main, static # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + main, + static, + ) assert models.DBSession is not None @@ -213,9 +212,7 @@ def get_user_from_remember( class OidcRemember: - """ - Build the abject that we want to remember in the cookie. - """ + """Build the abject that we want to remember in the cookie.""" def __init__(self, request: pyramid.request.Request): self.request = request @@ -229,10 +226,7 @@ def remember( ), host: str, ) -> OidcRememberObject: - """ - Remember the user in the cookie. - """ - + """Remember the user in the cookie.""" del host # Used for cache key if isinstance(token_response, simple_openid_connect.data.TokenErrorResponse): @@ -297,8 +291,6 @@ def remember( def includeme(config: pyramid.config.Configurator) -> None: - """ - Pyramid includeme function. - """ + """Pyramid includeme function.""" config.add_request_method(get_remember_from_user_info, name="get_remember_from_user_info") config.add_request_method(get_user_from_remember, name="get_user_from_remember") diff --git a/geoportal/c2cgeoportal_geoportal/lib/wmstparsing.py b/geoportal/c2cgeoportal_geoportal/lib/wmstparsing.py index c6d4c1d7f1..6376d63d58 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/wmstparsing.py +++ b/geoportal/c2cgeoportal_geoportal/lib/wmstparsing.py @@ -57,10 +57,10 @@ class TimeInformation: Collect the WMS time information. Arguments: - extent: A time extent instance (``TimeExtentValue`` or ``TimeExtentInterval``) mode: The layer mode ("single", "range" or "disabled") widget: The layer mode ("slider" (default) or "datepicker") + """ def __init__(self) -> None: @@ -98,7 +98,7 @@ def merge_mode(self, mode: str) -> None: self.mode = mode def merge_widget(self, widget: str | None) -> None: - widget = "slider" if not widget else widget + widget = widget if widget else "slider" assert widget is not None if self.widget is not None: @@ -134,11 +134,11 @@ def __init__( Initialize. Arguments: - values: A set() of datetime resolution: The resolution from the mapfile time definition min_def_value: the minimum default value as a datetime max_def_value: the maximum default value as a datetime + """ self.values = values self.resolution = resolution @@ -183,13 +183,13 @@ def __init__( Initialize. Arguments: - start: The start value as a datetime end: The end value as a datetime interval: The interval as a tuple (years, months, days, seconds) resolution: The resolution from the mapfile time definition min_def_value: the minimum default value as a datetime max_def_value: the maximum default value as a datetime + """ self.start = start self.end = end @@ -311,7 +311,7 @@ def _parse_date(date: str) -> tuple[str, datetime.datetime]: try: dt = datetime.datetime.strptime(date, pattern) return resolution, dt.replace(tzinfo=isodate.UTC) - except Exception: # pylint: disable=broad-exception-caught # nosec + except Exception: # pylint: disable=broad-exception-caught pass try: diff --git a/geoportal/c2cgeoportal_geoportal/lib/xsd.py b/geoportal/c2cgeoportal_geoportal/lib/xsd.py index 96611f793e..6c2a8e6104 100644 --- a/geoportal/c2cgeoportal_geoportal/lib/xsd.py +++ b/geoportal/c2cgeoportal_geoportal/lib/xsd.py @@ -42,10 +42,8 @@ def _element_callback(tb: str, column: sqlalchemy.sql.elements.NamedColumn[Any]) -> None: if column.info.get("readonly"): - with tag(tb, "xsd:annotation"): - with tag(tb, "xsd:appinfo"): - with tag(tb, "readonly", {"value": "true"}): - pass + with tag(tb, "xsd:annotation"), tag(tb, "xsd:appinfo"), tag(tb, "readonly", {"value": "true"}): + pass class XSDGenerator(PapyrusXSDGenerator): # type: ignore @@ -105,7 +103,9 @@ def add_column_property_xsd(self, tb: str, column_property: ColumnProperty[Any]) super().add_column_property_xsd(tb, column_property) def add_association_proxy_xsd(self, tb: str, column_property: ColumnProperty[Any]) -> None: - from c2cgeoportal_commons.models import DBSession # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models import ( # pylint: disable=import-outside-toplevel + DBSession, + ) assert DBSession is not None @@ -125,11 +125,13 @@ def add_association_proxy_xsd(self, tb: str, column_property: ColumnProperty[Any attrs["nillable"] = "true" attrs["name"] = proxy with tag(tb, "xsd:element", attrs) as tb2: - with tag(tb2, "xsd:simpleType") as tb3: - with tag(tb3, "xsd:restriction", {"base": "xsd:string"}) as tb4: - for (value,) in query: - with tag(tb4, "xsd:enumeration", {"value": value}): - pass + with ( + tag(tb2, "xsd:simpleType") as tb3, + tag(tb3, "xsd:restriction", {"base": "xsd:string"}) as tb4, + ): + for (value,) in query: + with tag(tb4, "xsd:enumeration", {"value": value}): + pass self.element_callback(tb4, column) diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.prospector.yaml b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.prospector.yaml index 1123c6f3a0..fb7d19ccf0 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.prospector.yaml +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/.prospector.yaml @@ -1,30 +1,3 @@ inherits: - duplicated - - utils:no-design-checks - -max-line-length: 110 - -pycodestyle: - options: - max-line-length: 110 - disable: - - E203 - -pylint: - options: - max-line-length: 110 - disable: - - too-many-locals - - too-many-statements - - too-many-branches - - c-extension-no-member - -pyflakes: - disable: - - F401 - -mccabe: - run: false - -bandit: - run: true + - utils:base-less-strict diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/gunicorn.conf.py b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/gunicorn.conf.py index 4e5dc9fffd..b256426f93 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/gunicorn.conf.py +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/gunicorn.conf.py @@ -47,7 +47,7 @@ max_requests = int(os.environ.get("GUNICORN_MAX_REQUESTS", 1000)) max_requests_jitter = int(os.environ.get("GUNICORN_MAX_REQUESTS_JITTER", 100)) -worker_tmp_dir = "/dev/shm" # nosec +worker_tmp_dir = "/dev/shm" # noqa: S108 limit_request_line = int(os.environ.get("GUNICORN_LIMIT_REQUEST_LINE", 8190)) accesslog = "-" @@ -109,7 +109,6 @@ def on_starting(server: gunicorn.arbiter.Arbiter) -> None: Called just before the master process is initialized. """ - del server prometheus.start() @@ -121,7 +120,6 @@ def post_fork(server: gunicorn.arbiter.Arbiter, worker: gunicorn.workers.base.Wo Called just after a worker has been forked. """ - del server, worker prometheus.cleanup() @@ -133,7 +131,6 @@ def child_exit(server: gunicorn.arbiter.Arbiter, worker: gunicorn.workers.base.W Called just after a worker has been exited, in the master process. """ - del server multiprocess.mark_process_dead(worker.pid) diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/authentication.py b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/authentication.py index b101439948..98cec23448 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/authentication.py +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/authentication.py @@ -1,8 +1,7 @@ +from c2cgeoportal_geoportal.lib.authentication import create_authentication from pyramid.authorization import ACLAuthorizationPolicy from pyramid.config import Configurator -from c2cgeoportal_geoportal.lib.authentication import create_authentication - def includeme(config: Configurator) -> None: """Initialize the authentication( for a Pyramid app.""" diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/models.py b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/models.py index 1882e6726a..18e8462241 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/models.py +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/models.py @@ -1,8 +1,7 @@ import logging -from pyramid.i18n import TranslationStringFactory - from c2cgeoportal_commons.models.main import * # noqa: ignore=F401, pylint: disable=unused-wildcard-import +from pyramid.i18n import TranslationStringFactory _ = TranslationStringFactory("{{cookiecutter.package}}_geoportal-server") _LOG = logging.getLogger(__name__) diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/multi_organization.py b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/multi_organization.py index 25d9664418..2be921b74e 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/multi_organization.py +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/advance_create/{{cookiecutter.project}}/geoportal/{{cookiecutter.package}}_geoportal/multi_organization.py @@ -3,5 +3,4 @@ def includeme(config: Configurator) -> None: """Initialize the multi organization.""" - del config # Unused diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Makefile b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Makefile index ae0343adca..c5deb1f8dc 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Makefile +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/Makefile @@ -28,7 +28,7 @@ checks: prospector eslint ## Runs the checks .PHONY: prospector prospector: ## Runs the Prospector checks docker compose run --entrypoint= --no-deps --rm --volume=$(CURDIR)/geoportal:/app geoportal \ - prospector --output-format=pylint --die-on-tool-error + prospector --without=ruff --output-format=pylint --die-on-tool-error .PHONY: eslint eslint: ## Runs the eslint checks diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-lib.yaml b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-lib.yaml index 9210b816e5..6612b23f2c 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-lib.yaml +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/docker-compose-lib.yaml @@ -192,7 +192,7 @@ services: - redis_slave tilecloudchain: - image: &tilecloudchain-image camptocamp/tilecloud-chain:1.21 + image: &tilecloudchain-image camptocamp/tilecloud-chain:1.22 user: www-data restart: unless-stopped environment: diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/pyproject.toml b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/pyproject.toml index f1c3219848..354842adf7 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/pyproject.toml +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/create/{{cookiecutter.project}}/pyproject.toml @@ -1,7 +1,2 @@ -[tool.black] -line-length = 110 -target-version = ['py39'] - -[tool.isort] -profile = "black" -line_length = 110 +[tool.ruff] +target-version = 'py310' diff --git a/geoportal/c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_create_template/tests/test_testapp.py b/geoportal/c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_create_template/tests/test_testapp.py index 2440797b94..b1d34188d6 100644 --- a/geoportal/c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_create_template/tests/test_testapp.py +++ b/geoportal/c2cgeoportal_geoportal/scaffolds/update/{{cookiecutter.project}}/CONST_create_template/tests/test_testapp.py @@ -41,7 +41,7 @@ def test_desktop_alt(url: str) -> None: def test_enum() -> None: - """Test the enumerations view""" + """Test the enumerations view.""" response = requests.get("https://front/layers/test/values/type", verify=False, timeout=30) # nosec assert response.status_code == 200, response.text diff --git a/geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py b/geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py index f9d2305ca3..de2bf2d6e2 100644 --- a/geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py +++ b/geoportal/c2cgeoportal_geoportal/scripts/c2cupgrade.py @@ -32,13 +32,14 @@ import os import re import shutil -import subprocess +import subprocess # nosec import sys +import tempfile from argparse import ArgumentParser, Namespace from collections.abc import Callable from json.decoder import JSONDecodeError -from subprocess import call, check_call, check_output -from typing import Any, Union, cast +from subprocess import call, check_call, check_output # nosec +from typing import Any, cast import pkg_resources import requests @@ -61,51 +62,10 @@ def fix_style() -> None: """Fix the style of all the project files using isort, Black and Prettier.""" - - file_to_clean = [] - for filename, content in ( - (".prettierignore", "*.min.js\n"), - ("pyproject.toml", "[tool.black]\nline-length = 110\ntarget-version = ['py39']\n"), - (".prettierrc.yaml", "bracketSpacing: false\nquoteProps: preserve\n"), - ( - ".editorconfig", - """root = true -[*] -max_line_length = 110 -""", - ), - ): - if not os.path.exists(filename): - file_to_clean.append(filename) - if os.path.exists(os.path.join("CONST_create_template", filename)): - shutil.copyfile(os.path.join("CONST_create_template", filename), filename) - else: - with open(filename, "w", encoding="utf8") as file_: - file_.write(content) - - if os.path.exists("ci/config.yaml"): - os.rename("ci/config.yaml", "ci/config.yaml_") if os.path.exists(".pre-commit-config.yaml"): print("Run pre-commit to fix the style.") sys.stdout.flush() subprocess.run(["pre-commit", "run", "--all-files"]) # pylint: disable=subprocess-run-check - else: - print("Run c2cciutils-checks to fix the style.") - sys.stdout.flush() - subprocess.run( # pylint: disable=subprocess-run-check - ["c2cciutils-checks", "--fix", "--check=isort"] - ) - subprocess.run( # pylint: disable=subprocess-run-check - ["c2cciutils-checks", "--fix", "--check=black"] - ) - subprocess.run( # pylint: disable=subprocess-run-check - ["c2cciutils-checks", "--fix", "--check=prettier"] - ) - if os.path.exists("ci/config.yaml_"): - os.rename("ci/config.yaml_", "ci/config.yaml") - - for filename in file_to_clean: - os.remove(filename) def main() -> None: @@ -217,7 +177,7 @@ def get_upgrade(section: str) -> list[Any] | dict[str, Any]: sys.exit(1) with open(".upgrade.yaml", encoding="utf8") as project_file: - return cast(Union[list[Any], dict[str, Any]], yaml.safe_load(project_file)[section]) + return cast(list[Any] | dict[str, Any], yaml.safe_load(project_file)[section]) def print_step( self, @@ -254,7 +214,7 @@ def test_checkers(self) -> tuple[bool, str | None]: resp = requests.get( self.project["checker_url"], headers=self.project.get("checker_headers"), - verify=False, # nosec + verify=False, # noqa: S501 timeout=120, ) except requests.exceptions.ConnectionError as exception: @@ -324,12 +284,13 @@ def step0(self, step: int) -> None: @Step(1) def step1(self, step: int) -> None: - shutil.copyfile("project.yaml", "/tmp/project.yaml") - try: - check_call(["git", "reset", "--hard"]) - check_call(["git", "clean", "--force", "-d"]) - finally: - shutil.copyfile("/tmp/project.yaml", "project.yaml") + with tempfile.NamedTemporaryFile("w") as project_temp_file: + shutil.copyfile("project.yaml", project_temp_file.name) + try: + check_call(["git", "reset", "--hard"]) + check_call(["git", "clean", "--force", "-d"]) + finally: + shutil.copyfile(project_temp_file.name, "project.yaml") self.run_step(step + 1) @@ -343,36 +304,36 @@ def step2(self, step: int) -> None: @Step(3) def step3(self, step: int) -> None: - project_path = os.path.join("/tmp", self.project["project_folder"]) - os.mkdir(project_path) - shutil.copyfile("/src/project.yaml", os.path.join(project_path, "project.yaml")) - check_call( - [ - "pcreate", - "--overwrite", - "--scaffold=update", - project_path, - ] - ) - if self.get_project().get("advance", False): + with tempfile.TemporaryDirectory() as temp_directory_name: + project_path = os.path.join(temp_directory_name, self.project["project_folder"]) + os.mkdir(project_path) + shutil.copyfile("/src/project.yaml", os.path.join(project_path, "project.yaml")) check_call( [ "pcreate", "--overwrite", - "--scaffold=advance_update", + "--scaffold=update", project_path, ] ) + if self.get_project().get("advance", False): + check_call( + [ + "pcreate", + "--overwrite", + "--scaffold=advance_update", + project_path, + ] + ) - shutil.copyfile(os.path.join(project_path, ".upgrade.yaml"), ".upgrade.yaml") - for upgrade_file in cast(list[dict[str, Any]], self.get_upgrade("upgrade_files")): - action = upgrade_file["action"] - if action == "remove": - self.files_to_remove(upgrade_file, prefix="CONST_create_template", force=True) - if action == "move": - self.files_to_move(upgrade_file, prefix="CONST_create_template", force=True) + shutil.copyfile(os.path.join(project_path, ".upgrade.yaml"), ".upgrade.yaml") + for upgrade_file in cast(list[dict[str, Any]], self.get_upgrade("upgrade_files")): + action = upgrade_file["action"] + if action == "remove": + self.files_to_remove(upgrade_file, prefix="CONST_create_template", force=True) + if action == "move": + self.files_to_move(upgrade_file, prefix="CONST_create_template", force=True) - shutil.rmtree(project_path) os.remove(".upgrade.yaml") check_call(["git", "add", "--all", "--force", "CONST_create_template/"]) @@ -385,26 +346,26 @@ def step4(self, step: int) -> None: if os.path.exists("CONST_create_template"): check_call(["git", "rm", "-r", "--force", "CONST_create_template/"]) - project_path = os.path.join("/tmp", self.project["project_folder"]) - check_call(["ln", "-s", "/src", project_path]) - check_call( - [ - "pcreate", - "--overwrite", - "--scaffold=update", - project_path, - ] - ) - if self.get_project().get("advance", False): + with tempfile.TemporaryDirectory() as temp_directory_name: + project_path = os.path.join(temp_directory_name, self.project["project_folder"]) + check_call(["ln", "-s", "/src", project_path]) check_call( [ "pcreate", "--overwrite", - "--scaffold=advance_update", + "--scaffold=update", project_path, ] ) - os.remove(project_path) + if self.get_project().get("advance", False): + check_call( + [ + "pcreate", + "--overwrite", + "--scaffold=advance_update", + project_path, + ] + ) check_call(["git", "add", "--all", "CONST_create_template/"]) @@ -644,10 +605,7 @@ def is_managed(self, file_: str, files_to_get: bool = False) -> bool: if not managed: for files in self.project["managed_files"]: - if isinstance(files, str): - pattern = files - else: - pattern = files["pattern"] + pattern = files if isinstance(files, str) else files["pattern"] if re.match(pattern + "$", file_): print(f"File '{file_}' included by project config pattern `managed_files` '{pattern}'.") print("managed", file_, pattern) diff --git a/geoportal/c2cgeoportal_geoportal/scripts/manage_users.py b/geoportal/c2cgeoportal_geoportal/scripts/manage_users.py index 9890e2ed82..de184a6b93 100644 --- a/geoportal/c2cgeoportal_geoportal/scripts/manage_users.py +++ b/geoportal/c2cgeoportal_geoportal/scripts/manage_users.py @@ -37,7 +37,6 @@ def get_argparser() -> argparse.ArgumentParser: """Get the argument parser for this script.""" - usage = """Reset a user password. The username is used as password if the password is not provided with the corresponding option. User can be created if it does not exist yet.""" @@ -66,7 +65,6 @@ def main() -> None: to get the options list, do: docker compose exec geoportal manage-users --help """ - parser = get_argparser() options = parser.parse_args() username = options.user @@ -76,8 +74,12 @@ def main() -> None: session = get_session(settings, transaction.manager) # Must be done only once we have loaded the project config - from c2cgeoportal_commons.models.main import Role # pylint: disable=import-outside-toplevel - from c2cgeoportal_commons.models.static import User # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Role, + ) + from c2cgeoportal_commons.models.static import ( # pylint: disable=import-outside-toplevel + User, + ) print("\n") diff --git a/geoportal/c2cgeoportal_geoportal/scripts/pcreate.py b/geoportal/c2cgeoportal_geoportal/scripts/pcreate.py index c292b002d9..978475a448 100644 --- a/geoportal/c2cgeoportal_geoportal/scripts/pcreate.py +++ b/geoportal/c2cgeoportal_geoportal/scripts/pcreate.py @@ -29,10 +29,10 @@ import json import os import re -import subprocess +import subprocess # nosec import sys from argparse import ArgumentParser -from typing import Any, Union, cast +from typing import Any, cast import pkg_resources import requests @@ -46,7 +46,6 @@ def get_argparser() -> ArgumentParser: """Get the argument parser for this script.""" - parser = ArgumentParser( prog=sys.argv[0], add_help=True, @@ -183,9 +182,7 @@ def get_context(self) -> dict[str, str | int]: } context.update(self.read_project_file()) if os.environ.get("CI") == "true": - context["authtkt_secret"] = ( # nosec - "io7heoDui8xaikie1rushaeGeiph8Bequei6ohchaequob6viejei0xooWeuvohf" - ) + context["authtkt_secret"] = "io7heoDui8xaikie1rushaeGeiph8Bequei6ohchaequob6viejei0xooWeuvohf" # noqa: S105 self.get_var(context, "srid", "Spatial Reference System Identifier (e.g. 2056): ", int) srid = cast(int, context["srid"]) @@ -246,7 +243,7 @@ def read_project_file(self) -> dict[str, str | int]: if os.path.exists(project_file): with open(project_file, encoding="utf8") as f: project = yaml.safe_load(f) - return cast(dict[str, Union[str, int]], project.get("template_vars", {})) + return cast(dict[str, str | int], project.get("template_vars", {})) else: return {} @@ -280,16 +277,12 @@ def epsg2bbox(srid: int) -> list[str] | None: r = requests.get(f"https://epsg.io/?format=json&q={srid}", timeout=60) bbox = r.json()["results"][0]["bbox"] r = requests.get( - "https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[1]},{bbox[0]}".format( - srid=srid, bbox=bbox - ), + f"https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[1]},{bbox[0]}", timeout=60, ) r1 = r.json()[0] r = requests.get( - "https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[3]},{bbox[2]}".format( - srid=srid, bbox=bbox - ), + f"https://epsg.io/trans?s_srs=4326&t_srs={srid}&data={bbox[3]},{bbox[2]}", timeout=60, ) r2 = r.json()[0] diff --git a/geoportal/c2cgeoportal_geoportal/scripts/theme2fts.py b/geoportal/c2cgeoportal_geoportal/scripts/theme2fts.py index 3256786eff..04c8951f99 100644 --- a/geoportal/c2cgeoportal_geoportal/scripts/theme2fts.py +++ b/geoportal/c2cgeoportal_geoportal/scripts/theme2fts.py @@ -1,4 +1,4 @@ -# Copyright (c) 2014-2023, Camptocamp SA +# Copyright (c) 2014-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -95,7 +95,6 @@ def get_argparser() -> ArgumentParser: def main() -> None: """Run the command.""" - options = get_argparser().parse_args() settings = get_appsettings(options) @@ -180,7 +179,9 @@ def _add_fts( action: str, role: Optional["c2cgeoportal_commons.models.main.Role"], ) -> None: - from c2cgeoportal_commons.models.main import FullTextSearch # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + FullTextSearch, + ) key = ( item.name if self.options.name else item.id, @@ -248,7 +249,9 @@ def _add_group( export: bool, role: Optional["c2cgeoportal_commons.models.main.Role"], ) -> bool: - from c2cgeoportal_commons.models.main import LayerGroup # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + LayerGroup, + ) fill = False for child in group.children: diff --git a/geoportal/c2cgeoportal_geoportal/scripts/urllogin.py b/geoportal/c2cgeoportal_geoportal/scripts/urllogin.py index bedf0e6153..a1fb840214 100644 --- a/geoportal/c2cgeoportal_geoportal/scripts/urllogin.py +++ b/geoportal/c2cgeoportal_geoportal/scripts/urllogin.py @@ -56,7 +56,6 @@ def create_token(aeskey: str, user: str, password: str, valid: bool) -> str: def get_argparser() -> argparse.ArgumentParser: """Get the argument parser for this script.""" - parser = argparse.ArgumentParser(description="Generate an auth token") fill_arguments(parser, use_attribute=True) parser.add_argument("user", help="The username") @@ -67,7 +66,6 @@ def get_argparser() -> argparse.ArgumentParser: def main() -> None: """Run the command.""" - args = get_argparser().parse_args() settings = get_appsettings(args) urllogin = settings.get("urllogin", {}) diff --git a/geoportal/c2cgeoportal_geoportal/views/dev.py b/geoportal/c2cgeoportal_geoportal/views/dev.py index dc11dad84f..d61f78bef7 100644 --- a/geoportal/c2cgeoportal_geoportal/views/dev.py +++ b/geoportal/c2cgeoportal_geoportal/views/dev.py @@ -1,4 +1,4 @@ -# Copyright (c) 2011-2021, Camptocamp SA +# Copyright (c) 2011-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,10 +31,10 @@ import pyramid.request import pyramid.response +from c2cgeoportal_commons.lib.url import Url from pyramid.httpexceptions import HTTPFound from pyramid.view import view_config -from c2cgeoportal_commons.lib.url import Url from c2cgeoportal_geoportal.views.proxy import Proxy logger = logging.getLogger(__name__) diff --git a/geoportal/c2cgeoportal_geoportal/views/dynamic.py b/geoportal/c2cgeoportal_geoportal/views/dynamic.py index 7441ab71ab..b12813a7a2 100644 --- a/geoportal/c2cgeoportal_geoportal/views/dynamic.py +++ b/geoportal/c2cgeoportal_geoportal/views/dynamic.py @@ -31,12 +31,12 @@ from typing import Any, cast import pyramid.request +from c2cgeoportal_commons import models +from c2cgeoportal_commons.models import main from pyramid.httpexceptions import HTTPNotFound from pyramid.view import view_config from sqlalchemy import func -from c2cgeoportal_commons import models -from c2cgeoportal_commons.models import main from c2cgeoportal_geoportal import is_allowed_host from c2cgeoportal_geoportal.lib.cacheversion import get_cache_version from c2cgeoportal_geoportal.lib.caching import get_region @@ -78,13 +78,12 @@ def _interface( Get the interface configuration. Arguments: - interface_config: Current interface configuration interface_name: Interface name (we use in the configuration) original_interface_name: Original interface name (directly for the query string) dynamic: The values that's dynamically generated - """ + """ if "extends" in interface_config: constants = self._interface( self.interfaces_config[interface_config["extends"]], diff --git a/geoportal/c2cgeoportal_geoportal/views/entry.py b/geoportal/c2cgeoportal_geoportal/views/entry.py index ec566170c3..444008d203 100644 --- a/geoportal/c2cgeoportal_geoportal/views/entry.py +++ b/geoportal/c2cgeoportal_geoportal/views/entry.py @@ -114,7 +114,6 @@ def _get_ngeo_resources(pattern: str) -> list[str]: def canvas_view(request: pyramid.request.Request, interface_config: dict[str, Any]) -> dict[str, Any]: """Get view used as entry point of a canvas interface.""" - js_files = _get_ngeo_resources(f"{interface_config.get('layout', interface_config['name'])}*.js") css_files = _get_ngeo_resources(f"{interface_config.get('layout', interface_config['name'])}*.css") css = "\n ".join( @@ -149,7 +148,6 @@ def custom_view( request: pyramid.request.Request, interface_config: dict[str, Any] ) -> pyramid.response.Response: """Get view used as entry point of a canvas interface.""" - set_common_headers(request, "index", Cache.PUBLIC_NO, content_type="text/html") html_filename = interface_config.get("html_filename", f"{interface_config['name']}.html") diff --git a/geoportal/c2cgeoportal_geoportal/views/fulltextsearch.py b/geoportal/c2cgeoportal_geoportal/views/fulltextsearch.py index d2f1f8cbd1..ee041b8e21 100644 --- a/geoportal/c2cgeoportal_geoportal/views/fulltextsearch.py +++ b/geoportal/c2cgeoportal_geoportal/views/fulltextsearch.py @@ -29,14 +29,14 @@ import re import pyramid.request +from c2cgeoportal_commons.models import DBSession +from c2cgeoportal_commons.models.main import FullTextSearch, Interface from geoalchemy2.shape import to_shape from geojson import Feature, FeatureCollection from pyramid.httpexceptions import HTTPBadRequest, HTTPInternalServerError from pyramid.view import view_config from sqlalchemy import ColumnElement, and_, desc, func, or_ -from c2cgeoportal_commons.models import DBSession -from c2cgeoportal_commons.models.main import FullTextSearch, Interface from c2cgeoportal_geoportal import locale_negotiator from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers diff --git a/geoportal/c2cgeoportal_geoportal/views/geometry_processing.py b/geoportal/c2cgeoportal_geoportal/views/geometry_processing.py index 47f66abdc7..de7620570d 100644 --- a/geoportal/c2cgeoportal_geoportal/views/geometry_processing.py +++ b/geoportal/c2cgeoportal_geoportal/views/geometry_processing.py @@ -27,6 +27,7 @@ import pyramid.request +from c2cgeoportal_commons.models import DBSession from geoalchemy2.shape import from_shape, to_shape from geojson import loads from pyramid.httpexceptions import HTTPBadRequest @@ -35,8 +36,6 @@ from shapely.geometry.base import BaseGeometry from sqlalchemy import func -from c2cgeoportal_commons.models import DBSession - class GeometryProcessing: """ diff --git a/geoportal/c2cgeoportal_geoportal/views/i18n.py b/geoportal/c2cgeoportal_geoportal/views/i18n.py index aa21d76eed..5e7cdb1e78 100644 --- a/geoportal/c2cgeoportal_geoportal/views/i18n.py +++ b/geoportal/c2cgeoportal_geoportal/views/i18n.py @@ -68,7 +68,6 @@ def locale(request: pyramid.request.Request) -> pyramid.response.Response: @view_config(route_name="localepot") # type: ignore def localepot(request: pyramid.request.Request) -> pyramid.response.Response: """Get the pot from an HTTP request.""" - # Build the list of files to be processed sources = [] sources += glob.glob(f"/app/{request.registry.package_name}/static-ngeo/js/apps/*.html.ejs") diff --git a/geoportal/c2cgeoportal_geoportal/views/layers.py b/geoportal/c2cgeoportal_geoportal/views/layers.py index 491d2a8e14..13328badbe 100644 --- a/geoportal/c2cgeoportal_geoportal/views/layers.py +++ b/geoportal/c2cgeoportal_geoportal/views/layers.py @@ -40,6 +40,7 @@ import sqlalchemy.ext.declarative import sqlalchemy.orm import sqlalchemy.orm.query +from c2cgeoportal_commons import models from geoalchemy2 import Geometry from geoalchemy2.shape import from_shape, to_shape from geojson.feature import Feature, FeatureCollection @@ -56,19 +57,27 @@ from shapely import unary_union from shapely.errors import TopologicalError from sqlalchemy import Enum, Numeric, String, Text, Unicode, UnicodeText, exc, func -from sqlalchemy.orm.exc import MultipleResultsFound, NoResultFound # type: ignore[attr-defined] +from sqlalchemy.orm.exc import ( # type: ignore[attr-defined] + MultipleResultsFound, + NoResultFound, +) from sqlalchemy.orm.properties import ColumnProperty from sqlalchemy.orm.util import class_mapper from sqlalchemy.sql import and_, or_ -from c2cgeoportal_commons import models from c2cgeoportal_geoportal.lib import get_roles_id from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers -from c2cgeoportal_geoportal.lib.dbreflection import _AssociationProxy, get_class, get_table +from c2cgeoportal_geoportal.lib.dbreflection import ( + _AssociationProxy, + get_class, + get_table, +) if TYPE_CHECKING: - from c2cgeoportal_commons.models import main # pylint: disable=ungrouped-imports.useless-suppression + from c2cgeoportal_commons.models import ( + main, # pylint: disable=ungrouped-imports.useless-suppression + ) _LOG = logging.getLogger(__name__) @@ -222,7 +231,9 @@ def _get_geom_col_info(layer: "main.Layer") -> tuple[str, int]: @staticmethod def _get_layer(layer_id: int) -> "main.Layer": """Return a ``Layer`` object for ``layer_id``.""" - from c2cgeoportal_commons.models.main import Layer # pylint: disable=import-outside-toplevel + from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel + Layer, + ) assert models.DBSession is not None @@ -281,7 +292,6 @@ def _get_protocol_for_request(self) -> Protocol: def _proto_read(self, layer: "main.Layer") -> FeatureCollection: """Read features for the layer based on the self.request.""" - from c2cgeoportal_commons.models.main import ( # pylint: disable=import-outside-toplevel Layer, RestrictionArea, @@ -495,7 +505,7 @@ def enumerate_attribute_values(self) -> dict[str, Any]: raise HTTPInternalServerError("Missing configuration") layername = self.request.matchdict["layer_name"] fieldname = self.request.matchdict["field_name"] - # TODO check if layer is public or not + # TODO check if layer is public or not # pylint: disable=fixme return cast(dict[str, Any], self._enumerate_attribute_values(layername, fieldname)) @@ -541,15 +551,14 @@ def get_layer_class(layer: "main.Layer", with_last_update_columns: bool = False) Get the SQLAlchemy class to edit a GeoMapFish layer. Keyword Arguments: - layer: The GeoMapFish layer with_last_update_columns: False to just have a class to access to the table and be able to modify the last_update_columns, True to have a correct class to build the UI (without the hidden column). Returns: SQLAlchemy class - """ + """ assert layer.geo_table is not None # Exclude the columns used to record the last features update @@ -616,7 +625,6 @@ class ColumnProperties(TypedDict, total=False): def get_layer_metadata(layer: "main.Layer") -> list[ColumnProperties]: """Get the metadata related to a layer.""" - assert models.DBSession is not None cls = get_layer_class(layer, with_last_update_columns=True) @@ -690,7 +698,7 @@ def _convert_column_type(column_type: object) -> ColumnProperties: return restriction # String type - if isinstance(column_type, (String, Text, Unicode, UnicodeText)): + if isinstance(column_type, String | Text | Unicode | UnicodeText): if column_type.length is None: return {"type": "xsd:string"} return {"type": "xsd:string", "maxLength": int(column_type.length)} diff --git a/geoportal/c2cgeoportal_geoportal/views/login.py b/geoportal/c2cgeoportal_geoportal/views/login.py index 2d812a3c94..fd061bf6a0 100644 --- a/geoportal/c2cgeoportal_geoportal/views/login.py +++ b/geoportal/c2cgeoportal_geoportal/views/login.py @@ -38,6 +38,9 @@ import pyotp import pyramid.request import pyramid.response +from c2cgeoportal_commons import models +from c2cgeoportal_commons.lib.email_ import send_email_config +from c2cgeoportal_commons.models import static from pyramid.httpexceptions import ( HTTPBadRequest, HTTPForbidden, @@ -50,9 +53,6 @@ from pyramid.view import forbidden_view_config, view_config from sqlalchemy.orm.exc import NoResultFound # type: ignore[attr-defined] -from c2cgeoportal_commons import models -from c2cgeoportal_commons.lib.email_ import send_email_config -from c2cgeoportal_commons.models import static from c2cgeoportal_geoportal import is_allowed_url, is_valid_referrer from c2cgeoportal_geoportal.lib import get_setting, is_intranet, oauth2, oidc from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers @@ -126,9 +126,7 @@ def loginform(self) -> dict[str, Any]: @staticmethod def _validate_2fa_totp(user: static.User, otp: str) -> bool: - if pyotp.TOTP(user.tech_data.get("2fa_totp_secret", "")).verify(otp): - return True - return False + return bool(pyotp.TOTP(user.tech_data.get("2fa_totp_secret", "")).verify(otp)) @view_config(route_name="login") # type: ignore def login(self) -> pyramid.response.Response: @@ -374,10 +372,9 @@ def change_password(self) -> pyramid.response.Response: _LOG.info("The login '%s' does not exist.", login) raise HTTPUnauthorized("See server logs for details") - if self.two_factor_auth: - if not self._validate_2fa_totp(user, otp): - _LOG.info("The second factor is wrong for user '%s'.", login) - raise HTTPUnauthorized("See server logs for details") + if self.two_factor_auth and not self._validate_2fa_totp(user, otp): + _LOG.info("The second factor is wrong for user '%s'.", login) + raise HTTPUnauthorized("See server logs for details") else: user = self.request.user @@ -662,7 +659,7 @@ def oidc_callback(self) -> pyramid.response.Response: "login", Cache.PRIVATE_NO, response=Response( - # TODO respect the user interface... + # TODO respect the user interface... # pylint: disable=fixme json.dumps( { "username": user.display_name, diff --git a/geoportal/c2cgeoportal_geoportal/views/mapserverproxy.py b/geoportal/c2cgeoportal_geoportal/views/mapserverproxy.py index 474530112f..3e9875a416 100644 --- a/geoportal/c2cgeoportal_geoportal/views/mapserverproxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/mapserverproxy.py @@ -29,13 +29,18 @@ import logging from typing import Any -from pyramid.httpexceptions import HTTPForbidden, HTTPFound, HTTPInternalServerError, HTTPUnauthorized +from c2cgeoportal_commons.lib.url import Url +from c2cgeoportal_commons.models import main +from pyramid.httpexceptions import ( + HTTPForbidden, + HTTPFound, + HTTPInternalServerError, + HTTPUnauthorized, +) from pyramid.request import Request from pyramid.response import Response from pyramid.view import view_config -from c2cgeoportal_commons.lib.url import Url -from c2cgeoportal_commons.models import main from c2cgeoportal_geoportal.lib import get_roles_id, get_roles_name from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache diff --git a/geoportal/c2cgeoportal_geoportal/views/ogcproxy.py b/geoportal/c2cgeoportal_geoportal/views/ogcproxy.py index ca0298e3de..dd1d12c2ca 100644 --- a/geoportal/c2cgeoportal_geoportal/views/ogcproxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/ogcproxy.py @@ -28,11 +28,11 @@ import logging import pyramid.request +from c2cgeoportal_commons.lib.url import Url, get_url2 +from c2cgeoportal_commons.models import DBSession, main from pyramid.httpexceptions import HTTPBadRequest from sqlalchemy.orm.exc import NoResultFound # type: ignore[attr-defined] -from c2cgeoportal_commons.lib.url import Url, get_url2 -from c2cgeoportal_commons.models import DBSession, main from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.views.proxy import Proxy diff --git a/geoportal/c2cgeoportal_geoportal/views/pdfreport.py b/geoportal/c2cgeoportal_geoportal/views/pdfreport.py index 4342daf0cd..3d5d2b9505 100644 --- a/geoportal/c2cgeoportal_geoportal/views/pdfreport.py +++ b/geoportal/c2cgeoportal_geoportal/views/pdfreport.py @@ -31,12 +31,12 @@ import pyramid.request import pyramid.response -from pyramid.httpexceptions import HTTPBadRequest, HTTPForbidden -from pyramid.view import view_config - from c2cgeoportal_commons import models from c2cgeoportal_commons.lib.url import Url from c2cgeoportal_commons.models import main +from pyramid.httpexceptions import HTTPBadRequest, HTTPForbidden +from pyramid.view import view_config + from c2cgeoportal_geoportal.lib.common_headers import Cache from c2cgeoportal_geoportal.lib.layers import get_private_layers, get_protected_layers from c2cgeoportal_geoportal.views.ogcproxy import OGCProxy @@ -121,7 +121,7 @@ def get_report(self) -> pyramid.response.Response: ) if layer_config["check_credentials"]: - # FIXME: support of mapserver groups + # FIXME: support of mapserver groups # pylint: disable=fixme ogc_server = ( models.DBSession.query(main.OGCServer) .filter(main.OGCServer.name == layer_config["ogc_server"]) diff --git a/geoportal/c2cgeoportal_geoportal/views/printproxy.py b/geoportal/c2cgeoportal_geoportal/views/printproxy.py index a46f2736f8..2cd8bc1384 100644 --- a/geoportal/c2cgeoportal_geoportal/views/printproxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/printproxy.py @@ -33,10 +33,10 @@ import pyramid.request import pyramid.response import requests +from c2cgeoportal_commons.lib.url import Url from pyramid.httpexceptions import HTTPBadGateway, HTTPFound from pyramid.view import view_config -from c2cgeoportal_commons.lib.url import Url from c2cgeoportal_geoportal.lib import is_intranet from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache diff --git a/geoportal/c2cgeoportal_geoportal/views/proxy.py b/geoportal/c2cgeoportal_geoportal/views/proxy.py index f18a2c8c80..427308ebcf 100644 --- a/geoportal/c2cgeoportal_geoportal/views/proxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/proxy.py @@ -33,9 +33,9 @@ import pyramid.request import pyramid.response import requests +from c2cgeoportal_commons.lib.url import Url from pyramid.httpexceptions import HTTPBadGateway, exception_response -from c2cgeoportal_commons.lib.url import Url from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers from c2cgeoportal_geoportal.views import restrict_headers diff --git a/geoportal/c2cgeoportal_geoportal/views/raster.py b/geoportal/c2cgeoportal_geoportal/views/raster.py index 5f476cc2b6..5c742dcf95 100644 --- a/geoportal/c2cgeoportal_geoportal/views/raster.py +++ b/geoportal/c2cgeoportal_geoportal/views/raster.py @@ -36,11 +36,11 @@ import numpy import pyramid.request import zope.event.classhandler +from c2cgeoportal_commons.models import InvalidateCacheEvent from pyramid.httpexceptions import HTTPBadRequest, HTTPNotFound from pyramid.view import view_config from rasterio.io import DatasetReader -from c2cgeoportal_commons.models import InvalidateCacheEvent from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers if TYPE_CHECKING: @@ -108,7 +108,9 @@ def _get_data(self, layer: dict[str, Any], name: str) -> "fiona.collection.Colle path = layer["file"] if layer.get("type", "shp_index") == "shp_index": # Avoid loading if not needed - from fiona.collection import Collection # pylint: disable=import-outside-toplevel + from fiona.collection import ( # pylint: disable=import-outside-toplevel + Collection, + ) self.data[name] = Collection(path) elif layer.get("type") == "gdal": diff --git a/geoportal/c2cgeoportal_geoportal/views/resourceproxy.py b/geoportal/c2cgeoportal_geoportal/views/resourceproxy.py index 3ccef3c067..460c9423d1 100644 --- a/geoportal/c2cgeoportal_geoportal/views/resourceproxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/resourceproxy.py @@ -65,7 +65,7 @@ def proxy(self) -> pyramid.response.Response: response = self._build_response( response, response.content, cache_control, "externalresource", content_type=content_type ) - for header in response.headers.keys(): + for header in response.headers: if header not in self.settings["allowed_headers"]: response.headers.pop(header) return response diff --git a/geoportal/c2cgeoportal_geoportal/views/shortener.py b/geoportal/c2cgeoportal_geoportal/views/shortener.py index 2480734a62..ba2aaf2d43 100644 --- a/geoportal/c2cgeoportal_geoportal/views/shortener.py +++ b/geoportal/c2cgeoportal_geoportal/views/shortener.py @@ -34,11 +34,16 @@ from urllib.parse import urlparse import pyramid.request -from pyramid.httpexceptions import HTTPBadRequest, HTTPFound, HTTPInternalServerError, HTTPNotFound -from pyramid.view import view_config - from c2cgeoportal_commons.lib.email_ import send_email_config from c2cgeoportal_commons.models import DBSession, static +from pyramid.httpexceptions import ( + HTTPBadRequest, + HTTPFound, + HTTPInternalServerError, + HTTPNotFound, +) +from pyramid.view import view_config + from c2cgeoportal_geoportal import is_allowed_url from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers @@ -107,7 +112,7 @@ def create(self) -> dict[str, str]: tries = 0 while not shortened: ref = "".join( - random.choice(string.ascii_letters + string.digits) # nosec + random.choice(string.ascii_letters + string.digits) # noqa: S311 for i in range(self.settings.get("length", 4)) ) test_url = DBSession.query(static.Shorturl).filter(static.Shorturl.ref == ref).all() diff --git a/geoportal/c2cgeoportal_geoportal/views/theme.py b/geoportal/c2cgeoportal_geoportal/views/theme.py index 42f2b2c5fb..754390cd95 100644 --- a/geoportal/c2cgeoportal_geoportal/views/theme.py +++ b/geoportal/c2cgeoportal_geoportal/views/theme.py @@ -33,9 +33,11 @@ import re import sys import time +import xml.etree +import xml.etree.ElementTree # nosec from collections import Counter from math import sqrt -from typing import Any, Optional, Union, cast +from typing import Any, cast import dogpile.cache.api import pyramid.httpexceptions @@ -43,19 +45,23 @@ import requests import sqlalchemy import sqlalchemy.orm.query +from c2cgeoportal_commons import models +from c2cgeoportal_commons.lib.url import Url, get_url2 +from c2cgeoportal_commons.models import cache_invalidate_cb, main from c2cwsgiutils.auth import auth_view from defusedxml import lxml -from lxml import etree # nosec from owslib.wms import WebMapService from pyramid.view import view_config from sqlalchemy.orm import subqueryload from sqlalchemy.orm.exc import NoResultFound # type: ignore[attr-defined] -from c2cgeoportal_commons import models -from c2cgeoportal_commons.lib.url import Url, get_url2 -from c2cgeoportal_commons.models import cache_invalidate_cb, main from c2cgeoportal_geoportal import is_allowed_host, is_allowed_url -from c2cgeoportal_geoportal.lib import get_roles_id, get_typed, get_types_map, is_intranet +from c2cgeoportal_geoportal.lib import ( + get_roles_id, + get_typed, + get_types_map, + is_intranet, +) from c2cgeoportal_geoportal.lib.caching import get_region from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers from c2cgeoportal_geoportal.lib.functionality import get_mapserver_substitution_params @@ -72,7 +78,7 @@ _CACHE_OGC_SERVER_REGION = get_region("ogc-server") _TIMEOUT = int(os.environ.get("C2CGEOPORTAL_THEME_TIMEOUT", "300")) -Metadata = Union[str, int, float, bool, list[Any], dict[str, Any]] +Metadata = str | int | float | bool | list[Any] | dict[str, Any] async def get_http_cached( @@ -323,7 +329,6 @@ async def _wms_getcap_cached( def _create_layer_query(self, interface: str) -> sqlalchemy.orm.query.RowReturningQuery[tuple[str]]: """Create an SQLAlchemy query for Layer and for the role identified to by ``role_id``.""" - assert models.DBSession is not None query: sqlalchemy.orm.query.RowReturningQuery[tuple[str]] = models.DBSession.query( @@ -596,7 +601,6 @@ def _layer_included(tree_item: main.TreeItem) -> bool: def _get_ogc_servers(self, group: main.LayerGroup, depth: int) -> set[str | bool]: """Get unique identifier for each child by recursing on all the children.""" - ogc_servers: set[str | bool] = set() # escape loop @@ -713,15 +717,14 @@ async def _group( ) group_theme["mixed"] = mixed - if org_depth == 1: - if not mixed: - assert time_ is not None - assert dim is not None - group_theme["ogcServer"] = cast(list[Any], ogc_servers)[0] - if time_.has_time() and time_.layer is None: - group_theme["time"] = time_.to_dict() + if org_depth == 1 and not mixed: + assert time_ is not None + assert dim is not None + group_theme["ogcServer"] = cast(list[Any], ogc_servers)[0] + if time_.has_time() and time_.layer is None: + group_theme["time"] = time_.to_dict() - group_theme["dimensions"] = dim.get_dimensions() + group_theme["dimensions"] = dim.get_dimensions() return group_theme, errors return None, errors @@ -775,7 +778,6 @@ async def _themes( self, interface: str = "desktop", filter_themes: bool = True, min_levels: int = 1 ) -> tuple[list[dict[str, Any]], set[str]]: """Return theme information for the role identified by ``role_id``.""" - assert models.DBSession is not None self._load_tree_items() @@ -894,7 +896,7 @@ def _get_role_ids(self) -> set[int] | None: async def _wfs_get_features_type( self, wfs_url: Url, ogc_server: main.OGCServer, preload: bool = False, cache: bool = True - ) -> tuple[Optional[etree.Element], set[str]]: # pylint: disable=c-extension-no-member + ) -> tuple[xml.etree.ElementTree.Element | None, set[str]]: # pylint: disable=c-extension-no-member errors = set() wfs_url.add_query( @@ -987,7 +989,8 @@ async def _preload(self, errors: set[str]) -> None: for ogc_server, nb_layers in ( models.DBSession.query( - main.OGCServer, sqlalchemy.func.count(main.LayerWMS.id) # pylint: disable=not-callable + main.OGCServer, + sqlalchemy.func.count(main.LayerWMS.id), # pylint: disable=not-callable ) .filter(main.LayerWMS.ogc_server_id == main.OGCServer.id) .group_by(main.OGCServer.id) @@ -1037,7 +1040,7 @@ def _get_features_attributes_cache( "available namespaces: %s (OGC server: %s)", type_namespace, name, - ", ".join([str(k) for k in child.nsmap.keys()]), + ", ".join([str(k) for k in child.nsmap]), ogc_server_name, ) elif child.nsmap[type_namespace] != namespace: @@ -1069,7 +1072,7 @@ def _get_features_attributes_cache( "available namespaces: %s (OGC server: %s)", type_namespace, name, - ", ".join([str(k) for k in child.nsmap.keys()]), + ", ".join([str(k) for k in child.nsmap]), ogc_server_name, ) for key, value in children.attrib.items(): @@ -1114,7 +1117,6 @@ def _get_features_attributes_cache( @view_config(route_name="themes", renderer="json") # type: ignore[misc] def themes(self) -> dict[str, dict[str, dict[str, Any]] | list[str]]: - is_allowed_host(self.request) interface = self.request.params.get("interface", "desktop") @@ -1145,7 +1147,8 @@ async def get_theme() -> dict[str, dict[str, Any] | list[str]]: result["ogcServers"] = {} for ogc_server, nb_layers in ( models.DBSession.query( - main.OGCServer, sqlalchemy.func.count(main.LayerWMS.id) # pylint: disable=not-callable + main.OGCServer, + sqlalchemy.func.count(main.LayerWMS.id), # pylint: disable=not-callable ) .filter(main.LayerWMS.ogc_server_id == main.OGCServer.id) .group_by(main.OGCServer.id) @@ -1239,7 +1242,7 @@ def get_theme_anonymous( if self.request.user is None: return cast( - dict[str, Union[dict[str, dict[str, Any]], list[str]]], + dict[str, dict[str, dict[str, Any]] | list[str]], get_theme_anonymous( is_intranet(self.request), interface, diff --git a/geoportal/c2cgeoportal_geoportal/views/tinyowsproxy.py b/geoportal/c2cgeoportal_geoportal/views/tinyowsproxy.py index 9edd8efb1f..c33ca1c3c2 100644 --- a/geoportal/c2cgeoportal_geoportal/views/tinyowsproxy.py +++ b/geoportal/c2cgeoportal_geoportal/views/tinyowsproxy.py @@ -30,13 +30,18 @@ from typing import Any import pyramid.request -from defusedxml import ElementTree -from pyramid.httpexceptions import HTTPBadRequest, HTTPForbidden, HTTPInternalServerError, HTTPUnauthorized -from pyramid.view import view_config - from c2cgeoportal_commons import models from c2cgeoportal_commons.lib.url import Url from c2cgeoportal_commons.models import main +from defusedxml import ElementTree +from pyramid.httpexceptions import ( + HTTPBadRequest, + HTTPForbidden, + HTTPInternalServerError, + HTTPUnauthorized, +) +from pyramid.view import view_config + from c2cgeoportal_geoportal.lib.common_headers import Cache from c2cgeoportal_geoportal.lib.filter_capabilities import ( filter_wfst_capabilities, @@ -97,11 +102,10 @@ def proxy(self) -> pyramid.response.Response: if operation is None or operation == "": operation = "getcapabilities" - if operation == "describefeaturetype": - # for DescribeFeatureType we require that exactly one type-name - # is given, otherwise we would have to filter the result - if len(typenames) != 1: - raise HTTPBadRequest("Exactly one type-name must be given for DescribeFeatureType requests") + # for DescribeFeatureType we require that exactly one type-name + # is given, otherwise we would have to filter the result + if operation == "describefeaturetype" and len(typenames) != 1: + raise HTTPBadRequest("Exactly one type-name must be given for DescribeFeatureType requests") if not self._is_allowed(typenames): raise HTTPForbidden("No access rights for at least one of the given type-names") diff --git a/geoportal/c2cgeoportal_geoportal/views/vector_tiles.py b/geoportal/c2cgeoportal_geoportal/views/vector_tiles.py index 9fd929ba9e..78c8062bd4 100644 --- a/geoportal/c2cgeoportal_geoportal/views/vector_tiles.py +++ b/geoportal/c2cgeoportal_geoportal/views/vector_tiles.py @@ -28,6 +28,7 @@ import logging import sqlalchemy +from c2cgeoportal_commons.models import DBSession, main from pyramid.httpexceptions import HTTPNotFound from pyramid.request import Request from pyramid.response import Response @@ -35,7 +36,6 @@ from tilecloud import TileCoord from tilecloud.grid.free import FreeTileGrid -from c2cgeoportal_commons.models import DBSession, main from c2cgeoportal_geoportal.lib.common_headers import Cache, set_common_headers _LOG = logging.getLogger(__name__) diff --git a/geoportal/package-lock.json b/geoportal/package-lock.json index 7b53e455a7..91b791db97 100644 --- a/geoportal/package-lock.json +++ b/geoportal/package-lock.json @@ -10,13 +10,13 @@ "dependencies": { "@snyk/protect": "1.1294.2", "ngeo": "2.9.0-version-2.9-latest.20241204T095745Z.b6b15744f.2.9", - "puppeteer": "23.7.1" + "puppeteer": "23.9.0" }, "devDependencies": { - "@eslint/eslintrc": "3.1.0", - "@eslint/js": "9.14.0", + "@eslint/eslintrc": "3.2.0", + "@eslint/js": "9.16.0", "@popperjs/core": "2.11.8", - "@sentry/browser": "8.37.1", + "@sentry/browser": "8.42.0", "angular-gettext-tools": "2.5.3", "angular-mocks": "1.8.3", "commander": "12.1.0", @@ -25,7 +25,7 @@ "cy-mobile-commands": "0.3.0", "ejs-loader": "0.5.0", "eslint": "8.57.1", - "eslint-plugin-jsdoc": "50.4.3", + "eslint-plugin-jsdoc": "50.6.0", "eslint-plugin-lit": "1.15.0", "eslint-plugin-storybook": "0.11.1", "eslint-plugin-wc": "2.2.0", @@ -40,7 +40,7 @@ "minify-html-literals-loader": "1.1.1", "neat-csv": "7.0.0", "parse-absolute-css-unit": "1.0.2", - "sass": "1.80.7", + "sass": "1.81.1", "sass-loader": "16.0.3", "simple-html-tokenizer": "0.5.11", "svgo": "3.3.2", @@ -48,7 +48,7 @@ "terser": "5.36.0", "terser-webpack-plugin": "5.3.10", "url-parse": "1.5.10", - "webpack": "5.96.1", + "webpack": "5.97.0", "webpack-cli": "5.1.4", "webpack-dev-server": "5.1.0", "webpack-merge": "6.0.1" @@ -577,9 +577,9 @@ } }, "node_modules/@eslint/eslintrc": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.1.0.tgz", - "integrity": "sha512-4Bfj15dVJdoy3RfZmmo86RK1Fwzn6SstsvK9JS+BaVKqC6QQQQyXekNaC+g+LKNgkQ+2VhGAzm6hO40AhMR3zQ==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.2.0.tgz", + "integrity": "sha512-grOjVNN8P3hjJn/eIETF1wwd12DdnwFDoyceUJLYYdkpbwq3nLi+4fqrTAONx7XDALqlL220wC/RHSC/QTI/0w==", "dev": true, "license": "MIT", "dependencies": { @@ -601,9 +601,9 @@ } }, "node_modules/@eslint/js": { - "version": "9.14.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.14.0.tgz", - "integrity": "sha512-pFoEtFWCPyDOl+C6Ift+wC7Ro89otjigCf5vcuWqWgqNSQbRrpjSvdeE6ofLz4dHmyxD5f7gIdGT4+p36L6Twg==", + "version": "9.16.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.16.0.tgz", + "integrity": "sha512-tw2HxzQkrbeuvyj1tG2Yqq+0H9wGoI2IMk4EOsQeX+vmd75FtJAzf+gTA69WF+baUKRYQ3x2kbLE08js5OsTVg==", "dev": true, "license": "MIT", "engines": { @@ -1315,123 +1315,86 @@ } }, "node_modules/@sentry-internal/browser-utils": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry-internal/browser-utils/-/browser-utils-8.37.1.tgz", - "integrity": "sha512-OSR/V5GCsSCG7iapWtXCT/y22uo3HlawdEgfM1NIKk1mkP15UyGQtGEzZDdih2H+SNuX1mp9jQLTjr5FFp1A5w==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry-internal/browser-utils/-/browser-utils-8.42.0.tgz", + "integrity": "sha512-xzgRI0wglKYsPrna574w1t38aftuvo44gjOKFvPNGPnYfiW9y4m+64kUz3JFbtanvOrKPcaITpdYiB4DeJXEbA==", "dev": true, "license": "MIT", "dependencies": { - "@sentry/core": "8.37.1", - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" + "@sentry/core": "8.42.0" }, "engines": { "node": ">=14.18" } }, "node_modules/@sentry-internal/feedback": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry-internal/feedback/-/feedback-8.37.1.tgz", - "integrity": "sha512-Se25NXbSapgS2S+JssR5YZ48b3OY4UGmAuBOafgnMW91LXMxRNWRbehZuNUmjjHwuywABMxjgu+Yp5uJDATX+g==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry-internal/feedback/-/feedback-8.42.0.tgz", + "integrity": "sha512-dkIw5Wdukwzngg5gNJ0QcK48LyJaMAnBspqTqZ3ItR01STi6Z+6+/Bt5XgmrvDgRD+FNBinflc5zMmfdFXXhvw==", "dev": true, "license": "MIT", "dependencies": { - "@sentry/core": "8.37.1", - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" + "@sentry/core": "8.42.0" }, "engines": { "node": ">=14.18" } }, "node_modules/@sentry-internal/replay": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry-internal/replay/-/replay-8.37.1.tgz", - "integrity": "sha512-E/Plhisk/pXJjOdOU12sg8m/APTXTA21iEniidP6jW3/+O0tD/H/UovEqa4odNTqxPMa798xHQSQNt5loYiaLA==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry-internal/replay/-/replay-8.42.0.tgz", + "integrity": "sha512-oNcJEBlDfXnRFYC5Mxj5fairyZHNqlnU4g8kPuztB9G5zlsyLgWfPxzcn1ixVQunth2/WZRklDi4o1ZfyHww7w==", "dev": true, "license": "MIT", "dependencies": { - "@sentry-internal/browser-utils": "8.37.1", - "@sentry/core": "8.37.1", - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" + "@sentry-internal/browser-utils": "8.42.0", + "@sentry/core": "8.42.0" }, "engines": { "node": ">=14.18" } }, "node_modules/@sentry-internal/replay-canvas": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry-internal/replay-canvas/-/replay-canvas-8.37.1.tgz", - "integrity": "sha512-1JLAaPtn1VL5vblB0BMELFV0D+KUm/iMGsrl4/JpRm0Ws5ESzQl33DhXVv1IX/ZAbx9i14EjR7MG9+Hj70tieQ==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry-internal/replay-canvas/-/replay-canvas-8.42.0.tgz", + "integrity": "sha512-XrPErqVhPsPh/oFLVKvz7Wb+Fi2J1zCPLeZCxWqFuPWI2agRyLVu0KvqJyzSpSrRAEJC/XFzuSVILlYlXXSfgA==", "dev": true, "license": "MIT", "dependencies": { - "@sentry-internal/replay": "8.37.1", - "@sentry/core": "8.37.1", - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" + "@sentry-internal/replay": "8.42.0", + "@sentry/core": "8.42.0" }, "engines": { "node": ">=14.18" } }, "node_modules/@sentry/browser": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry/browser/-/browser-8.37.1.tgz", - "integrity": "sha512-5ym+iGiIpjIKKpMWi9S3/tXh9xneS+jqxwRTJqed3cb8i4ydfMAAP8sM3U8xMCWWABpWyIUW+fpewC0tkhE1aQ==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry/browser/-/browser-8.42.0.tgz", + "integrity": "sha512-lStrEk609KJHwXfDrOgoYVVoFFExixHywxSExk7ZDtwj2YPv6r6Y1gogvgr7dAZj7jWzadHkxZ33l9EOSJBfug==", "dev": true, "license": "MIT", "dependencies": { - "@sentry-internal/browser-utils": "8.37.1", - "@sentry-internal/feedback": "8.37.1", - "@sentry-internal/replay": "8.37.1", - "@sentry-internal/replay-canvas": "8.37.1", - "@sentry/core": "8.37.1", - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" + "@sentry-internal/browser-utils": "8.42.0", + "@sentry-internal/feedback": "8.42.0", + "@sentry-internal/replay": "8.42.0", + "@sentry-internal/replay-canvas": "8.42.0", + "@sentry/core": "8.42.0" }, "engines": { "node": ">=14.18" } }, "node_modules/@sentry/core": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry/core/-/core-8.37.1.tgz", - "integrity": "sha512-82csXby589iDupM3VgCHJeWZagUyEEaDnbFcoZ/Z91QX2Sjq8FcF5OsforoXjw09i0XTFqlkFAnQVpDBmMXcpQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@sentry/types": "8.37.1", - "@sentry/utils": "8.37.1" - }, - "engines": { - "node": ">=14.18" - } - }, - "node_modules/@sentry/types": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry/types/-/types-8.37.1.tgz", - "integrity": "sha512-ryMOTROLSLINKFEbHWvi7GigNrsQhsaScw2NddybJGztJQ5UhxIGESnxGxWCufBmWFDwd7+5u0jDPCVUJybp7w==", + "version": "8.42.0", + "resolved": "https://registry.npmjs.org/@sentry/core/-/core-8.42.0.tgz", + "integrity": "sha512-ac6O3pgoIbU6rpwz6LlwW0wp3/GAHuSI0C5IsTgIY6baN8rOBnlAtG6KrHDDkGmUQ2srxkDJu9n1O6Td3cBCqw==", "dev": true, "license": "MIT", "engines": { "node": ">=14.18" } }, - "node_modules/@sentry/utils": { - "version": "8.37.1", - "resolved": "https://registry.npmjs.org/@sentry/utils/-/utils-8.37.1.tgz", - "integrity": "sha512-Qtn2IfpII12K17txG/ZtTci35XYjYi4CxbQ3j7nXY7toGv/+MqPXwV5q2i9g94XaSXlE5Wy9/hoCZoZpZs/djA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@sentry/types": "8.37.1" - }, - "engines": { - "node": ">=14.18" - } - }, "node_modules/@sindresorhus/merge-streams": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/@sindresorhus/merge-streams/-/merge-streams-2.3.0.tgz", @@ -5309,9 +5272,9 @@ "license": "MIT" }, "node_modules/devtools-protocol": { - "version": "0.0.1354347", - "resolved": "https://registry.npmjs.org/devtools-protocol/-/devtools-protocol-0.0.1354347.tgz", - "integrity": "sha512-BlmkSqV0V84E2WnEnoPnwyix57rQxAM5SKJjf4TbYOCGLAWtz8CDH8RIaGOjPgPCXo2Mce3kxSY497OySidY3Q==", + "version": "0.0.1367902", + "resolved": "https://registry.npmjs.org/devtools-protocol/-/devtools-protocol-0.0.1367902.tgz", + "integrity": "sha512-XxtPuC3PGakY6PD7dG66/o8KwJ/LkH2/EKe19Dcw58w53dv4/vSQEkn/SzuyhHE2q4zPgCkxQBxus3VV4ql+Pg==", "license": "BSD-3-Clause" }, "node_modules/dns-packet": { @@ -5770,9 +5733,9 @@ } }, "node_modules/eslint-plugin-jsdoc": { - "version": "50.4.3", - "resolved": "https://registry.npmjs.org/eslint-plugin-jsdoc/-/eslint-plugin-jsdoc-50.4.3.tgz", - "integrity": "sha512-uWtwFxGRv6B8sU63HZM5dAGDhgsatb+LONwmILZJhdRALLOkCX2HFZhdL/Kw2ls8SQMAVEfK+LmnEfxInRN8HA==", + "version": "50.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-jsdoc/-/eslint-plugin-jsdoc-50.6.0.tgz", + "integrity": "sha512-tCNp4fR79Le3dYTPB0dKEv7yFyvGkUCa+Z3yuTrrNGGOxBlXo9Pn0PEgroOZikUQOGjxoGMVKNjrOHcYEdfszg==", "dev": true, "license": "BSD-3-Clause", "dependencies": { @@ -10409,17 +10372,17 @@ } }, "node_modules/puppeteer": { - "version": "23.7.1", - "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-23.7.1.tgz", - "integrity": "sha512-jS6XehagMvxQ12etwY/4EOYZ0Sm8GAsrtGhdQn4AqpJAyHc3RYl7tGd4QYh/MmShDw8sF9FWYQqGidhoXaqokQ==", + "version": "23.9.0", + "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-23.9.0.tgz", + "integrity": "sha512-WfB8jGwFV+qrD9dcJJVvWPFJBU6kxeu2wxJz9WooDGfM3vIiKLgzImEDBxUQnCBK/2cXB3d4dV6gs/LLpgfLDg==", "hasInstallScript": true, "license": "Apache-2.0", "dependencies": { "@puppeteer/browsers": "2.4.1", "chromium-bidi": "0.8.0", "cosmiconfig": "^9.0.0", - "devtools-protocol": "0.0.1354347", - "puppeteer-core": "23.7.1", + "devtools-protocol": "0.0.1367902", + "puppeteer-core": "23.9.0", "typed-query-selector": "^2.12.0" }, "bin": { @@ -10430,15 +10393,15 @@ } }, "node_modules/puppeteer-core": { - "version": "23.7.1", - "resolved": "https://registry.npmjs.org/puppeteer-core/-/puppeteer-core-23.7.1.tgz", - "integrity": "sha512-Om/qCZhd+HLoAr7GltrRAZpS3uOXwHu7tXAoDbNcJADHjG2zeAlDArgyIPXYGG4QB/EQUHk13Q6RklNxGM73Pg==", + "version": "23.9.0", + "resolved": "https://registry.npmjs.org/puppeteer-core/-/puppeteer-core-23.9.0.tgz", + "integrity": "sha512-hLVrav2HYMVdK0YILtfJwtnkBAwNOztUdR4aJ5YKDvgsbtagNr6urUJk9HyjRA9e+PaLI3jzJ0wM7A4jSZ7Qxw==", "license": "Apache-2.0", "dependencies": { "@puppeteer/browsers": "2.4.1", "chromium-bidi": "0.8.0", "debug": "^4.3.7", - "devtools-protocol": "0.0.1354347", + "devtools-protocol": "0.0.1367902", "typed-query-selector": "^2.12.0", "ws": "^8.18.0" }, @@ -11148,9 +11111,9 @@ "license": "MIT" }, "node_modules/sass": { - "version": "1.80.7", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.80.7.tgz", - "integrity": "sha512-MVWvN0u5meytrSjsU7AWsbhoXi1sc58zADXFllfZzbsBT1GHjjar6JwBINYPRrkx/zqnQ6uqbQuHgE95O+C+eQ==", + "version": "1.81.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.81.1.tgz", + "integrity": "sha512-VNLgf4FC5yFyKwAumAAwwNh8X4SevlVREq3Y8aDZIkm0lI/zO1feycMXQ4hn+eB6FVhRbleSQ1Yb/q8juSldTA==", "dev": true, "license": "MIT", "dependencies": { @@ -12877,17 +12840,17 @@ "license": "BSD-2-Clause" }, "node_modules/webpack": { - "version": "5.96.1", - "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.96.1.tgz", - "integrity": "sha512-l2LlBSvVZGhL4ZrPwyr8+37AunkcYj5qh8o6u2/2rzoPc8gxFJkLj1WxNgooi9pnoc06jh0BjuXnamM4qlujZA==", + "version": "5.97.0", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.97.0.tgz", + "integrity": "sha512-CWT8v7ShSfj7tGs4TLRtaOLmOCPWhoKEvp+eA7FVx8Xrjb3XfT0aXdxDItnRZmE8sHcH+a8ayDrJCOjXKxVFfQ==", "dev": true, "license": "MIT", "dependencies": { "@types/eslint-scope": "^3.7.7", "@types/estree": "^1.0.6", - "@webassemblyjs/ast": "^1.12.1", - "@webassemblyjs/wasm-edit": "^1.12.1", - "@webassemblyjs/wasm-parser": "^1.12.1", + "@webassemblyjs/ast": "^1.14.1", + "@webassemblyjs/wasm-edit": "^1.14.1", + "@webassemblyjs/wasm-parser": "^1.14.1", "acorn": "^8.14.0", "browserslist": "^4.24.0", "chrome-trace-event": "^1.0.2", diff --git a/geoportal/package.json b/geoportal/package.json index 4ad69a66c4..691f8809e4 100644 --- a/geoportal/package.json +++ b/geoportal/package.json @@ -10,13 +10,13 @@ "dependencies": { "@snyk/protect": "1.1294.2", "ngeo": "2.9.0-version-2.9-latest.20241204T095745Z.b6b15744f.2.9", - "puppeteer": "23.7.1" + "puppeteer": "23.9.0" }, "devDependencies": { - "@eslint/eslintrc": "3.1.0", - "@eslint/js": "9.14.0", + "@eslint/eslintrc": "3.2.0", + "@eslint/js": "9.16.0", "@popperjs/core": "2.11.8", - "@sentry/browser": "8.37.1", + "@sentry/browser": "8.42.0", "angular-gettext-tools": "2.5.3", "angular-mocks": "1.8.3", "commander": "12.1.0", @@ -25,7 +25,7 @@ "cy-mobile-commands": "0.3.0", "ejs-loader": "0.5.0", "eslint": "8.57.1", - "eslint-plugin-jsdoc": "50.4.3", + "eslint-plugin-jsdoc": "50.6.0", "eslint-plugin-lit": "1.15.0", "eslint-plugin-storybook": "0.11.1", "eslint-plugin-wc": "2.2.0", @@ -40,7 +40,7 @@ "minify-html-literals-loader": "1.1.1", "neat-csv": "7.0.0", "parse-absolute-css-unit": "1.0.2", - "sass": "1.80.7", + "sass": "1.81.1", "sass-loader": "16.0.3", "simple-html-tokenizer": "0.5.11", "svgo": "3.3.2", @@ -48,7 +48,7 @@ "terser": "5.36.0", "terser-webpack-plugin": "5.3.10", "url-parse": "1.5.10", - "webpack": "5.96.1", + "webpack": "5.97.0", "webpack-cli": "5.1.4", "webpack-dev-server": "5.1.0", "webpack-merge": "6.0.1" diff --git a/geoportal/tests/__init__.py b/geoportal/tests/__init__.py index 465a966acb..b981949bc6 100644 --- a/geoportal/tests/__init__.py +++ b/geoportal/tests/__init__.py @@ -1,4 +1,4 @@ -# Copyright (c) 2011-2023, Camptocamp SA +# Copyright (c) 2011-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -26,9 +26,7 @@ # either expressed or implied, of the FreeBSD Project. -""" -Pyramid application test package. -""" +"""Pyramid application test package.""" import logging import os @@ -36,9 +34,8 @@ import warnings import sqlalchemy.exc -from pyramid.testing import DummyRequest as PyramidDummyRequest - from c2cgeoportal_geoportal.lib import caching +from pyramid.testing import DummyRequest as PyramidDummyRequest with warnings.catch_warnings(): warnings.simplefilter("ignore", category=sqlalchemy.exc.SAWarning) diff --git a/geoportal/tests/functional/__init__.py b/geoportal/tests/functional/__init__.py index 855e0dbf1e..aa8a397c13 100644 --- a/geoportal/tests/functional/__init__.py +++ b/geoportal/tests/functional/__init__.py @@ -26,27 +26,26 @@ # either expressed or implied, of the FreeBSD Project. -""" -Pyramid application test package. -""" +"""Pyramid application test package.""" import logging from configparser import ConfigParser from typing import TYPE_CHECKING, Any +import c2cgeoportal_geoportal +import c2cgeoportal_geoportal.lib import pyramid.registry import pyramid.request -import tests +import sqlalchemy.exc import transaction import webob.acceptparse from c2c.template.config import config as configuration +from c2cgeoportal_commons import models +from c2cgeoportal_geoportal.lib import caching from pyramid import testing from sqlalchemy.orm.session import Session -import c2cgeoportal_geoportal -import c2cgeoportal_geoportal.lib -from c2cgeoportal_commons import models -from c2cgeoportal_geoportal.lib import caching +import tests if TYPE_CHECKING: from c2cgeoportal_commons.models import main, static @@ -64,9 +63,7 @@ class DummyRoute: def cleanup_db() -> None: - """ - Cleanup the database. - """ + """Cleanup the database.""" from c2cgeoportal_commons import models from c2cgeoportal_commons.models.main import ( FullTextSearch, @@ -103,9 +100,7 @@ def cleanup_db() -> None: def setup_db() -> None: - """ - Cleanup the database. - """ + """Cleanup the database.""" cleanup_db() from c2cgeoportal_commons.models import DBSession @@ -211,8 +206,12 @@ def init_registry(registry=None): def testing_legacySecurityPolicy( config, userid=None, groupids=(), permissive=True, remember_result=None, forget_result=None ): - """Compatibility mode for deprecated AuthorizationPolicy and AuthenticationPolicy in our tests""" - from pyramid.interfaces import IAuthenticationPolicy, IAuthorizationPolicy, ISecurityPolicy + """Compatibility mode for deprecated AuthorizationPolicy and AuthenticationPolicy in our tests.""" + from pyramid.interfaces import ( + IAuthenticationPolicy, + IAuthorizationPolicy, + ISecurityPolicy, + ) from pyramid.security import LegacySecurityPolicy from pyramid.testing import DummySecurityPolicy @@ -228,10 +227,9 @@ def testing_legacySecurityPolicy( def create_dummy_request( additional_settings=None, authentication=True, user=None, *args: Any, **kargs: Any ) -> pyramid.request.Request: - from pyramid.interfaces import IAuthenticationPolicy - from c2cgeoportal_geoportal import create_get_user_from_request from c2cgeoportal_geoportal.lib.authentication import create_authentication + from pyramid.interfaces import IAuthenticationPolicy if additional_settings is None: additional_settings = {} diff --git a/geoportal/tests/functional/conftest.py b/geoportal/tests/functional/conftest.py index 049cbdc8cc..b31a7c0e11 100644 --- a/geoportal/tests/functional/conftest.py +++ b/geoportal/tests/functional/conftest.py @@ -34,13 +34,18 @@ import sqlalchemy.orm import transaction from c2c.template.config import config as configuration +from c2cgeoportal_commons.testing import ( + generate_mappers, + get_engine, + get_session_factory, + get_tm_session, +) +from c2cgeoportal_commons.testing.initializedb import truncate_tables +from c2cgeoportal_geoportal.lib import caching from pyramid.testing import DummyRequest from sqlalchemy.orm.session import Session, SessionTransaction -from tests.functional import setup_common as setup_module -from c2cgeoportal_commons.testing import generate_mappers, get_engine, get_session_factory, get_tm_session -from c2cgeoportal_commons.testing.initializedb import truncate_tables -from c2cgeoportal_geoportal.lib import caching +from tests.functional import setup_common as setup_module _LOG = logging.getLogger(__name__) mapserv_url = "http://mapserver:8080/" diff --git a/geoportal/tests/functional/test_authentication.py b/geoportal/tests/functional/test_authentication.py index aad67c7a73..c228fa529d 100644 --- a/geoportal/tests/functional/test_authentication.py +++ b/geoportal/tests/functional/test_authentication.py @@ -32,15 +32,14 @@ from unittest.mock import patch import transaction -from tests.functional import create_dummy_request -from tests.functional import setup_common as setup_module # noqa -from tests.functional import teardown_common as teardown_module # noqa - -from c2cgeoportal_geoportal.lib import authentication from c2cgeoportal_geoportal.lib.authentication import UrlAuthenticationPolicy from c2cgeoportal_geoportal.resources import defaultgroupsfinder from c2cgeoportal_geoportal.scripts.urllogin import create_token +from tests.functional import create_dummy_request +from tests.functional import setup_common as setup_module # noqa +from tests.functional import teardown_common as teardown_module # noqa + class TestUrlAuthenticationPolicy(TestCase): def setup_method(self, _): @@ -110,9 +109,7 @@ def test_wrong_key(self): @patch("c2cgeoportal_geoportal.lib.authentication._LOG.error", side_effect=Exception()) def test_wrong_method(self, log_mock): # pylint: disable=unused-argument - """ - POST requests with input named "auth" must not raise exceptions due to urllogin. - """ + """POST requests with input named "auth" must not raise exceptions due to urllogin.""" def _get_user(method): request = create_dummy_request(params={"auth": "this is a wrong field value"}, method=method) diff --git a/geoportal/tests/functional/test_dbreflection.py b/geoportal/tests/functional/test_dbreflection.py index 01ac7b22d6..30b004a13a 100644 --- a/geoportal/tests/functional/test_dbreflection.py +++ b/geoportal/tests/functional/test_dbreflection.py @@ -30,11 +30,11 @@ from unittest import TestCase +from c2cgeoportal_geoportal.lib.caching import init_region + from tests.functional import setup_common as setup_module from tests.functional import teardown_common as teardown_module # noqa -from c2cgeoportal_geoportal.lib.caching import init_region - class TestReflection(TestCase): _tables = None @@ -60,11 +60,10 @@ def _create_table(self, tablename): Each test function should call this function only once. And there should not be two test functions that call this function with the same ptable_name value. """ - from geoalchemy2 import Geometry - from sqlalchemy import Column, ForeignKey, Table, types - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Base + from geoalchemy2 import Geometry + from sqlalchemy import Column, ForeignKey, Table, types if self._tables is None: self._tables = [] @@ -104,16 +103,14 @@ def _create_table(self, tablename): self.metadata = Base.metadata def test_get_class_nonexisting_table(self): - from sqlalchemy.exc import NoSuchTableError - from c2cgeoportal_geoportal.lib.dbreflection import get_class + from sqlalchemy.exc import NoSuchTableError self.assertRaises(NoSuchTableError, get_class, "nonexisting") def test_get_class(self): - from geoalchemy2 import Geometry - from c2cgeoportal_geoportal.lib.dbreflection import _AssociationProxy, get_class + from geoalchemy2 import Geometry init_region({"backend": "dogpile.cache.memory"}, "std") init_region({"backend": "dogpile.cache.memory"}, "obj") @@ -142,8 +139,8 @@ def test_get_class(self): assert modelclass.child2_id.info.get("association_proxy") == "child2" child1_asso_proxy = getattr(modelclass, modelclass.child1_id.info["association_proxy"]) - assert "name" == child1_asso_proxy.value_attr - assert "name" == child1_asso_proxy.order_by + assert child1_asso_proxy.value_attr == "name" + assert child1_asso_proxy.order_by == "name" # test the Table object table = modelclass.__table__ @@ -192,14 +189,11 @@ def test_get_class_dotted_notation(self): assert modelclass.__table__.schema == "public" def test_mixing_get_class_and_queries(self): - """ - This test shows that we can mix the use of DBSession and the db reflection API. - """ + """This test shows that we can mix the use of DBSession and the db reflection API.""" import transaction - from sqlalchemy import text - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_geoportal.lib.dbreflection import get_class + from sqlalchemy import text self._create_table("table_c") @@ -239,16 +233,16 @@ def test_get_class_enumerations_config(self): assert enumerations_config == cls.__enumerations_config__ association_proxy = getattr(cls, cls.child1_id.info["association_proxy"]) - assert "id" == association_proxy.value_attr - assert "name" == association_proxy.order_by + assert association_proxy.value_attr == "id" + assert association_proxy.order_by == "name" # Without order_by. enumerations_config = {"child1_id": {"value": "id"}} cls = get_class("table_d", enumerations_config=enumerations_config) association_proxy = getattr(cls, cls.child1_id.info["association_proxy"]) - assert "id" == association_proxy.value_attr - assert "id" == association_proxy.order_by + assert association_proxy.value_attr == "id" + assert association_proxy.order_by == "id" def test_get_class_readonly_attributes(self): from c2cgeoportal_geoportal.lib.dbreflection import get_class @@ -258,5 +252,5 @@ def test_get_class_readonly_attributes(self): self._create_table("table_d") cls = get_class("table_d", readonly_attributes=readonly_attributes) - assert True == cls.child1_id.info.get("readonly") - assert True == cls.point.info.get("readonly") + assert True is cls.child1_id.info.get("readonly") + assert True is cls.point.info.get("readonly") diff --git a/geoportal/tests/functional/test_dynamicview.py b/geoportal/tests/functional/test_dynamicview.py index 1c70b54af1..b76ed2c84f 100644 --- a/geoportal/tests/functional/test_dynamicview.py +++ b/geoportal/tests/functional/test_dynamicview.py @@ -31,23 +31,22 @@ from unittest import TestCase import pyramid.url +from c2cgeoportal_geoportal.lib.caching import init_region from pyramid import testing from pyramid.testing import testConfig -from tests import DummyRequest -from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import -from c2cgeoportal_geoportal.lib.caching import init_region +from tests import DummyRequest +from tests.functional import setup_common as setup_module # pylint: disable=unused-import +from tests.functional import teardown_common as teardown_module # pylint: disable=unused-import class TestDynamicView(TestCase): def setup_method(self, _): import transaction - from geoalchemy2 import WKTElement - from sqlalchemy import func - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import FullTextSearch + from geoalchemy2 import WKTElement + from sqlalchemy import func entry1 = FullTextSearch() entry1.label = "label 1" @@ -81,7 +80,6 @@ def teardown_method(self, _): testing.tearDown() import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import FullTextSearch @@ -101,7 +99,9 @@ def _get_settings(settings): } @staticmethod - def _request(query={}): + def _request(query=None): + if query is None: + query = {} query_ = {"interface": "test"} query_.update(query) request = DummyRequest(query_) diff --git a/geoportal/tests/functional/test_entry.py b/geoportal/tests/functional/test_entry.py index ab84ca9fc8..b0fbdb1f7b 100644 --- a/geoportal/tests/functional/test_entry.py +++ b/geoportal/tests/functional/test_entry.py @@ -35,9 +35,15 @@ import transaction from geoalchemy2 import WKTElement from pyramid import testing -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request, mapserv_url + +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + mapserv_url, + setup_db, +) from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import _LOG = logging.getLogger(__name__) @@ -50,10 +56,6 @@ def setup_method(self, _): self.maxDiff = None # pylint: disable=invalid-name self._tables = [] - from geoalchemy2 import Geometry - from sqlalchemy import Column, Table, func, types - from sqlalchemy.ext.declarative import declarative_base - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import ( OGCSERVER_AUTH_GEOSERVER, @@ -69,6 +71,9 @@ def setup_method(self, _): Theme, ) from c2cgeoportal_commons.models.static import User + from geoalchemy2 import Geometry + from sqlalchemy import Column, Table, func, types + from sqlalchemy.ext.declarative import declarative_base setup_db() diff --git a/geoportal/tests/functional/test_fulltextsearch.py b/geoportal/tests/functional/test_fulltextsearch.py index 5a399c2790..e59a5b29c0 100644 --- a/geoportal/tests/functional/test_fulltextsearch.py +++ b/geoportal/tests/functional/test_fulltextsearch.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -33,6 +33,7 @@ import webob.acceptparse from pyramid import testing from pyramid.response import Response + from tests.functional import create_dummy_request from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -41,12 +42,11 @@ class TestFulltextsearchView(TestCase): def setup_method(self, _): import transaction - from geoalchemy2 import WKTElement - from sqlalchemy import func - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import FullTextSearch, Interface, Role from c2cgeoportal_commons.models.static import User + from geoalchemy2 import WKTElement + from sqlalchemy import func role1 = Role(name="__test_role1", description="__test_role1") user1 = User(username="__test_user1", password="__test_user1", settings_role=role1, roles=[role1]) @@ -135,7 +135,6 @@ def teardown_method(self, _): testing.tearDown() import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import FullTextSearch, Interface, Role from c2cgeoportal_commons.models.static import User @@ -176,9 +175,8 @@ def _create_dummy_request(username=None, params=None): return request def test_no_default_laguage(self): - from pyramid.httpexceptions import HTTPInternalServerError - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from pyramid.httpexceptions import HTTPInternalServerError request = self._create_dummy_request() del request.registry.settings["default_locale_name"] @@ -189,9 +187,8 @@ def test_no_default_laguage(self): self.assertTrue(isinstance(response, HTTPInternalServerError)) def test_unknown_laguage(self): - from pyramid.httpexceptions import HTTPInternalServerError - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from pyramid.httpexceptions import HTTPInternalServerError request = self._create_dummy_request() request.registry.settings["default_locale_name"] = "it" @@ -201,9 +198,8 @@ def test_unknown_laguage(self): self.assertTrue(isinstance(response, HTTPInternalServerError)) def test_badrequest_noquery(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from pyramid.httpexceptions import HTTPBadRequest request = self._create_dummy_request() fts = FullTextSearchView(request) @@ -211,9 +207,8 @@ def test_badrequest_noquery(self): self.assertTrue(isinstance(response, HTTPBadRequest)) def test_badrequest_limit(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from pyramid.httpexceptions import HTTPBadRequest request = self._create_dummy_request(params=dict(query="text", limit="bad")) fts = FullTextSearchView(request) @@ -221,9 +216,8 @@ def test_badrequest_limit(self): self.assertTrue(isinstance(response, HTTPBadRequest)) def test_badrequest_partitionlimit(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from pyramid.httpexceptions import HTTPBadRequest request = self._create_dummy_request(params=dict(query="text", partitionlimit="bad")) fts = FullTextSearchView(request) @@ -231,9 +225,8 @@ def test_badrequest_partitionlimit(self): self.assertTrue(isinstance(response, HTTPBadRequest)) def test_limit(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra sol", limit=1)) fts = FullTextSearchView(request) @@ -244,9 +237,8 @@ def test_limit(self): assert response.features[0].properties["layer_name"] == "layer1" def test_toobig_limit(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra sol", limit=2000)) fts = FullTextSearchView(request) @@ -259,9 +251,8 @@ def test_toobig_limit(self): assert response.features[1].properties["layer_name"] == "layer1" def test_toobig_partitionlimit(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra sol", partitionlimit=2000)) fts = FullTextSearchView(request) @@ -274,9 +265,8 @@ def test_toobig_partitionlimit(self): assert response.features[1].properties["layer_name"] == "layer1" def test_match(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra sol", limit=40)) fts = FullTextSearchView(request) @@ -289,9 +279,8 @@ def test_match(self): assert response.features[1].properties["layer_name"] == "layer1" def test_nomatch(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="foo")) fts = FullTextSearchView(request) @@ -300,9 +289,8 @@ def test_nomatch(self): assert len(response.features) == 0 def test_private_nomatch(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="pl sem", limit=40)) fts = FullTextSearchView(request) @@ -311,9 +299,8 @@ def test_private_nomatch(self): assert len(response.features) == 0 def test_private_match(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="pl sem", limit=40), username="__test_user1") fts = FullTextSearchView(request) @@ -324,9 +311,8 @@ def test_private_match(self): assert response.features[0].properties["layer_name"] == "layer2" def test_private_with_role_nomatch(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="ven nei", limit=40), username="__test_user1") fts = FullTextSearchView(request) @@ -335,9 +321,8 @@ def test_private_with_role_nomatch(self): assert len(response.features) == 0 def test_private_with_role_match(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="ven nei", limit=40), username="__test_user2") fts = FullTextSearchView(request) @@ -348,9 +333,8 @@ def test_private_with_role_match(self): assert response.features[0].properties["layer_name"] == "layer3" def test_match_partitionlimit(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra sol", limit=40, partitionlimit=1)) fts = FullTextSearchView(request) @@ -361,9 +345,8 @@ def test_match_partitionlimit(self): assert response.features[0].properties["layer_name"] == "layer1" def test_params_actions(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="lausanne", limit=10)) fts = FullTextSearchView(request) @@ -377,9 +360,8 @@ def test_params_actions(self): ) def test_interface(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="lausanne", limit=10, interface="main")) fts = FullTextSearchView(request) @@ -388,9 +370,8 @@ def test_interface(self): self.assertEqual({feature.properties["label"] for feature in response.features}, {"label5", "label6"}) def test_rank_order_with_similarity(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="simi", limit=3)) fts = FullTextSearchView(request) @@ -402,9 +383,8 @@ def test_rank_order_with_similarity(self): assert response.features[2].properties["label"] == "A 71 simi" def test_rank_order_with_ts_rank_cd(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="simi", limit=3, ranksystem="ts_rank_cd")) fts = FullTextSearchView(request) @@ -416,9 +396,8 @@ def test_rank_order_with_ts_rank_cd(self): assert [f.properties["label"] for f in response.features] == ["A 7 simi", "A 70 simi", "A 71 simi"] def test_extra_quote(self): - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.fulltextsearch import FullTextSearchView + from geojson.feature import FeatureCollection request = self._create_dummy_request(params=dict(query="tra 'sol")) fts = FullTextSearchView(request) diff --git a/geoportal/tests/functional/test_functionalities.py b/geoportal/tests/functional/test_functionalities.py index 9aea693bce..f56c150f12 100644 --- a/geoportal/tests/functional/test_functionalities.py +++ b/geoportal/tests/functional/test_functionalities.py @@ -38,7 +38,6 @@ class TestFunctionalities(TestCase): def setup_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Functionality, Role from c2cgeoportal_commons.models.static import User @@ -67,7 +66,6 @@ def setup_method(self, _): def teardown_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Functionality, OGCServer, Role from c2cgeoportal_commons.models.static import User @@ -89,12 +87,12 @@ def teardown_method(self, _): transaction.commit() def test_functionalities(self): - from tests.functional import create_dummy_request - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.static import User from c2cgeoportal_geoportal.lib.functionality import get_functionality + from tests.functional import create_dummy_request + request = create_dummy_request() request.user = None request1 = create_dummy_request() @@ -189,11 +187,11 @@ def test_functionalities(self): self.assertEqual(set(get_functionality("__test_a", request2, False)), {"db1", "db2"}) def test_web_client_functionalities(self): - from tests.functional import create_dummy_request - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.static import User + from tests.functional import create_dummy_request + request = create_dummy_request() request.static_url = lambda url: "http://example.com/dummy/static/url" request1 = create_dummy_request() diff --git a/geoportal/tests/functional/test_geometry_processing.py b/geoportal/tests/functional/test_geometry_processing.py index 1966ef9db5..aa2273394a 100644 --- a/geoportal/tests/functional/test_geometry_processing.py +++ b/geoportal/tests/functional/test_geometry_processing.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -37,8 +37,6 @@ class TestGeometryProcessing(TestCase): def test_difference(self): - from shapely.geometry import Polygon - from c2cgeoportal_geoportal.views.geometry_processing import GeometryProcessing request = create_dummy_request() diff --git a/geoportal/tests/functional/test_groups_finder.py b/geoportal/tests/functional/test_groups_finder.py index 9aaf9e9edd..1602a5e42a 100644 --- a/geoportal/tests/functional/test_groups_finder.py +++ b/geoportal/tests/functional/test_groups_finder.py @@ -1,4 +1,4 @@ -# Copyright (c) 2011-2023, Camptocamp SA +# Copyright (c) 2011-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -38,7 +38,6 @@ class TestGroupsFinder(TestCase): def setup_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Role from c2cgeoportal_commons.models.static import User @@ -51,7 +50,6 @@ def setup_method(self, _): def teardown_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Role from c2cgeoportal_commons.models.static import User diff --git a/geoportal/tests/functional/test_layers.py b/geoportal/tests/functional/test_layers.py index 0c06fd87c6..31a8decd50 100644 --- a/geoportal/tests/functional/test_layers.py +++ b/geoportal/tests/functional/test_layers.py @@ -31,9 +31,13 @@ from typing import TYPE_CHECKING, Any from unittest import TestCase -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + setup_db, +) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa if TYPE_CHECKING: @@ -48,7 +52,6 @@ def setup_method(self, _: Any) -> None: setup_module() import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Interface, Role from c2cgeoportal_commons.models.static import User @@ -77,7 +80,6 @@ def setup_method(self, _: Any) -> None: def teardown_method(self, _: Any) -> None: import transaction - from c2cgeoportal_commons.models import DBSession transaction.abort() @@ -105,13 +107,16 @@ def _create_layer( It creates a layer with two features, and associates a restriction area to it. """ import transaction + from c2cgeoportal_commons.models import DBSession + from c2cgeoportal_commons.models.main import ( + LayerWMS, + OGCServer, + RestrictionArea, + ) from geoalchemy2 import Geometry, WKTElement from sqlalchemy import CheckConstraint, Column, ForeignKey, Table, types from sqlalchemy.ext.declarative import declarative_base - from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import LayerWMS, OGCServer, RestrictionArea - if self._tables is None: self._tables = [] @@ -228,9 +233,8 @@ def _get_request(layerid, username=None) -> None: return request def test_read_public(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer(public=True) request = self._get_request(layer_id) @@ -240,9 +244,8 @@ def test_read_public(self) -> None: self.assertEqual([f.properties["child"] for f in collection.features], ["c1é", "c2é"]) def test_read_many_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -251,9 +254,8 @@ def test_read_many_no_auth(self) -> None: self.assertRaises(HTTPForbidden, layers.read_many) def test_read_many(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -264,9 +266,8 @@ def test_read_many(self) -> None: assert [f.properties["child"] for f in collection.features] == ["c1é"] def test_read_many_layer_not_found(self) -> None: - from pyramid.httpexceptions import HTTPNotFound - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPNotFound self._create_layer() request = self._get_request(10000, username="__test_user") @@ -275,9 +276,8 @@ def test_read_many_layer_not_found(self) -> None: self.assertRaises(HTTPNotFound, layers.read_many) def test_read_many_multi(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id1 = self._create_layer() layer_id2 = self._create_layer() @@ -294,9 +294,8 @@ def test_read_many_multi(self) -> None: ) def test_read_one_public(self) -> None: - from geojson.feature import Feature - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import Feature layer_id = self._create_layer(public=True) request = self._get_request(layer_id) @@ -310,9 +309,8 @@ def test_read_one_public(self) -> None: assert feature.properties["child"] == "c1é" def test_read_one_public_notfound(self) -> None: - from pyramid.httpexceptions import HTTPNotFound - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPNotFound layer_id = self._create_layer(public=True) request = self._get_request(layer_id) @@ -323,9 +321,8 @@ def test_read_one_public_notfound(self) -> None: self.assertTrue(isinstance(feature, HTTPNotFound)) def test_read_one_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -335,9 +332,8 @@ def test_read_one_no_auth(self) -> None: self.assertRaises(HTTPForbidden, layers.read_one) def test_read_one_no_perm(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -347,9 +343,8 @@ def test_read_one_no_perm(self) -> None: self.assertRaises(HTTPForbidden, layers.read_one) def test_read_one(self) -> None: - from geojson.feature import Feature - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import Feature layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -373,9 +368,8 @@ def test_count(self) -> None: assert response == 2 def test_create_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -385,9 +379,8 @@ def test_create_no_auth(self) -> None: self.assertRaises(HTTPForbidden, layers.create) def test_create_no_perm(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -397,9 +390,8 @@ def test_create_no_perm(self) -> None: self.assertRaises(HTTPForbidden, layers.create) def test_create(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -427,12 +419,11 @@ def test_create_with_constraint_fail_integrity(self) -> None: def test_create_log(self) -> None: from datetime import datetime - from geojson.feature import FeatureCollection - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Metadata from c2cgeoportal_commons.models.static import User from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection self.assertEqual(DBSession.query(User.username).all(), [("__test_user",)]) @@ -472,10 +463,9 @@ def test_create_validation_fails(self) -> None: assert response["message"] == "Too few points in geometry component[5 45]" def test_create_no_validation(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_commons.models.main import Metadata from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection metadatas = [Metadata("geometryValidation", "False")] layer_id = self._create_layer(metadatas=metadatas, geom_type=False) @@ -489,9 +479,8 @@ def test_create_no_validation(self) -> None: assert len(collection.features) == 2 def test_update_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -502,9 +491,8 @@ def test_update_no_auth(self) -> None: self.assertRaises(HTTPForbidden, layers.update) def test_update_no_perm_dst_geom(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -515,9 +503,8 @@ def test_update_no_perm_dst_geom(self) -> None: self.assertRaises(HTTPForbidden, layers.update) def test_update_no_perm_src_geom(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -626,9 +613,8 @@ def test_update_no_validation(self) -> None: assert feature.child == "c2é" def test_delete_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -638,9 +624,8 @@ def test_delete_no_auth(self) -> None: self.assertRaises(HTTPForbidden, layers.delete) def test_delete_no_perm(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id, username="__test_user") @@ -661,9 +646,8 @@ def test_delete(self) -> None: assert response.status_int == 204 def test_metadata_no_auth(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer() request = self._get_request(layer_id) @@ -742,9 +726,8 @@ def test_metadata_editing_enumeration_config(self) -> None: # # # With None area # # # def test_read_public_none_area(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer(public=True, none_area=True) request = self._get_request(layer_id) @@ -755,9 +738,8 @@ def test_read_public_none_area(self) -> None: self.assertEqual([f.properties["child"] for f in collection.features], ["c1é", "c2é"]) def test_read_many_no_auth_none_area(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id) @@ -766,9 +748,8 @@ def test_read_many_no_auth_none_area(self) -> None: self.assertRaises(HTTPForbidden, layers.read_many) def test_read_many_none_area(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id, username="__test_user") @@ -781,9 +762,8 @@ def test_read_many_none_area(self) -> None: assert collection.features[1].properties["child"] == "c2é" def test_read_one_public_none_area(self) -> None: - from geojson.feature import Feature - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import Feature layer_id = self._create_layer(public=True, none_area=True) request = self._get_request(layer_id) @@ -797,9 +777,8 @@ def test_read_one_public_none_area(self) -> None: assert feature.properties["child"] == "c1é" def test_read_one_no_auth_none_area(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id) @@ -809,9 +788,8 @@ def test_read_one_no_auth_none_area(self) -> None: self.assertRaises(HTTPForbidden, layers.read_one) def test_read_one_none_area(self) -> None: - from geojson.feature import Feature - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import Feature layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id, username="__test_user") @@ -835,9 +813,8 @@ def test_count_none_area(self) -> None: assert response == 2 def test_create_no_auth_none_area(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id) @@ -847,9 +824,8 @@ def test_create_no_auth_none_area(self) -> None: self.assertRaises(HTTPForbidden, layers.create) def test_create_none_area(self) -> None: - from geojson.feature import FeatureCollection - from c2cgeoportal_geoportal.views.layers import Layers + from geojson.feature import FeatureCollection layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id, username="__test_user") @@ -861,9 +837,8 @@ def test_create_none_area(self) -> None: assert len(collection.features) == 2 def test_update_no_auth_none_area(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id) @@ -888,9 +863,8 @@ def test_update_none_area(self) -> None: assert feature.child == "c2é" def test_delete_no_auth_none_area(self) -> None: - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.layers import Layers + from pyramid.httpexceptions import HTTPForbidden layer_id = self._create_layer(none_area=True) request = self._get_request(layer_id) diff --git a/geoportal/tests/functional/test_lingva_extractor_config.py b/geoportal/tests/functional/test_lingva_extractor_config.py index 8c3c75b528..fc9113a54a 100644 --- a/geoportal/tests/functional/test_lingva_extractor_config.py +++ b/geoportal/tests/functional/test_lingva_extractor_config.py @@ -32,10 +32,10 @@ import pytest import yaml from c2c.template.config import config as configuration -from tests.functional import setup_common as setup_module - from c2cgeoportal_geoportal.lib.lingva_extractor import GeomapfishConfigExtractor +from tests.functional import setup_common as setup_module + GMF_CONFIG = """ vars: interfaces_config: @@ -93,9 +93,8 @@ def dbsession_db1(settings, dbsession_old): @pytest.fixture(scope="function") def test_data(dbsession_db1, transact_old): - from sqlalchemy import text - from c2cgeoportal_commons.models import main + from sqlalchemy import text dbsession_db1.execute( text( diff --git a/geoportal/tests/functional/test_lingva_extractor_themes.py b/geoportal/tests/functional/test_lingva_extractor_themes.py index 55f2556f62..ee5600423e 100644 --- a/geoportal/tests/functional/test_lingva_extractor_themes.py +++ b/geoportal/tests/functional/test_lingva_extractor_themes.py @@ -29,8 +29,6 @@ from unittest.mock import Mock -import pytest - from c2cgeoportal_geoportal.lib.lingva_extractor import GeomapfishThemeExtractor diff --git a/geoportal/tests/functional/test_login.py b/geoportal/tests/functional/test_login.py index 80d8e67148..4a1b2297e8 100644 --- a/geoportal/tests/functional/test_login.py +++ b/geoportal/tests/functional/test_login.py @@ -36,10 +36,14 @@ import transaction from geoalchemy2 import WKTElement from pyramid import testing -from tests.functional import cleanup_db, create_dummy_request, fill_tech_user_functionality, mapserv_url -from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import setup_db -from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import + +from tests.functional import ( + cleanup_db, + create_dummy_request, + fill_tech_user_functionality, + mapserv_url, + setup_db, +) _LOG = logging.getLogger(__name__) @@ -102,11 +106,10 @@ def _create_request_obj(username=None, params=None, **kwargs): # def test_login_success(self): - from pyramid.httpexceptions import HTTPUnauthorized - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.static import User from c2cgeoportal_geoportal.views.login import Login + from pyramid.httpexceptions import HTTPUnauthorized user = DBSession.query(User).filter_by(username="__test_user1").one() user.is_password_changed = True @@ -136,9 +139,8 @@ def test_login_success(self): self.assertRaises(HTTPUnauthorized, login.login) def test_logout_no_auth(self): - from pyramid.httpexceptions import HTTPUnauthorized - from c2cgeoportal_geoportal.views.login import Login + from pyramid.httpexceptions import HTTPUnauthorized request = self._create_request_obj(path="/", params={"came_from": "/came_from"}) login = Login(request) @@ -209,18 +211,16 @@ def test_reset_password(self): self.assertNotEqual(len(user.password), 0) def test_change_password_no_params(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.login import Login + from pyramid.httpexceptions import HTTPBadRequest request = self._create_request_obj(username="__test_user1", params={"lang": "en"}, POST={}) login = Login(request) self.assertRaises(HTTPBadRequest, login.change_password) def test_change_password_wrong_old(self): - from pyramid.httpexceptions import HTTPUnauthorized - from c2cgeoportal_geoportal.views.login import Login + from pyramid.httpexceptions import HTTPUnauthorized request = self._create_request_obj( username="__test_user1", @@ -232,9 +232,8 @@ def test_change_password_wrong_old(self): login.change_password() def test_change_password_different(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.login import Login + from pyramid.httpexceptions import HTTPBadRequest request = self._create_request_obj( username="__test_user1", @@ -362,10 +361,10 @@ class F: assert request.response.headers["Vary"] == "Origin, Access-Control-Request-Headers, Cookie" def test_intranet(self): - from tests import DummyRequest - from c2cgeoportal_geoportal.views.login import Login + from tests import DummyRequest + request = DummyRequest() request.registry.settings = {"intranet": {"networks": ["192.168.1.0/255.255.255.0"]}} request.user = None diff --git a/geoportal/tests/functional/test_login_2fa.py b/geoportal/tests/functional/test_login_2fa.py index 164abfbbb5..c69db084ff 100644 --- a/geoportal/tests/functional/test_login_2fa.py +++ b/geoportal/tests/functional/test_login_2fa.py @@ -37,10 +37,8 @@ import pytest import transaction from pyramid import testing -from tests.functional import cleanup_db, create_dummy_request -from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import setup_db -from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import + +from tests.functional import cleanup_db, create_dummy_request, setup_db _LOG = logging.getLogger(__name__) diff --git a/geoportal/tests/functional/test_mapserverproxy.py b/geoportal/tests/functional/test_mapserverproxy.py index 9250dc5cb8..1d31d04826 100644 --- a/geoportal/tests/functional/test_mapserverproxy.py +++ b/geoportal/tests/functional/test_mapserverproxy.py @@ -62,15 +62,16 @@ import transaction from geoalchemy2 import WKTElement + from tests.functional import ( cleanup_db, create_default_ogcserver, create_dummy_request, fill_tech_user_functionality, mapserv_url, + setup_db, ) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa from tests.functional.geodata_model import PointTest @@ -1006,9 +1007,8 @@ def test_geoserver(self): assert "testpoint_protected" in response.body.decode("utf-8") def test_authentication_required(self): - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.mapserverproxy import MapservProxy + from pyramid.httpexceptions import HTTPForbidden request = self._create_getcap_request() request.params.update( diff --git a/geoportal/tests/functional/test_mapserverproxy_capabilities.py b/geoportal/tests/functional/test_mapserverproxy_capabilities.py index f3c824d892..5d00dbff0f 100644 --- a/geoportal/tests/functional/test_mapserverproxy_capabilities.py +++ b/geoportal/tests/functional/test_mapserverproxy_capabilities.py @@ -31,9 +31,15 @@ from unittest import TestCase import transaction -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request, mapserv_url + +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + mapserv_url, + setup_db, +) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa diff --git a/geoportal/tests/functional/test_mapserverproxy_group.py b/geoportal/tests/functional/test_mapserverproxy_group.py index 55cdba395d..56930856ff 100644 --- a/geoportal/tests/functional/test_mapserverproxy_group.py +++ b/geoportal/tests/functional/test_mapserverproxy_group.py @@ -32,6 +32,7 @@ import transaction from geoalchemy2 import WKTElement + from tests.functional import create_default_ogcserver, create_dummy_request from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -40,7 +41,12 @@ class TestMapserverproxyViewGroup(TestCase): def setup_method(self, _): from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerWMS, RestrictionArea, Role + from c2cgeoportal_commons.models.main import ( + Interface, + LayerWMS, + RestrictionArea, + Role, + ) from c2cgeoportal_commons.models.static import User ogc_server_internal = create_default_ogcserver(DBSession) @@ -72,7 +78,13 @@ def setup_method(self, _): def teardown_method(self, _): from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerWMS, OGCServer, RestrictionArea, Role + from c2cgeoportal_commons.models.main import ( + Interface, + LayerWMS, + OGCServer, + RestrictionArea, + Role, + ) from c2cgeoportal_commons.models.static import User DBSession.delete(DBSession.query(User).filter(User.username == "__test_user1").one()) diff --git a/geoportal/tests/functional/test_mobile_desktop.py b/geoportal/tests/functional/test_mobile_desktop.py index c64c1cad26..6ec5d0ac0b 100644 --- a/geoportal/tests/functional/test_mobile_desktop.py +++ b/geoportal/tests/functional/test_mobile_desktop.py @@ -32,6 +32,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -44,7 +45,12 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Theme, + ) create_default_ogcserver(DBSession) main = Interface(name="main") @@ -102,7 +108,13 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, OGCServer, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + OGCServer, + Theme, + ) for t in DBSession.query(Theme).all(): DBSession.delete(t) diff --git a/geoportal/tests/functional/test_oauth2.py b/geoportal/tests/functional/test_oauth2.py index 145de44097..8ace7a1ded 100644 --- a/geoportal/tests/functional/test_oauth2.py +++ b/geoportal/tests/functional/test_oauth2.py @@ -37,6 +37,7 @@ import pyramid.testing import pytest import transaction + from tests.functional import cleanup_db, create_dummy_request, init_registry from tests.functional import setup_common as setup_module # pylint: disable=unused-import from tests.functional import teardown_common as teardown_module # pylint: disable=unused-import @@ -588,7 +589,7 @@ def test_pkce_oauth2_protocol_test_login_get_token_refresh_token_is_login(self) request.method = "POST" request.body = "" with pytest.raises(pyramid.httpexceptions.HTTPFound) as exc_info: - assert False, Login(request).login().body + raise AssertionError(Login(request).login().body) url = exc_info.value.headers["Location"] url_split = urllib.parse.urlsplit(url) query = dict(urllib.parse.parse_qsl(url_split.fragment)) @@ -684,7 +685,7 @@ def test_pkce_state_oauth2_protocol_test_login_get_token_refresh_token_is_login( request.method = "POST" request.body = "" with pytest.raises(pyramid.httpexceptions.HTTPFound) as exc_info: - assert False, Login(request).login().body + raise AssertionError(Login(request).login().body) url = exc_info.value.headers["Location"] url_split = urllib.parse.urlsplit(url) query = dict(urllib.parse.parse_qsl(url_split.fragment)) @@ -766,7 +767,7 @@ def test_oauth2_protocol_test_login_no_state(self) -> None: request.host = "127.0.0.1:7070" request.method = "POST" request.body = "" - with pytest.raises(pyramid.httpexceptions.HTTPBadRequest) as exc_info: + with pytest.raises(pyramid.httpexceptions.HTTPBadRequest): Login(request).login() def test_oauth2_protocol_test_login_no_pkce(self) -> None: @@ -814,7 +815,7 @@ def test_pkce_oauth2_protocol_test_login_no_state(self) -> None: request.host = "127.0.0.1:7070" request.method = "POST" request.body = "" - with pytest.raises(pyramid.httpexceptions.HTTPBadRequest) as exc_info: + with pytest.raises(pyramid.httpexceptions.HTTPBadRequest): Login(request).login() def test_pkce_oauth2_protocol_test_login_wrong_code(self) -> None: @@ -846,7 +847,7 @@ def test_pkce_oauth2_protocol_test_login_wrong_code(self) -> None: request.method = "POST" request.body = "" with pytest.raises(pyramid.httpexceptions.HTTPFound) as exc_info: - assert False, Login(request).login().body + raise AssertionError(Login(request).login().body) url = exc_info.value.headers["Location"] url_split = urllib.parse.urlsplit(url) query = dict(urllib.parse.parse_qsl(url_split.fragment)) diff --git a/geoportal/tests/functional/test_ogcproxy.py b/geoportal/tests/functional/test_ogcproxy.py index d026f48e66..7e5c92c022 100644 --- a/geoportal/tests/functional/test_ogcproxy.py +++ b/geoportal/tests/functional/test_ogcproxy.py @@ -1,4 +1,4 @@ -# Copyright (c) 2022-2023, Camptocamp SA +# Copyright (c) 2022-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,9 +31,9 @@ from unittest import TestCase import transaction -from tests.functional import cleanup_db, create_dummy_request + +from tests.functional import cleanup_db, create_dummy_request, setup_db from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa diff --git a/geoportal/tests/functional/test_oidc.py b/geoportal/tests/functional/test_oidc.py index 21e7e8d688..98326b9026 100644 --- a/geoportal/tests/functional/test_oidc.py +++ b/geoportal/tests/functional/test_oidc.py @@ -7,15 +7,14 @@ import jwt import responses +from c2cgeoportal_geoportal.lib import oidc from cryptography.hazmat.primitives.asymmetric import rsa from pyramid import testing -from tests.functional import cleanup_db, create_dummy_request + +from tests.functional import cleanup_db, create_dummy_request, setup_db from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import -from c2cgeoportal_geoportal.lib import oidc - _OIDC_CONFIGURATION = { "issuer": "https://sso.example.com", "authorization_endpoint": "https://sso.example.com/authorize", diff --git a/geoportal/tests/functional/test_request_property.py b/geoportal/tests/functional/test_request_property.py index 8b0f74bfaf..3f17926336 100644 --- a/geoportal/tests/functional/test_request_property.py +++ b/geoportal/tests/functional/test_request_property.py @@ -1,4 +1,4 @@ -# Copyright (c) 2011-2023, Camptocamp SA +# Copyright (c) 2011-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -39,7 +39,6 @@ class TestRequestProperty(TestCase): def setup_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Role from c2cgeoportal_commons.models.static import User @@ -52,7 +51,6 @@ def setup_method(self, _): def teardown_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import Role from c2cgeoportal_commons.models.static import User @@ -65,7 +63,7 @@ def teardown_method(self, _): def test_request_no_auth(self): request = create_dummy_request() - assert request.user == None + assert request.user is None def test_request_auth(self): request = create_dummy_request(authentication=False, user="__test_user") @@ -91,7 +89,7 @@ def test_request_wrong_auth(self): } ) - assert request.user == None + assert request.user is None def test_request_auth_overwritten_property(self): def setter(request): diff --git a/geoportal/tests/functional/test_shortener.py b/geoportal/tests/functional/test_shortener.py index 23b1b76189..3f27f21bfc 100644 --- a/geoportal/tests/functional/test_shortener.py +++ b/geoportal/tests/functional/test_shortener.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,6 +31,7 @@ from unittest import TestCase from pyramid import testing + from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -40,7 +41,6 @@ def teardown_method(self, _): testing.tearDown() import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.static import Shorturl @@ -48,10 +48,10 @@ def teardown_method(self, _): transaction.commit() def test_shortener(self): + from c2cgeoportal_geoportal.views.shortener import Shortener from pyramid.httpexceptions import HTTPBadRequest, HTTPNotFound - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest def route_url(name, *args, **kwargs): del name # Unused @@ -78,10 +78,10 @@ def route_url(name, *args, **kwargs): self.assertRaises(HTTPBadRequest, shortener.create) def test_shortener_create_1(self): + from c2cgeoportal_geoportal.views.shortener import Shortener from pyramid.httpexceptions import HTTPFound - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest def route_url(name, *elements, **kw): return "https://example.com/short/" + kw["ref"] @@ -106,10 +106,10 @@ def route_url(name, *elements, **kw): assert result.location == "https://example.com/hi" def test_shortener_create_2(self): + from c2cgeoportal_geoportal.views.shortener import Shortener from pyramid.httpexceptions import HTTPFound - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest def route_url(name, *elements, **kw): return "https://example.com/short/" + kw["ref"] @@ -136,10 +136,10 @@ def route_url(name, *elements, **kw): assert result.location == "https://example.com/hi" def test_shortener_noreplace_1(self): - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest + def route_url(name, *elements, **kw): return "https://example.com/short/" + kw["ref"] @@ -156,10 +156,10 @@ def route_url(name, *elements, **kw): assert result["short_url"] == "https://example.com/s/truite" def test_shortener_noreplace_2(self): - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest + def route_url(name, *elements, **kw): return "https://example.com/short/" + kw["ref"] @@ -176,10 +176,10 @@ def route_url(name, *elements, **kw): assert result["short_url"] == "https://example.com/s/truite" def test_shortener_baseurl(self): - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest + def route_url(name, *elements, **kw): return "https://example.com/short/" + kw["ref"] @@ -197,10 +197,10 @@ def route_url(name, *elements, **kw): assert result["short_url"][:index] == "http://my_host/my_short" def test_shortener_dev(self): - from tests import DummyRequest - from c2cgeoportal_geoportal.views.shortener import Shortener + from tests import DummyRequest + def route_url(name, *elements, **kw): return "https://localhost:8484/s/" + kw["ref"] diff --git a/geoportal/tests/functional/test_theme2fts.py b/geoportal/tests/functional/test_theme2fts.py index 302fd93cc7..82d0ca8405 100644 --- a/geoportal/tests/functional/test_theme2fts.py +++ b/geoportal/tests/functional/test_theme2fts.py @@ -32,7 +32,7 @@ import pytest from c2c.template.config import config as configuration -from sqlalchemy import func + from tests.functional import setup_common as setup_module @@ -54,9 +54,7 @@ def settings(): def add_parent(dbsession_old, item, group): - """ - Utility function to add a TreeItem in a TreeGroup. - """ + """Utility function to add a TreeItem in a TreeGroup.""" from c2cgeoportal_commons.models import main dbsession_old.add(main.LayergroupTreeitem(group=group, item=item, ordering=len(group.children_relation))) diff --git a/geoportal/tests/functional/test_themes.py b/geoportal/tests/functional/test_themes.py index 9261669204..6796c19217 100644 --- a/geoportal/tests/functional/test_themes.py +++ b/geoportal/tests/functional/test_themes.py @@ -29,8 +29,10 @@ import pytest -import transaction + from tests.functional import create_dummy_request +from tests.functional import setup_common as setup_module # noqa +from tests.functional import teardown_common as teardown_module # noqa @pytest.fixture() diff --git a/geoportal/tests/functional/test_themes_dimensions.py b/geoportal/tests/functional/test_themes_dimensions.py index 4fca063143..aec8eaf697 100644 --- a/geoportal/tests/functional/test_themes_dimensions.py +++ b/geoportal/tests/functional/test_themes_dimensions.py @@ -33,6 +33,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -152,7 +153,12 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Dimension, Interface, OGCServer, TreeItem + from c2cgeoportal_commons.models.main import ( + Dimension, + Interface, + OGCServer, + TreeItem, + ) DBSession.query(Dimension).delete() for item in DBSession.query(TreeItem).all(): diff --git a/geoportal/tests/functional/test_themes_edit_columns.py b/geoportal/tests/functional/test_themes_edit_columns.py index d4ee1a6e4a..595534b242 100644 --- a/geoportal/tests/functional/test_themes_edit_columns.py +++ b/geoportal/tests/functional/test_themes_edit_columns.py @@ -32,9 +32,13 @@ from unittest import TestCase from urllib.parse import urlencode -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + setup_db, +) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa if TYPE_CHECKING: @@ -54,7 +58,6 @@ def setup_method(self, _: Any) -> None: self._tables = [] import transaction - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import ( OGCSERVER_AUTH_NOAUTH, @@ -93,7 +96,6 @@ def setup_method(self, _: Any) -> None: def teardown_method(self, _: Any) -> None: import transaction - from c2cgeoportal_commons.models import DBSession cleanup_db() @@ -115,13 +117,12 @@ def _create_layer( It creates a layer with two features, and associates a restriction area to it. """ import transaction + from c2cgeoportal_commons.models import DBSession + from c2cgeoportal_commons.models.main import LayerWMS, RestrictionArea from geoalchemy2 import Geometry from sqlalchemy import Column, ForeignKey, Table, types from sqlalchemy.ext.declarative import declarative_base - from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import LayerWMS, RestrictionArea - self.__class__._table_index += 1 id = self.__class__._table_index @@ -224,7 +225,6 @@ def _get_request(layerid, username=None, params=None) -> None: return request def test_themes_edit_columns(self) -> None: - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_geoportal.views.theme import Theme layer_id = self._create_layer(geom_type=True) @@ -297,7 +297,7 @@ def test_themes_edit_columns_extras(self) -> None: theme_view = Theme(self._get_request(layer_id, username="__test_user", params={"interface": "main"})) themes = theme_view.themes() - assert [] == themes["errors"] + assert themes["errors"] == [] layers = themes["themes"][0]["children"][0]["children"] self.assertEqual( diff --git a/geoportal/tests/functional/test_themes_editing.py b/geoportal/tests/functional/test_themes_editing.py index 7a3bb496fe..59187db73a 100644 --- a/geoportal/tests/functional/test_themes_editing.py +++ b/geoportal/tests/functional/test_themes_editing.py @@ -33,9 +33,15 @@ import transaction from geoalchemy2 import WKTElement from pyramid import testing -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request, mapserv_url + +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + mapserv_url, + setup_db, +) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa @@ -46,10 +52,6 @@ def setup_method(self, _): self.maxDiff = None self._tables = [] - from geoalchemy2 import Geometry - from sqlalchemy import Column, Table, types - from sqlalchemy.ext.declarative import declarative_base - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import ( Interface, @@ -60,6 +62,9 @@ def setup_method(self, _): Theme, ) from c2cgeoportal_commons.models.static import User + from geoalchemy2 import Geometry + from sqlalchemy import Column, Table, types + from sqlalchemy.ext.declarative import declarative_base setup_db() diff --git a/geoportal/tests/functional/test_themes_entry.py b/geoportal/tests/functional/test_themes_entry.py index cd702bd676..2dad788b78 100644 --- a/geoportal/tests/functional/test_themes_entry.py +++ b/geoportal/tests/functional/test_themes_entry.py @@ -32,14 +32,19 @@ from unittest import TestCase import transaction +from c2cgeoportal_geoportal.lib.caching import invalidate_region from geoalchemy2 import WKTElement from pyramid import testing -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request, mapserv_url -from tests.functional import setup_common as setup_module # noqa, pylint: disable=unused-import -from tests.functional import setup_db -from tests.functional import teardown_common as teardown_module # noqa, pylint: disable=unused-import -from c2cgeoportal_geoportal.lib.caching import invalidate_region +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + mapserv_url, + setup_db, +) +from tests.functional import setup_common as setup_module # pylint: disable=unused-import +from tests.functional import teardown_common as teardown_module # pylint: disable=unused-import _LOG = logging.getLogger(__name__) @@ -51,10 +56,6 @@ def setup_method(self, _): self.maxDiff = None # pylint: disable=invalid-name self._tables = [] - from geoalchemy2 import Geometry - from sqlalchemy import Column, Table, func, types - from sqlalchemy.ext.declarative import declarative_base - from c2cgeoportal_commons.models import DBSession from c2cgeoportal_commons.models.main import ( OGCSERVER_AUTH_GEOSERVER, @@ -70,6 +71,9 @@ def setup_method(self, _): Theme, ) from c2cgeoportal_commons.models.static import User + from geoalchemy2 import Geometry + from sqlalchemy import Column, Table, func, types + from sqlalchemy.ext.declarative import declarative_base setup_db() diff --git a/geoportal/tests/functional/test_themes_layermultinameerror.py b/geoportal/tests/functional/test_themes_layermultinameerror.py index c9c4f7f553..75be46c4dd 100644 --- a/geoportal/tests/functional/test_themes_layermultinameerror.py +++ b/geoportal/tests/functional/test_themes_layermultinameerror.py @@ -32,6 +32,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -44,7 +45,12 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Theme, + ) main = Interface(name="desktop") diff --git a/geoportal/tests/functional/test_themes_loop.py b/geoportal/tests/functional/test_themes_loop.py index 2731c10edc..8186a58e30 100644 --- a/geoportal/tests/functional/test_themes_loop.py +++ b/geoportal/tests/functional/test_themes_loop.py @@ -32,6 +32,7 @@ import transaction from pyramid import testing + from tests import DummyRequest from tests.functional import create_default_ogcserver, mapserv_url from tests.functional import setup_common as setup_module # noqa @@ -45,7 +46,12 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Theme, + ) ogc_server = create_default_ogcserver(DBSession) main = Interface(name="desktop2") @@ -69,7 +75,12 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import LayerGroup, LayerWMS, OGCServer, Theme + from c2cgeoportal_commons.models.main import ( + LayerGroup, + LayerWMS, + OGCServer, + Theme, + ) for t in DBSession.query(Theme).filter(Theme.name == "__test_theme").all(): DBSession.delete(t) diff --git a/geoportal/tests/functional/test_themes_metadata.py b/geoportal/tests/functional/test_themes_metadata.py index d1e262b98a..34330b99bb 100644 --- a/geoportal/tests/functional/test_themes_metadata.py +++ b/geoportal/tests/functional/test_themes_metadata.py @@ -33,6 +33,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -45,7 +46,13 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Metadata, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Metadata, + Theme, + ) desktop = Interface(name="desktop") diff --git a/geoportal/tests/functional/test_themes_mixed.py b/geoportal/tests/functional/test_themes_mixed.py index fb920569d3..3a678a3097 100644 --- a/geoportal/tests/functional/test_themes_mixed.py +++ b/geoportal/tests/functional/test_themes_mixed.py @@ -33,6 +33,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -152,7 +153,13 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Dimension, Interface, Metadata, OGCServer, TreeItem + from c2cgeoportal_commons.models.main import ( + Dimension, + Interface, + Metadata, + OGCServer, + TreeItem, + ) DBSession.query(Metadata).delete() DBSession.query(Dimension).delete() diff --git a/geoportal/tests/functional/test_themes_nameerror.py b/geoportal/tests/functional/test_themes_nameerror.py index e309fc7c09..316c46b5af 100644 --- a/geoportal/tests/functional/test_themes_nameerror.py +++ b/geoportal/tests/functional/test_themes_nameerror.py @@ -32,6 +32,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -44,7 +45,12 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Theme, + ) main = Interface(name="desktop") @@ -69,7 +75,13 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, Layer, LayerGroup, OGCServer, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + Layer, + LayerGroup, + OGCServer, + Theme, + ) for layer in DBSession.query(Layer).all(): DBSession.delete(layer) diff --git a/geoportal/tests/functional/test_themes_ogc_server_cache_clean.py b/geoportal/tests/functional/test_themes_ogc_server_cache_clean.py index 9ad8151e65..6f75bba8e5 100644 --- a/geoportal/tests/functional/test_themes_ogc_server_cache_clean.py +++ b/geoportal/tests/functional/test_themes_ogc_server_cache_clean.py @@ -34,13 +34,13 @@ import pytest import responses import transaction +from c2cgeoportal_geoportal.lib import caching from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa -from c2cgeoportal_geoportal.lib import caching - CAP = """ @@ -215,7 +215,11 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Dimension, Interface, Metadata, OGCServer, TreeItem + from c2cgeoportal_commons.models.main import ( + Interface, + OGCServer, + TreeItem, + ) for item in DBSession.query(TreeItem).all(): DBSession.delete(item) diff --git a/geoportal/tests/functional/test_themes_private.py b/geoportal/tests/functional/test_themes_private.py index 6a58695601..e188c556f2 100644 --- a/geoportal/tests/functional/test_themes_private.py +++ b/geoportal/tests/functional/test_themes_private.py @@ -31,6 +31,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -120,7 +121,13 @@ def tearDown(self): # noqa @staticmethod def clean(): from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, OGCServer, RestrictionArea, Role, TreeItem + from c2cgeoportal_commons.models.main import ( + Interface, + OGCServer, + RestrictionArea, + Role, + TreeItem, + ) from c2cgeoportal_commons.models.static import User for obj in DBSession.query(RestrictionArea).all(): diff --git a/geoportal/tests/functional/test_themes_scale.py b/geoportal/tests/functional/test_themes_scale.py index 87658ae1dd..46e9b52dfc 100644 --- a/geoportal/tests/functional/test_themes_scale.py +++ b/geoportal/tests/functional/test_themes_scale.py @@ -31,6 +31,7 @@ import transaction from pyramid import testing + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -43,7 +44,13 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, Metadata, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + Metadata, + Theme, + ) main = Interface(name="desktop") @@ -96,7 +103,13 @@ def teardown_method(self, _): testing.tearDown() from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, Layer, LayerGroup, OGCServer, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + Layer, + LayerGroup, + OGCServer, + Theme, + ) for layer in DBSession.query(Layer).all(): DBSession.delete(layer) diff --git a/geoportal/tests/functional/test_themes_time.py b/geoportal/tests/functional/test_themes_time.py index 1d8c3ce765..946d1d1a85 100644 --- a/geoportal/tests/functional/test_themes_time.py +++ b/geoportal/tests/functional/test_themes_time.py @@ -38,6 +38,7 @@ from pyramid import testing from sqlalchemy import Column from sqlalchemy.types import DateTime, Integer, Unicode + from tests.functional import create_default_ogcserver, create_dummy_request, mapserv_url from tests.functional import setup_common as setup_module # noqa from tests.functional import teardown_common as teardown_module # noqa @@ -61,7 +62,13 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerGroup, LayerWMS, LayerWMTS, Theme + from c2cgeoportal_commons.models.main import ( + Interface, + LayerGroup, + LayerWMS, + LayerWMTS, + Theme, + ) DBSession.query(PointTest).delete() diff --git a/geoportal/tests/functional/test_tinyowsproxy.py b/geoportal/tests/functional/test_tinyowsproxy.py index defaacd638..bc3ca1827a 100644 --- a/geoportal/tests/functional/test_tinyowsproxy.py +++ b/geoportal/tests/functional/test_tinyowsproxy.py @@ -33,11 +33,15 @@ import responses import transaction from geoalchemy2 import WKTElement -from pyramid.response import Response + from tests import load_file -from tests.functional import cleanup_db, create_default_ogcserver, create_dummy_request +from tests.functional import ( + cleanup_db, + create_default_ogcserver, + create_dummy_request, + setup_db, +) from tests.functional import setup_common as setup_module # noqa -from tests.functional import setup_db from tests.functional import teardown_common as teardown_module # noqa @@ -74,7 +78,12 @@ def setup_method(self, _): self.maxDiff = None from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_commons.models.main import Interface, LayerWMS, RestrictionArea, Role + from c2cgeoportal_commons.models.main import ( + Interface, + LayerWMS, + RestrictionArea, + Role, + ) from c2cgeoportal_commons.models.static import User setup_db() @@ -124,9 +133,8 @@ def teardown_method(self, _): cleanup_db() def test_proxy_not_auth(self): - from pyramid.httpexceptions import HTTPUnauthorized - from c2cgeoportal_geoportal.views.tinyowsproxy import TinyOWSProxy + from pyramid.httpexceptions import HTTPUnauthorized request = _create_dummy_request() @@ -150,7 +158,7 @@ def test_proxy_get_capabilities_user1(self): load_file(self.capabilities_response_filtered_file1).strip(), response.body.decode().replace(" \n", "").strip(), ) - assert "200 OK" == response.status + assert response.status == "200 OK" @responses.activate def test_proxy_get_capabilities_user2(self): @@ -169,7 +177,7 @@ def test_proxy_get_capabilities_user2(self): load_file(self.capabilities_response_filtered_file2).strip(), response.body.decode().replace(" \n", "").strip(), ) - assert "200 OK" == response.status + assert response.status == "200 OK" @responses.activate def test_proxy_get_capabilities_get(self): @@ -189,7 +197,7 @@ def test_proxy_get_capabilities_get(self): load_file(self.capabilities_response_filtered_file1).strip(), response.body.decode().replace(" \n", "").strip(), ) - assert "200 OK" == response.status + assert response.status == "200 OK" @responses.activate def test_proxy_get_capabilities_post(self): @@ -211,12 +219,11 @@ def test_proxy_get_capabilities_post(self): response.body.decode().replace(" \n", "").strip(), ) - assert "200 OK" == response.status + assert response.status == "200 OK" def test_proxy_get_capabilities_post_invalid_body(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.tinyowsproxy import TinyOWSProxy + from pyramid.httpexceptions import HTTPBadRequest request = _create_dummy_request(username="__test_user1") request.method = "POST" @@ -248,12 +255,11 @@ def test_proxy_describe_feature_type_get(self): response = TinyOWSProxy(request).proxy() - assert "200 OK" == response.status + assert response.status == "200 OK" def test_proxy_describe_feature_type_invalid_layer(self): - from pyramid.httpexceptions import HTTPForbidden - from c2cgeoportal_geoportal.views.tinyowsproxy import TinyOWSProxy + from pyramid.httpexceptions import HTTPForbidden request = _create_dummy_request(username="__test_user1") request.params.update( @@ -278,12 +284,11 @@ def test_proxy_describe_feature_type_post(self): response = TinyOWSProxy(request).proxy() - assert "200 OK" == response.status + assert response.status == "200 OK" def test_proxy_describe_feature_type_post_multiple_types(self): - from pyramid.httpexceptions import HTTPBadRequest - from c2cgeoportal_geoportal.views.tinyowsproxy import TinyOWSProxy + from pyramid.httpexceptions import HTTPBadRequest request = _create_dummy_request(username="__test_user1") request.method = "POST" @@ -324,7 +329,7 @@ def test_parse_body_getcapabilities(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "getcapabilities" == operation + assert operation == "getcapabilities" assert set() == typename def test_parse_body_describefeaturetype(self): @@ -335,7 +340,7 @@ def test_parse_body_describefeaturetype(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "describefeaturetype" == operation + assert operation == "describefeaturetype" assert {"layer_1"} == typename def test_parse_body_getfeature(self): @@ -346,7 +351,7 @@ def test_parse_body_getfeature(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "getfeature" == operation + assert operation == "getfeature" assert {"parks"} == typename def test_parse_body_lockfeature(self): @@ -357,7 +362,7 @@ def test_parse_body_lockfeature(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "lockfeature" == operation + assert operation == "lockfeature" assert {"parks"} == typename def test_parse_body_transaction_update(self): @@ -368,7 +373,7 @@ def test_parse_body_transaction_update(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "transaction" == operation + assert operation == "transaction" assert {"parks"} == typename def test_parse_body_transaction_delete(self): @@ -379,7 +384,7 @@ def test_parse_body_transaction_delete(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "transaction" == operation + assert operation == "transaction" assert {"parks"} == typename def test_parse_body_transaction_insert(self): @@ -390,5 +395,5 @@ def test_parse_body_transaction_insert(self): (operation, typename) = TinyOWSProxy(request)._parse_body(request.body) - assert "transaction" == operation + assert operation == "transaction" assert {"parks"} == typename diff --git a/geoportal/tests/functional/test_vector_tiles.py b/geoportal/tests/functional/test_vector_tiles.py index ddd038abc4..2ebf9b77a5 100644 --- a/geoportal/tests/functional/test_vector_tiles.py +++ b/geoportal/tests/functional/test_vector_tiles.py @@ -31,6 +31,7 @@ import pytest from geoalchemy2 import WKTElement from pyramid.httpexceptions import HTTPNotFound + from tests.functional.geodata_model import PointTest @@ -110,6 +111,6 @@ def test_vector_tiles_layer_not_found(self, dummy_request, test_data): request.matchdict["x"] = 0 request.matchdict["y"] = 0 - with pytest.raises(HTTPNotFound) as excinfo: + with pytest.raises(HTTPNotFound): VectorTilesViews(request).vector_tiles() assert "Not found any vector tile layer named not_existing_layer_name" diff --git a/geoportal/tests/functional/test_xsd.py b/geoportal/tests/functional/test_xsd.py index 18d65c9cb3..cb8bab0cb1 100644 --- a/geoportal/tests/functional/test_xsd.py +++ b/geoportal/tests/functional/test_xsd.py @@ -42,13 +42,12 @@ def setup_method(self, _): setup_module() import transaction + from c2cgeoportal_commons.models import DBSession + from c2cgeoportal_geoportal.lib.dbreflection import _AssociationProxy from sqlalchemy import Column, ForeignKey, types from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import relationship - from c2cgeoportal_commons.models import DBSession - from c2cgeoportal_geoportal.lib.dbreflection import _AssociationProxy - # Always see the diff # https://docs.python.org/2/library/unittest.html#unittest.TestCase.maxDiff self.maxDiff = None @@ -90,7 +89,6 @@ class Parent(Base): # type: ignore def teardown_method(self, _): import transaction - from c2cgeoportal_commons.models import DBSession transaction.commit() @@ -116,9 +114,8 @@ def test_add_class_properties_xsd_column_order(self, column_mock): @patch("c2cgeoportal_geoportal.lib.xsd.XSDGenerator.add_association_proxy_xsd") @patch("c2cgeoportal_geoportal.lib.xsd.PapyrusXSDGenerator.add_column_property_xsd") def test_add_column_property_xsd(self, column_mock, proxy_mock): - from sqlalchemy.orm.util import class_mapper - from c2cgeoportal_geoportal.lib.xsd import XSDGenerator + from sqlalchemy.orm.util import class_mapper gen = XSDGenerator(include_foreign_keys=True) @@ -136,9 +133,8 @@ def test_add_column_property_xsd(self, column_mock, proxy_mock): def test_add_column_readonly(self): from xml.etree.ElementTree import TreeBuilder, tostring - from sqlalchemy.orm.util import class_mapper - from c2cgeoportal_geoportal.lib.xsd import XSDGenerator + from sqlalchemy.orm.util import class_mapper gen = XSDGenerator(include_foreign_keys=True) mapper = class_mapper(self.cls) @@ -162,9 +158,8 @@ def test_add_column_readonly(self): def test_add_association_proxy_xsd(self): from xml.etree.ElementTree import TreeBuilder, tostring - from sqlalchemy.orm.util import class_mapper - from c2cgeoportal_geoportal.lib.xsd import XSDGenerator + from sqlalchemy.orm.util import class_mapper gen = XSDGenerator(include_foreign_keys=True) diff --git a/geoportal/tests/test_cachebuster.py b/geoportal/tests/test_cachebuster.py index ff8017e735..fd379b22a4 100644 --- a/geoportal/tests/test_cachebuster.py +++ b/geoportal/tests/test_cachebuster.py @@ -1,4 +1,4 @@ -# Copyright (c) 2011-2023, Camptocamp SA +# Copyright (c) 2011-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,10 +31,10 @@ from unittest import TestCase import pyramid.registry -from tests import DummyRequest - from c2cgeoportal_geoportal.lib.caching import init_region +from tests import DummyRequest + def handler(request): return request.response diff --git a/geoportal/tests/test_caching.py b/geoportal/tests/test_caching.py index 2e5b5a5ae7..1261cdd3b7 100644 --- a/geoportal/tests/test_caching.py +++ b/geoportal/tests/test_caching.py @@ -30,11 +30,15 @@ from unittest import TestCase -from tests import DummyRequest - from c2cgeoportal_geoportal.lib.cacheversion import get_cache_version from c2cgeoportal_geoportal.lib.caching import init_region, invalidate_region -from c2cgeoportal_geoportal.lib.common_headers import CORS_METHODS, Cache, set_common_headers +from c2cgeoportal_geoportal.lib.common_headers import ( + CORS_METHODS, + Cache, + set_common_headers, +) + +from tests import DummyRequest class TestSetCorsHeaders(TestCase): @@ -55,9 +59,7 @@ def _do(method, headers, credentials=False, settings=SETTINGS): return dict(request.response.headers) def test_simple(self): - """ - Tests specified in http://www.w3.org/TR/cors/#resource-requests. - """ + """Tests specified in http://www.w3.org/TR/cors/#resource-requests.""" # 1. If the Origin header is not present terminate this set of steps. # The request is outside the scope of this specification. assert self._do("POST", {}) == { @@ -102,9 +104,7 @@ def test_simple(self): # Not implemented def test_preflight(self): - """ - Tests specified in http://www.w3.org/TR/cors/#resource-preflight-requests. - """ + """Tests specified in http://www.w3.org/TR/cors/#resource-preflight-requests.""" # 1. If the Origin header is not present terminate this set of steps. # The request is outside the scope of this specification. assert self._do("OPTIONS", {"Access-Control-Request-Method": "GET"}) == { diff --git a/geoportal/tests/test_checker.py b/geoportal/tests/test_checker.py index adcef25660..fb1b8fed19 100644 --- a/geoportal/tests/test_checker.py +++ b/geoportal/tests/test_checker.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -30,10 +30,10 @@ from unittest import TestCase -from tests import DummyRequest - from c2cgeoportal_geoportal.lib.checker import build_url +from tests import DummyRequest + class TestExportCSVView(TestCase): def test_build_url_docker(self): diff --git a/geoportal/tests/test_headerstween.py b/geoportal/tests/test_headerstween.py index d8a815da88..64fe41d351 100644 --- a/geoportal/tests/test_headerstween.py +++ b/geoportal/tests/test_headerstween.py @@ -1,4 +1,4 @@ -# Copyright (c) 2018-2023, Camptocamp SA +# Copyright (c) 2018-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,6 +31,7 @@ from unittest import TestCase import pyramid.registry + from tests import DummyRequest diff --git a/geoportal/tests/test_init.py b/geoportal/tests/test_init.py index 996a5ceb5f..1c94ef14da 100644 --- a/geoportal/tests/test_init.py +++ b/geoportal/tests/test_init.py @@ -31,12 +31,9 @@ from unittest import TestCase from unittest.mock import patch +import c2cgeoportal_geoportal import pytest from c2c.template.config import config -from pyramid import testing -from tests import DummyRequest - -import c2cgeoportal_geoportal from c2cgeoportal_geoportal import ( call_hook, create_get_user_from_request, @@ -44,6 +41,9 @@ is_valid_referrer, set_user_validator, ) +from pyramid import testing + +from tests import DummyRequest class TestIncludeme(TestCase): @@ -127,9 +127,7 @@ def test_is_valid_referrer(authorized, value, expected): class TestReferer(TestCase): - """ - Check that accessing something with a bad HTTP referrer is equivalent to a not authenticated query. - """ + """Check that accessing something with a bad HTTP referrer is equivalent to a not authenticated query.""" BASE1 = "http://example.com/app" BASE2 = "http://friend.com/app2" diff --git a/geoportal/tests/test_locale_negociator.py b/geoportal/tests/test_locale_negociator.py index a26b8f8b3a..5bb76ca24c 100644 --- a/geoportal/tests/test_locale_negociator.py +++ b/geoportal/tests/test_locale_negociator.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -42,11 +42,10 @@ def test_lang_param(self): assert lang == "fr" def test_lang_is_not_available(self): + from c2cgeoportal_geoportal import locale_negotiator from pyramid.request import Request from pyramid.threadlocal import get_current_registry - from c2cgeoportal_geoportal import locale_negotiator - request = Request.blank("/") request.registry = get_current_registry() request.registry.settings = {"default_locale_name": "de", "available_locale_names": ["de", "es"]} @@ -56,11 +55,10 @@ def test_lang_is_not_available(self): assert lang == "de" def test_lang_is_available(self): + from c2cgeoportal_geoportal import locale_negotiator from pyramid.request import Request from pyramid.threadlocal import get_current_registry - from c2cgeoportal_geoportal import locale_negotiator - request = Request.blank("/") request.registry = get_current_registry() request.registry.settings = {"default_locale_name": "de", "available_locale_names": ["de", "es"]} diff --git a/geoportal/tests/test_mapserverproxy_route_predicate.py b/geoportal/tests/test_mapserverproxy_route_predicate.py index 3711365764..9d0ced65f5 100644 --- a/geoportal/tests/test_mapserverproxy_route_predicate.py +++ b/geoportal/tests/test_mapserverproxy_route_predicate.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -30,11 +30,10 @@ from unittest import TestCase +from c2cgeoportal_geoportal import MapserverproxyRoutePredicate from pyramid.request import Request from pyramid.threadlocal import get_current_registry -from c2cgeoportal_geoportal import MapserverproxyRoutePredicate - class TestMapserverproxyRoutePredicate(TestCase): predicate = MapserverproxyRoutePredicate(None, None) diff --git a/geoportal/tests/test_raster.py b/geoportal/tests/test_raster.py index 56b23ff3d8..d81ea9b161 100644 --- a/geoportal/tests/test_raster.py +++ b/geoportal/tests/test_raster.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -35,10 +35,10 @@ class TestRasterViews(TestCase): def test_raster(self): from decimal import Decimal + from c2cgeoportal_geoportal.views.raster import Raster from pyramid.httpexceptions import HTTPNotFound - from tests import DummyRequest - from c2cgeoportal_geoportal.views.raster import Raster + from tests import DummyRequest request = DummyRequest() request.registry.settings = { @@ -53,9 +53,9 @@ def test_raster(self): request.params["lon"] = "565000" request.params["lat"] = "218000" result = raster.raster() - assert result["dem1"] == None - assert result["dem2"] == None - assert result["dem3"] == None + assert result["dem1"] is None + assert result["dem2"] is None + assert result["dem3"] is None request.params["lon"] = "548000" request.params["lat"] = "216000" @@ -77,10 +77,10 @@ def test_raster(self): def test_raster_angle(self): from decimal import Decimal - from tests import DummyRequest - from c2cgeoportal_geoportal.views.raster import Raster + from tests import DummyRequest + request = DummyRequest() request.registry.settings = { "raster": { @@ -117,15 +117,15 @@ def test_raster_angle(self): request.params["lon"] = "547997.4" request.params["lat"] = "216003.5" result = raster.raster() - assert result["dem5"] == None + assert result["dem5"] is None def test_raster_vrt(self): from decimal import Decimal - from tests import DummyRequest - from c2cgeoportal_geoportal.views.raster import Raster + from tests import DummyRequest + request = DummyRequest() request.registry.settings = { "raster": { @@ -158,10 +158,10 @@ def test_absolute_path(self): def test_profile_json(self): from decimal import Decimal + from c2cgeoportal_geoportal.views.profile import Profile from pyramid.httpexceptions import HTTPNotFound - from tests import DummyRequest - from c2cgeoportal_geoportal.views.profile import Profile + from tests import DummyRequest request = DummyRequest() request.registry.settings = { diff --git a/geoportal/tests/test_wmstparsing.py b/geoportal/tests/test_wmstparsing.py index e4ccca04eb..9e0eb29429 100644 --- a/geoportal/tests/test_wmstparsing.py +++ b/geoportal/tests/test_wmstparsing.py @@ -1,4 +1,4 @@ -# Copyright (c) 2013-2023, Camptocamp SA +# Copyright (c) 2013-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -42,7 +42,10 @@ def test_parse_values(self): self.assertTrue(isinstance(extent, TimeExtentValue)) def test_parse_interval(self): - from c2cgeoportal_geoportal.lib.wmstparsing import TimeExtentInterval, parse_extent + from c2cgeoportal_geoportal.lib.wmstparsing import ( + TimeExtentInterval, + parse_extent, + ) extent = parse_extent(["2000/2005/P1Y"], "2002") self.assertTrue(isinstance(extent, TimeExtentInterval)) @@ -80,7 +83,10 @@ def test_merge_values(self): ) def test_merge_interval(self): - from c2cgeoportal_geoportal.lib.wmstparsing import TimeExtentInterval, parse_extent + from c2cgeoportal_geoportal.lib.wmstparsing import ( + TimeExtentInterval, + parse_extent, + ) e1 = parse_extent(["2000/2005/P1Y"], "2000/2005") e2 = parse_extent(["2006/2010/P1Y"], "2006/2010") @@ -121,35 +127,35 @@ def test_parse_date_year(self): from c2cgeoportal_geoportal.lib.wmstparsing import _parse_date date = _parse_date("2010") - assert "year" == date[0] + assert date[0] == "year" self.assertEqual(datetime.datetime(2010, 0o1, 0o1, tzinfo=isodate.UTC), date[1]) def test_parse_date_month(self): from c2cgeoportal_geoportal.lib.wmstparsing import _parse_date date = _parse_date("2010-02") - assert "month" == date[0] + assert date[0] == "month" self.assertEqual(datetime.datetime(2010, 0o2, 0o1, tzinfo=isodate.UTC), date[1]) def test_parse_date(self): from c2cgeoportal_geoportal.lib.wmstparsing import _parse_date date = _parse_date("2010-02-03") - assert "day" == date[0] + assert date[0] == "day" self.assertEqual(datetime.datetime(2010, 0o2, 0o3, tzinfo=isodate.UTC), date[1]) def test_parse_datetime(self): from c2cgeoportal_geoportal.lib.wmstparsing import _parse_date date = _parse_date("2010-02-03T12:34") - assert "second" == date[0] + assert date[0] == "second" self.assertEqual(datetime.datetime(2010, 0o2, 0o3, 12, 34, tzinfo=isodate.UTC), date[1]) def test_parse_datetime_tz(self): from c2cgeoportal_geoportal.lib.wmstparsing import _parse_date date = _parse_date("2010-02-03T12:34Z") - assert "second" == date[0] + assert date[0] == "second" self.assertEqual(datetime.datetime(2010, 0o2, 0o3, 12, 34, tzinfo=isodate.UTC), date[1]) def test_unsupported_format(self): @@ -163,13 +169,13 @@ def test_format(self): from c2cgeoportal_geoportal.lib.wmstparsing import _format_date dt = datetime.datetime(2010, 0o2, 0o1, 00, 00) - assert "2010-02-01T00:00:00Z" == _format_date(dt) + assert _format_date(dt) == "2010-02-01T00:00:00Z" def test_format_tz(self): from c2cgeoportal_geoportal.lib.wmstparsing import _format_date, _parse_date dt = _parse_date("2010-02-03T12:34:00+01:00") - assert "2010-02-03T12:34:00+01:00" == _format_date(dt[1]) + assert _format_date(dt[1]) == "2010-02-03T12:34:00+01:00" class TestParseDuration(TestCase): @@ -204,9 +210,8 @@ def test_second(self): self.assertEqual((0, 0, 0, 10), _parse_duration("PT10S")) def test_invalid(self): - from isodate import ISO8601Error - from c2cgeoportal_geoportal.lib.wmstparsing import _parse_duration + from isodate import ISO8601Error self.assertRaises(ISO8601Error, _parse_duration, "10S") diff --git a/poetry.lock b/poetry.lock index 3c1cc5d925..413a316cde 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. [[package]] name = "affine" @@ -129,13 +129,13 @@ aio = ["aiohttp (>=3.0)"] [[package]] name = "azure-storage-blob" -version = "12.23.1" +version = "12.24.0" description = "Microsoft Azure Blob Storage Client Library for Python" optional = false python-versions = ">=3.8" files = [ - {file = "azure_storage_blob-12.23.1-py3-none-any.whl", hash = "sha256:1c2238aa841d1545f42714a5017c010366137a44a0605da2d45f770174bfc6b4"}, - {file = "azure_storage_blob-12.23.1.tar.gz", hash = "sha256:a587e54d4e39d2a27bd75109db164ffa2058fe194061e5446c5a89bca918272f"}, + {file = "azure_storage_blob-12.24.0-py3-none-any.whl", hash = "sha256:4f0bb4592ea79a2d986063696514c781c9e62be240f09f6397986e01755bc071"}, + {file = "azure_storage_blob-12.24.0.tar.gz", hash = "sha256:eaaaa1507c8c363d6e1d1342bd549938fdf1adec9b1ada8658c8f5bf3aea844e"}, ] [package.dependencies] @@ -1246,13 +1246,13 @@ six = ">=1.8.0" [[package]] name = "geoalchemy2" -version = "0.15.2" +version = "0.16.0" description = "Using SQLAlchemy with Spatial Databases" optional = false python-versions = ">=3.7" files = [ - {file = "GeoAlchemy2-0.15.2-py3-none-any.whl", hash = "sha256:546455dc39f5bcdfc5b871e57d3f7546c8a6f798eb364c474200f488ace6fd32"}, - {file = "geoalchemy2-0.15.2.tar.gz", hash = "sha256:3af0272db927373e74ee3b064cdc9464ba08defdb945c51745db1b841482f5dc"}, + {file = "GeoAlchemy2-0.16.0-py3-none-any.whl", hash = "sha256:b0f27d5500ee757af4654c6262e0f834b7a843504d193653ec747ef1128d2ab5"}, + {file = "geoalchemy2-0.16.0.tar.gz", hash = "sha256:df64bb72af70daafaac3f359492c96501c37ab85ed20f9510c99cc6d02881100"}, ] [package.dependencies] @@ -2640,6 +2640,7 @@ pylint-flask = "0.6" pyroma = {version = ">=2.4", optional = true, markers = "extra == \"with-pyroma\" or extra == \"with_everything\""} PyYAML = "*" requirements-detector = ">=1.3.2" +ruff = {version = "*", optional = true, markers = "extra == \"with-ruff\" or extra == \"with_everything\""} setoptconf-tmp = ">=0.3.1,<0.4.0" toml = ">=0.10.2,<0.11.0" @@ -2654,26 +2655,29 @@ with-vulture = ["vulture (>=1.5)"] [[package]] name = "prospector-profile-duplicated" -version = "1.6.0" +version = "1.9.0" description = "Profile that can be used to disable the duplicated or conflict rules between Prospector and other tools" optional = false python-versions = "*" files = [ - {file = "prospector_profile_duplicated-1.6.0-py2.py3-none-any.whl", hash = "sha256:bf6a6aae0c7de48043b95e4d42e23ccd090c6c7115b6ee8c8ca472ffb1a2022b"}, - {file = "prospector_profile_duplicated-1.6.0.tar.gz", hash = "sha256:9c2d541076537405e8b2484cb6222276a2df17492391b6af1b192695770aab83"}, + {file = "prospector_profile_duplicated-1.9.0-py2.py3-none-any.whl", hash = "sha256:7b7a665e6fa8b44fac597d2bbef1a91de5b6f090b8d64890a5f38ee60e8473c2"}, + {file = "prospector_profile_duplicated-1.9.0.tar.gz", hash = "sha256:bb0b0d0946232d570ada944ee4a180fd142529db0978579b58e766d28f3c2a27"}, ] [[package]] name = "prospector-profile-utils" -version = "1.9.1" +version = "1.14.1" description = "Some utility Prospector profiles." optional = false -python-versions = "*" +python-versions = "<4.0,>=3.9" files = [ - {file = "prospector_profile_utils-1.9.1-py2.py3-none-any.whl", hash = "sha256:b458d8c4d59bdb1547e4630a2c6de4971946c4f0999443db6a9eef6d216b26b8"}, - {file = "prospector_profile_utils-1.9.1.tar.gz", hash = "sha256:008efa6797a85233fd8093dcb9d86f5fa5d89673e431c15cb1496a91c9b2c601"}, + {file = "prospector_profile_utils-1.14.1-py3-none-any.whl", hash = "sha256:1b7d79e4293c76f9ea5107b691c888e76933aa29d33e8f8b8750dc0aeea3b657"}, + {file = "prospector_profile_utils-1.14.1.tar.gz", hash = "sha256:5447086f9a7ddba02d8ee7322baa30c80c1376328677a6593165fab23e2e4bf2"}, ] +[package.dependencies] +prospector = ">=1.13.0" + [[package]] name = "protobuf" version = "5.29.0" @@ -3442,13 +3446,13 @@ dev = ["pytest", "pytest-cache", "pytest-cov", "ruff"] [[package]] name = "pyramid-tm" -version = "2.5" +version = "2.6" description = "A package which allows Pyramid requests to join the active transaction" optional = false -python-versions = ">=3.7" +python-versions = ">=3.9" files = [ - {file = "pyramid_tm-2.5-py2.py3-none-any.whl", hash = "sha256:6638721946e809de8b4bf3f405bd2daaaa76d58442cbdf46be30ebc259f1a354"}, - {file = "pyramid_tm-2.5.tar.gz", hash = "sha256:5c81dcecd33770f5e3596687d2be35ffc4f8ce5eda00a31acb00ae35a51430d0"}, + {file = "pyramid_tm-2.6-py3-none-any.whl", hash = "sha256:665a4ee1d6f41f0c7ffa5e54d9b01d70cd3e8e5bd76277529acbdd6b6bd6fe9e"}, + {file = "pyramid_tm-2.6.tar.gz", hash = "sha256:8148d2191285280c9a0c23e6df1018b3514b4cef02115b872dd0350a4d78709c"}, ] [package.dependencies] @@ -3964,6 +3968,33 @@ files = [ {file = "ruamel.yaml.clib-0.2.12.tar.gz", hash = "sha256:6c8fbb13ec503f99a91901ab46e0b07ae7941cd527393187039aec586fdfd36f"}, ] +[[package]] +name = "ruff" +version = "0.8.1" +description = "An extremely fast Python linter and code formatter, written in Rust." +optional = false +python-versions = ">=3.7" +files = [ + {file = "ruff-0.8.1-py3-none-linux_armv6l.whl", hash = "sha256:fae0805bd514066f20309f6742f6ee7904a773eb9e6c17c45d6b1600ca65c9b5"}, + {file = "ruff-0.8.1-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:b8a4f7385c2285c30f34b200ca5511fcc865f17578383db154e098150ce0a087"}, + {file = "ruff-0.8.1-py3-none-macosx_11_0_arm64.whl", hash = "sha256:cd054486da0c53e41e0086e1730eb77d1f698154f910e0cd9e0d64274979a209"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2029b8c22da147c50ae577e621a5bfbc5d1fed75d86af53643d7a7aee1d23871"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2666520828dee7dfc7e47ee4ea0d928f40de72056d929a7c5292d95071d881d1"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:333c57013ef8c97a53892aa56042831c372e0bb1785ab7026187b7abd0135ad5"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:288326162804f34088ac007139488dcb43de590a5ccfec3166396530b58fb89d"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b12c39b9448632284561cbf4191aa1b005882acbc81900ffa9f9f471c8ff7e26"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:364e6674450cbac8e998f7b30639040c99d81dfb5bbc6dfad69bc7a8f916b3d1"}, + {file = "ruff-0.8.1-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b22346f845fec132aa39cd29acb94451d030c10874408dbf776af3aaeb53284c"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:b2f2f7a7e7648a2bfe6ead4e0a16745db956da0e3a231ad443d2a66a105c04fa"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:adf314fc458374c25c5c4a4a9270c3e8a6a807b1bec018cfa2813d6546215540"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_i686.whl", hash = "sha256:a885d68342a231b5ba4d30b8c6e1b1ee3a65cf37e3d29b3c74069cdf1ee1e3c9"}, + {file = "ruff-0.8.1-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:d2c16e3508c8cc73e96aa5127d0df8913d2290098f776416a4b157657bee44c5"}, + {file = "ruff-0.8.1-py3-none-win32.whl", hash = "sha256:93335cd7c0eaedb44882d75a7acb7df4b77cd7cd0d2255c93b28791716e81790"}, + {file = "ruff-0.8.1-py3-none-win_amd64.whl", hash = "sha256:2954cdbe8dfd8ab359d4a30cd971b589d335a44d444b6ca2cb3d1da21b75e4b6"}, + {file = "ruff-0.8.1-py3-none-win_arm64.whl", hash = "sha256:55873cc1a473e5ac129d15eccb3c008c096b94809d693fc7053f588b67822737"}, + {file = "ruff-0.8.1.tar.gz", hash = "sha256:3583db9a6450364ed5ca3f3b4225958b24f78178908d5c4bc0f46251ccca898f"}, +] + [[package]] name = "semver" version = "3.0.2" @@ -4520,13 +4551,13 @@ urllib3 = ">=2" [[package]] name = "types-setuptools" -version = "75.3.0.20241112" +version = "75.6.0.20241126" description = "Typing stubs for setuptools" optional = false python-versions = ">=3.8" files = [ - {file = "types-setuptools-75.3.0.20241112.tar.gz", hash = "sha256:f9e1ebd17a56f606e16395c4ee4efa1cdc394b9a2a0ee898a624058b4b62ef8f"}, - {file = "types_setuptools-75.3.0.20241112-py3-none-any.whl", hash = "sha256:78cb5fef4a6056d2f37114d27da90f4655a306e4e38042d7034a8a880bc3f5dd"}, + {file = "types_setuptools-75.6.0.20241126-py3-none-any.whl", hash = "sha256:aaae310a0e27033c1da8457d4d26ac673b0c8a0de7272d6d4708e263f2ea3b9b"}, + {file = "types_setuptools-75.6.0.20241126.tar.gz", hash = "sha256:7bf25ad4be39740e469f9268b6beddda6e088891fa5a27e985c6ce68bf62ace0"}, ] [[package]] @@ -4862,4 +4893,4 @@ test = ["zope.testing"] [metadata] lock-version = "2.0" python-versions = ">=3.10,<3.13" -content-hash = "5cc5d16690ab634e8c0d47984b08a9e6d2f0c913cb84011fa9901c467df269f2" +content-hash = "80e474f923a3d4698658b1dd37c7992c113b525fac34bbd51abec1ce38526c13" diff --git a/pyproject.toml b/pyproject.toml index 503a754b1f..97a0522615 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,25 +1,5 @@ -[tool.mypy] -python_version = "3.10" -warn_redundant_casts = true -warn_unused_ignores = true -warn_return_any = true -ignore_missing_imports = true -disallow_untyped_defs = true -strict_optional = true -strict = true - -[[tool.mypy.overrides]] -module = "c2cgeoportal_admin.*" -disallow_untyped_defs = false - -[tool.black] -line-length = 110 -target-version = ['py310'] - -[tool.isort] -profile = "black" -line_length = 110 -known_local_folder = ["c2cgeoportal_commons", "c2cgeoportal_geoportal", "c2cgeoportal_admin", "geomapfish_qgisserver", "{{cookiecutter.package}}_geoportal"] +[tool.ruff] +target-version = 'py310' [tool.poetry] name = "c2cgeoportal" @@ -39,7 +19,7 @@ deform = "2.0.15" # commons, admin defusedxml = "0.7.1" # geoportal "dogpile.cache" = "1.3.3" # geoportal Fiona = "1.10.1" # geoportal raster -GeoAlchemy2 = "0.15.2" # commons, geoportal +GeoAlchemy2 = "0.16.0" # commons, geoportal geojson = "3.1.0" # geoportal getitfixed = "1.0.29" # geoportal isodate = "0.7.2" # geoportal @@ -56,7 +36,7 @@ pyramid_debugtoolbar = "4.12.1" # geoportal pyramid-jinja2 = "2.10.1" # admin pyramid_mako = "1.1.0" # geoportal pyramid_multiauth = "1.0.2" # geoportal -pyramid_tm = "2.5" # geoportal +pyramid_tm = "2.6" # geoportal python-dateutil = "2.9.0.post0" # geoportal PyYAML = "6.0.1" # geoportal rasterio = "1.4.2" # geoportal raster @@ -70,14 +50,14 @@ translationstring = "1.4" # admin c2cwsgiutils = { version = "6.1.5", extras = ["broadcast", "standard", "oauth2", "debug"] } oauthlib = "3.2.2" tilecloud = "1.12.3" # geoportal -azure-storage-blob = "12.23.1" +azure-storage-blob = "12.24.0" # simple_openid_connect = '1.0.1' # geoportal simple_openid_connect = { git = "https://github.com/sbrunner/py_simple_openid_connect.git", branch = "allows-pkce" } # geoportal pkce = '1.0.3' # geoportal basicauth = "1.0.0" -prospector = { extras = ["with_mypy", "with_bandit", "with_pyroma"], version = "1.13.3" } -prospector-profile-duplicated = "1.6.0" -prospector-profile-utils = "1.9.1" +prospector = { version = "1.13.3", extras = ["with_mypy", "with_bandit", "with_pyroma", "with_ruff"] } +prospector-profile-duplicated = "1.9.0" +prospector-profile-utils = "1.14.1" beautifulsoup4 = "4.12.3" [tool.poetry.group.dev.dependencies] @@ -94,7 +74,7 @@ types-pytz = "2024.2.0.20241003" types-pyyaml = "6.0.12.20240917" types-python-dateutil = "2.9.0.20241003" types-requests = "2.32.0.20241016" -types-setuptools = "75.3.0.20241112" +types-setuptools = "75.6.0.20241126" mappyfile = "1.0.2" # To be able to upgrade project from version <= 2.8 c2cciutils = { version = "1.5.8", extras = ["checks"] } diff --git a/scripts/get-version b/scripts/get-version index 827ea37c3a..0f1eca4af9 100755 --- a/scripts/get-version +++ b/scripts/get-version @@ -1,6 +1,6 @@ #!/usr/bin/env python3 -# Copyright (c) 2018-2023, Camptocamp SA +# Copyright (c) 2018-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -31,7 +31,7 @@ import argparse import os import re -import subprocess +import subprocess # nosec import sys import requests diff --git a/scripts/updated_version b/scripts/updated_version index 02707db4d3..11dffcd58d 100755 --- a/scripts/updated_version +++ b/scripts/updated_version @@ -29,7 +29,7 @@ import argparse import json -import subprocess +import subprocess # nosec from packaging.version import Version diff --git a/scripts/upgrade b/scripts/upgrade index c523410956..53801b87aa 100755 --- a/scripts/upgrade +++ b/scripts/upgrade @@ -1,6 +1,6 @@ #!/usr/bin/env python3 -# Copyright (c) 2019-2023, Camptocamp SA +# Copyright (c) 2019-2024, Camptocamp SA # All rights reserved. # Redistribution and use in source and binary forms, with or without @@ -32,7 +32,7 @@ import argparse import os import platform import re -import subprocess +import subprocess # nosec import sys parser = argparse.ArgumentParser(description="Upgrade the project")