django-gisserver 1.4.1__py3-none-any.whl → 2.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/METADATA +23 -13
- django_gisserver-2.0.dist-info/RECORD +66 -0
- {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/WHEEL +1 -1
- gisserver/__init__.py +1 -1
- gisserver/compat.py +23 -0
- gisserver/conf.py +7 -0
- gisserver/db.py +63 -60
- gisserver/exceptions.py +47 -9
- gisserver/extensions/__init__.py +4 -0
- gisserver/{parsers/fes20 → extensions}/functions.py +11 -5
- gisserver/extensions/queries.py +261 -0
- gisserver/features.py +267 -240
- gisserver/geometries.py +34 -39
- gisserver/management/__init__.py +0 -0
- gisserver/management/commands/__init__.py +0 -0
- gisserver/management/commands/loadgeojson.py +291 -0
- gisserver/operations/base.py +129 -305
- gisserver/operations/wfs20.py +428 -336
- gisserver/output/__init__.py +10 -48
- gisserver/output/base.py +198 -143
- gisserver/output/csv.py +81 -85
- gisserver/output/geojson.py +63 -72
- gisserver/output/gml32.py +310 -281
- gisserver/output/iters.py +207 -0
- gisserver/output/results.py +71 -30
- gisserver/output/stored.py +143 -0
- gisserver/output/utils.py +75 -154
- gisserver/output/xmlschema.py +86 -47
- gisserver/parsers/__init__.py +10 -10
- gisserver/parsers/ast.py +320 -0
- gisserver/parsers/fes20/__init__.py +15 -11
- gisserver/parsers/fes20/expressions.py +89 -50
- gisserver/parsers/fes20/filters.py +111 -43
- gisserver/parsers/fes20/identifiers.py +44 -26
- gisserver/parsers/fes20/lookups.py +144 -0
- gisserver/parsers/fes20/operators.py +336 -128
- gisserver/parsers/fes20/sorting.py +107 -34
- gisserver/parsers/gml/__init__.py +12 -11
- gisserver/parsers/gml/base.py +6 -3
- gisserver/parsers/gml/geometries.py +69 -35
- gisserver/parsers/ows/__init__.py +25 -0
- gisserver/parsers/ows/kvp.py +190 -0
- gisserver/parsers/ows/requests.py +158 -0
- gisserver/parsers/query.py +175 -0
- gisserver/parsers/values.py +26 -0
- gisserver/parsers/wfs20/__init__.py +37 -0
- gisserver/parsers/wfs20/adhoc.py +245 -0
- gisserver/parsers/wfs20/base.py +143 -0
- gisserver/parsers/wfs20/projection.py +103 -0
- gisserver/parsers/wfs20/requests.py +482 -0
- gisserver/parsers/wfs20/stored.py +192 -0
- gisserver/parsers/xml.py +249 -0
- gisserver/projection.py +357 -0
- gisserver/static/gisserver/index.css +12 -1
- gisserver/templates/gisserver/index.html +1 -1
- gisserver/templates/gisserver/service_description.html +2 -2
- gisserver/templates/gisserver/wfs/2.0.0/get_capabilities.xml +11 -11
- gisserver/templates/gisserver/wfs/feature_field.html +2 -2
- gisserver/templatetags/gisserver_tags.py +20 -0
- gisserver/types.py +375 -258
- gisserver/views.py +206 -75
- django_gisserver-1.4.1.dist-info/RECORD +0 -53
- gisserver/parsers/base.py +0 -149
- gisserver/parsers/fes20/query.py +0 -275
- gisserver/parsers/tags.py +0 -102
- gisserver/queries/__init__.py +0 -34
- gisserver/queries/adhoc.py +0 -181
- gisserver/queries/base.py +0 -146
- gisserver/queries/stored.py +0 -205
- gisserver/templates/gisserver/wfs/2.0.0/describe_stored_queries.xml +0 -20
- gisserver/templates/gisserver/wfs/2.0.0/list_stored_queries.xml +0 -14
- {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/LICENSE +0 -0
- {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/top_level.txt +0 -0
gisserver/operations/base.py
CHANGED
|
@@ -1,155 +1,93 @@
|
|
|
1
1
|
"""The base protocol to implement an operation.
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
3
|
+
The request itself is parsed by :mod:`gisserver.parsers.wfs20`, and handled here.
|
|
4
|
+
It can be seen as the "controller" that handles the actual request type.
|
|
5
|
+
|
|
6
|
+
All operations extend from an WFSOperation class.
|
|
7
|
+
This defines the metadata for the ``GetCapabilities`` call and possible output formats.
|
|
8
|
+
|
|
9
|
+
Each :class:`WFSOperation` can define a :attr:`~WFSOperation.parser_class`, or let it autodetect.
|
|
6
10
|
"""
|
|
7
11
|
|
|
8
12
|
from __future__ import annotations
|
|
9
13
|
|
|
14
|
+
import logging
|
|
10
15
|
import math
|
|
11
16
|
import re
|
|
12
|
-
|
|
13
|
-
from dataclasses import dataclass
|
|
14
|
-
from
|
|
17
|
+
import typing
|
|
18
|
+
from dataclasses import dataclass
|
|
19
|
+
from functools import cached_property
|
|
15
20
|
|
|
16
|
-
from django.core.exceptions import
|
|
21
|
+
from django.core.exceptions import SuspiciousOperation
|
|
17
22
|
from django.http import HttpResponse
|
|
18
23
|
from django.template.loader import render_to_string
|
|
19
24
|
|
|
20
|
-
from gisserver.exceptions import
|
|
21
|
-
ExternalParsingError,
|
|
22
|
-
InvalidParameterValue,
|
|
23
|
-
MissingParameterValue,
|
|
24
|
-
OperationParsingFailed,
|
|
25
|
-
)
|
|
25
|
+
from gisserver.exceptions import InvalidParameterValue
|
|
26
26
|
from gisserver.features import FeatureType
|
|
27
|
+
from gisserver.output.base import OutputRenderer
|
|
28
|
+
from gisserver.parsers import ows
|
|
29
|
+
from gisserver.parsers.values import fix_type_name
|
|
27
30
|
|
|
31
|
+
if typing.TYPE_CHECKING:
|
|
32
|
+
from gisserver.views import WFSView
|
|
33
|
+
|
|
34
|
+
logger = logging.getLogger(__name__)
|
|
28
35
|
NoneType = type(None)
|
|
36
|
+
R = typing.TypeVar("R", bound=OutputRenderer)
|
|
29
37
|
|
|
30
38
|
RE_SAFE_FILENAME = re.compile(r"\A[A-Za-z0-9]+[A-Za-z0-9.]*") # no dot at the start.
|
|
31
39
|
|
|
40
|
+
__all__ = (
|
|
41
|
+
"Parameter",
|
|
42
|
+
"OutputFormat",
|
|
43
|
+
"WFSOperation",
|
|
44
|
+
"OutputFormatMixin",
|
|
45
|
+
"XmlTemplateMixin",
|
|
46
|
+
)
|
|
47
|
+
|
|
32
48
|
|
|
33
49
|
@dataclass(frozen=True)
|
|
34
50
|
class Parameter:
|
|
35
|
-
"""The
|
|
51
|
+
"""The ``<ows:Parameter>`` tag to output in ``GetCapabilities``."""
|
|
36
52
|
|
|
37
|
-
These parameters should be defined in the
|
|
38
|
-
WFSMethod parameters attribute / get_parameters().
|
|
39
|
-
|
|
40
|
-
These fields is used to parse the incoming request.
|
|
41
|
-
The GetCapabilities method also reads the metadata of this value.
|
|
42
|
-
"""
|
|
43
|
-
|
|
44
|
-
#: Name of the parameter (using the XML casing style)
|
|
45
53
|
name: str
|
|
46
|
-
|
|
47
|
-
#: Alias name (e.g. typenames/typename)
|
|
48
|
-
alias: str | None = None
|
|
49
|
-
|
|
50
|
-
#: Whether the parameter is required
|
|
51
|
-
required: bool = False
|
|
52
|
-
|
|
53
|
-
#: Optional parser callback to convert the parameter into a Python type
|
|
54
|
-
parser: Callable[[str], Any] = None
|
|
55
|
-
|
|
56
|
-
#: Whether to list this parameter in GetCapabilities
|
|
57
|
-
in_capabilities: bool = False
|
|
58
|
-
|
|
59
|
-
#: List of allowed values (also shown in GetCapabilities)
|
|
60
|
-
allowed_values: list | tuple | set | NoneType = None
|
|
61
|
-
|
|
62
|
-
#: Default value if it's not given
|
|
63
|
-
default: Any = None
|
|
64
|
-
|
|
65
|
-
#: Overridable dict for error messages
|
|
66
|
-
error_messages: dict = field(default_factory=dict)
|
|
67
|
-
|
|
68
|
-
def value_from_query(self, KVP: dict): # noqa: C901
|
|
69
|
-
"""Parse a request variable using the type definition.
|
|
70
|
-
|
|
71
|
-
This uses the dataclass settings to parse the incoming request value.
|
|
72
|
-
"""
|
|
73
|
-
# URL-based key-value-pair parameters use uppercase.
|
|
74
|
-
kvp_name = self.name.upper()
|
|
75
|
-
value = KVP.get(kvp_name)
|
|
76
|
-
if not value and self.alias:
|
|
77
|
-
value = KVP.get(self.alias.upper())
|
|
78
|
-
|
|
79
|
-
# Check required field settings, both empty and missing value are treated the same.
|
|
80
|
-
if not value:
|
|
81
|
-
if not self.required:
|
|
82
|
-
return self.default
|
|
83
|
-
elif value is None:
|
|
84
|
-
raise MissingParameterValue(self.name, f"Missing {kvp_name} parameter")
|
|
85
|
-
else:
|
|
86
|
-
raise InvalidParameterValue(self.name, f"Empty {kvp_name} parameter")
|
|
87
|
-
|
|
88
|
-
# Allow conversion into a python object
|
|
89
|
-
if self.parser is not None:
|
|
90
|
-
try:
|
|
91
|
-
value = self.parser(value)
|
|
92
|
-
except ExternalParsingError as e:
|
|
93
|
-
raise OperationParsingFailed(
|
|
94
|
-
self.name, f"Unable to parse {kvp_name} argument: {e}"
|
|
95
|
-
) from None
|
|
96
|
-
except (TypeError, ValueError, NotImplementedError) as e:
|
|
97
|
-
# TypeError/ValueError are raised by most handlers for unexpected data
|
|
98
|
-
# The NotImplementedError can be raised by fes parsing.
|
|
99
|
-
raise InvalidParameterValue(
|
|
100
|
-
self.name, f"Invalid {kvp_name} argument: {e}"
|
|
101
|
-
) from None
|
|
102
|
-
|
|
103
|
-
# Validate against value choices
|
|
104
|
-
self.validate_value(value)
|
|
105
|
-
|
|
106
|
-
return value
|
|
107
|
-
|
|
108
|
-
def validate_value(self, value):
|
|
109
|
-
"""Validate the parsed value.
|
|
110
|
-
This method can be overwritten by a subclass if needed.
|
|
111
|
-
"""
|
|
112
|
-
if self.allowed_values is not None and value not in self.allowed_values:
|
|
113
|
-
msg = self.error_messages.get("invalid", "Invalid value for {name}: {value}")
|
|
114
|
-
raise InvalidParameterValue(self.name, msg.format(name=self.name, value=value))
|
|
54
|
+
allowed_values: list[str]
|
|
115
55
|
|
|
116
56
|
|
|
117
|
-
class
|
|
118
|
-
def value_from_query(self, KVP: dict):
|
|
119
|
-
kvp_name = self.name.upper()
|
|
120
|
-
if kvp_name in KVP:
|
|
121
|
-
raise InvalidParameterValue(self.name, f"Support for {self.name} is not implemented!")
|
|
122
|
-
return None
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
class OutputFormat:
|
|
57
|
+
class OutputFormat(typing.Generic[R]):
|
|
126
58
|
"""Declare an output format for the method.
|
|
127
59
|
|
|
128
|
-
These formats
|
|
60
|
+
These formats are used in the :meth:`get_output_formats` for
|
|
61
|
+
any WFSMethod that implements :class:`OutputFormatMixin`.
|
|
62
|
+
This also connects the output format with a :attr:`renderer_class`.
|
|
129
63
|
"""
|
|
130
64
|
|
|
131
65
|
def __init__(
|
|
132
66
|
self,
|
|
133
67
|
content_type,
|
|
134
|
-
|
|
68
|
+
*,
|
|
69
|
+
subtype=None,
|
|
70
|
+
renderer_class: type[R] | None = None, # None is special case for GetCapabilities
|
|
135
71
|
max_page_size=None,
|
|
136
72
|
title=None,
|
|
73
|
+
in_capabilities=True,
|
|
137
74
|
**extra,
|
|
138
75
|
):
|
|
139
76
|
"""
|
|
140
77
|
:param content_type: The MIME-type used in the request to select this type.
|
|
78
|
+
:param subtype: Shorter alias for the MIME-type.
|
|
141
79
|
:param renderer_class: The class that performs the output rendering.
|
|
142
|
-
If it's not given, the operation renders it's output using an XML template.
|
|
143
80
|
:param max_page_size: Used to override the ``max_page_size`` of the renderer_class.
|
|
144
|
-
:param title: A human-friendly name for
|
|
81
|
+
:param title: A human-friendly name for an HTML overview page.
|
|
82
|
+
:param in_capabilities: Whether this format needs to be advertised in GetCapabilities.
|
|
145
83
|
:param extra: Any additional key-value pairs for the definition.
|
|
146
|
-
Could include ``subtype`` as a shorter alias for the MIME-type.
|
|
147
84
|
"""
|
|
148
85
|
self.content_type = content_type
|
|
86
|
+
self.subtype = subtype
|
|
149
87
|
self.extra = extra
|
|
150
|
-
self.subtype = self.extra.get("subtype")
|
|
151
88
|
self.renderer_class = renderer_class
|
|
152
89
|
self.title = title
|
|
90
|
+
self.in_capabilities = in_capabilities
|
|
153
91
|
self._max_page_size = max_page_size
|
|
154
92
|
|
|
155
93
|
def matches(self, value):
|
|
@@ -182,174 +120,101 @@ class OutputFormat:
|
|
|
182
120
|
|
|
183
121
|
def __str__(self):
|
|
184
122
|
extra = "".join(f"; {name}={value}" for name, value in self.extra.items())
|
|
123
|
+
if self.subtype:
|
|
124
|
+
extra = f"; subtype={self.subtype}{extra}"
|
|
185
125
|
return f"{self.content_type}{extra}"
|
|
186
126
|
|
|
187
127
|
def __repr__(self):
|
|
188
128
|
return f"<OutputFormat: {self}>"
|
|
189
129
|
|
|
190
130
|
|
|
191
|
-
class
|
|
131
|
+
class WFSOperation:
|
|
192
132
|
"""Basic interface to implement an WFS method.
|
|
193
133
|
|
|
194
134
|
Each operation in this GIS-server extends from this base class. This class
|
|
195
135
|
also exposes all requires metadata for the GetCapabilities request.
|
|
196
136
|
"""
|
|
197
137
|
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
parameters: list[Parameter] = []
|
|
138
|
+
#: Optionally, mention explicitly what the parser class should be used.
|
|
139
|
+
#: Otherwise, it's automatically resolved from the registered types.
|
|
140
|
+
parser_class: type[ows.BaseOwsRequest] = None
|
|
202
141
|
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
#: Default template to use for rendering
|
|
207
|
-
xml_template_name = None
|
|
142
|
+
def __init__(self, view: WFSView, ows_request: ows.BaseOwsRequest):
|
|
143
|
+
self.view = view
|
|
144
|
+
self.ows_request = ows_request
|
|
208
145
|
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
"http://www.w3.org/XML/1998/namespace": "xml",
|
|
217
|
-
"http://www.opengis.net/wfs/2.0": "wfs",
|
|
218
|
-
"http://www.opengis.net/gml/3.2": "gml",
|
|
219
|
-
self.view.xml_namespace: "app",
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
def get_parameters(self):
|
|
223
|
-
"""Dynamically return the supported parameters for this method."""
|
|
224
|
-
parameters = [
|
|
225
|
-
# Always add SERVICE and VERSION as required parameters
|
|
226
|
-
# Part of BaseRequest:
|
|
227
|
-
Parameter(
|
|
228
|
-
"service",
|
|
229
|
-
required=not bool(self.view.default_service),
|
|
230
|
-
in_capabilities=True,
|
|
231
|
-
allowed_values=list(self.view.accept_operations.keys()),
|
|
232
|
-
default=self.view.default_service, # can be None or e.g. "WFS"
|
|
233
|
-
error_messages={"invalid": "Unsupported service type: {value}."},
|
|
234
|
-
),
|
|
235
|
-
Parameter(
|
|
236
|
-
"version",
|
|
237
|
-
required=True, # Mandatory except for GetCapabilities
|
|
238
|
-
allowed_values=self.view.accept_versions,
|
|
239
|
-
error_messages={
|
|
240
|
-
"invalid": "WFS Server does not support VERSION {value}.",
|
|
241
|
-
},
|
|
242
|
-
),
|
|
243
|
-
] + self.parameters
|
|
244
|
-
|
|
245
|
-
if self.output_formats:
|
|
246
|
-
parameters += [
|
|
247
|
-
# Part of StandardPresentationParameters:
|
|
248
|
-
Parameter(
|
|
249
|
-
"outputFormat",
|
|
250
|
-
in_capabilities=True,
|
|
251
|
-
allowed_values=self.output_formats,
|
|
252
|
-
parser=self._parse_output_format,
|
|
253
|
-
default=self.output_formats[0],
|
|
254
|
-
)
|
|
255
|
-
]
|
|
256
|
-
|
|
257
|
-
return parameters
|
|
146
|
+
def get_parameters(self) -> list[Parameter]:
|
|
147
|
+
"""Parameters to advertise in the capabilities for this method."""
|
|
148
|
+
return [
|
|
149
|
+
# Always advertise SERVICE and VERSION as required parameters:
|
|
150
|
+
Parameter("service", allowed_values=list(self.view.accept_operations.keys())),
|
|
151
|
+
Parameter("version", allowed_values=self.view.accept_versions),
|
|
152
|
+
]
|
|
258
153
|
|
|
259
|
-
def
|
|
260
|
-
"""
|
|
261
|
-
# When using ?OUTPUTFORMAT=application/gml+xml", it is actually an URL-encoded space
|
|
262
|
-
# character. Hence spaces are replaced back to a '+' character to allow such notation
|
|
263
|
-
# instead of forcing it to to be ?OUTPUTFORMAT=application/gml%2bxml".
|
|
264
|
-
for v in {value, value.replace(" ", "+")}:
|
|
265
|
-
for o in self.output_formats:
|
|
266
|
-
if o.matches(v):
|
|
267
|
-
return o
|
|
268
|
-
raise InvalidParameterValue(
|
|
269
|
-
"outputformat",
|
|
270
|
-
f"'{value}' is not a permitted output format for this operation.",
|
|
271
|
-
) from None
|
|
154
|
+
def validate_request(self, ows_request: ows.BaseOwsRequest):
|
|
155
|
+
"""Validate the request."""
|
|
272
156
|
|
|
273
|
-
def
|
|
274
|
-
"""
|
|
157
|
+
def process_request(self, ows_request: ows.BaseOwsRequest):
|
|
158
|
+
"""Default call implementation: render an XML template."""
|
|
159
|
+
raise NotImplementedError()
|
|
275
160
|
|
|
276
|
-
|
|
277
|
-
|
|
161
|
+
@cached_property
|
|
162
|
+
def all_feature_types_by_name(self) -> dict[str, FeatureType]:
|
|
163
|
+
"""Create a lookup for feature types by name.
|
|
164
|
+
This can be cached as the WFSMethod is instantiated with each request
|
|
278
165
|
"""
|
|
279
|
-
|
|
280
|
-
return {}
|
|
281
|
-
|
|
282
|
-
# example single value: xmlns(http://example.org)
|
|
283
|
-
# or: namespaces=xmlns(xml,http://www.w3.org/...),xmlns(wfs,http://www.opengis.net/...)
|
|
284
|
-
tokens = value.split(",")
|
|
285
|
-
|
|
286
|
-
namespaces = {}
|
|
287
|
-
tokens = iter(tokens)
|
|
288
|
-
for prefix in tokens:
|
|
289
|
-
if not prefix.startswith("xmlns("):
|
|
290
|
-
raise InvalidParameterValue("namespaces", f"Expected xmlns(...) format: {value}")
|
|
291
|
-
if prefix.endswith(")"):
|
|
292
|
-
# xmlns(http://...)
|
|
293
|
-
prefix = ""
|
|
294
|
-
uri = prefix[6:-1]
|
|
295
|
-
else:
|
|
296
|
-
uri = next(tokens, "")
|
|
297
|
-
if not uri.endswith(")"):
|
|
298
|
-
raise InvalidParameterValue(
|
|
299
|
-
"namespaces", f"Expected xmlns(prefix,uri) format: {value}"
|
|
300
|
-
)
|
|
301
|
-
prefix = prefix[6:]
|
|
302
|
-
uri = uri[:-1]
|
|
303
|
-
|
|
304
|
-
namespaces[uri] = prefix
|
|
305
|
-
|
|
306
|
-
return namespaces
|
|
307
|
-
|
|
308
|
-
def parse_request(self, KVP: dict) -> dict[str, Any]:
|
|
309
|
-
"""Parse the parameters of the request"""
|
|
310
|
-
self.namespaces.update(self._parse_namespaces(KVP.get("NAMESPACES")))
|
|
311
|
-
param_values = {param.name: param.value_from_query(KVP) for param in self.get_parameters()}
|
|
312
|
-
param_values["NAMESPACES"] = self.namespaces
|
|
166
|
+
return {ft.xml_name: ft for ft in self.view.get_bound_feature_types()}
|
|
313
167
|
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
168
|
+
def resolve_feature_type(self, type_name: str, locator: str = "typeNames") -> FeatureType:
|
|
169
|
+
"""Find the FeatureType defined in the application that corresponds with the XML type name."""
|
|
170
|
+
alt_type_name = fix_type_name(type_name, self.view.xml_namespace)
|
|
171
|
+
try:
|
|
172
|
+
return self.all_feature_types_by_name[alt_type_name]
|
|
173
|
+
except KeyError:
|
|
174
|
+
if alt_type_name:
|
|
175
|
+
logger.debug(
|
|
176
|
+
"Unable to locate '%s' (nor %s) in the server, options are: %r",
|
|
177
|
+
type_name,
|
|
178
|
+
alt_type_name,
|
|
179
|
+
list(self.all_feature_types_by_name),
|
|
180
|
+
)
|
|
181
|
+
else:
|
|
182
|
+
logger.debug(
|
|
183
|
+
"Unable to locate '%s' in the server, options are: %r",
|
|
184
|
+
type_name,
|
|
185
|
+
list(self.all_feature_types_by_name),
|
|
186
|
+
)
|
|
187
|
+
raise InvalidParameterValue(
|
|
188
|
+
f"Typename '{type_name}' doesn't exist in this server.",
|
|
189
|
+
locator=locator or "typeNames",
|
|
190
|
+
) from None
|
|
321
191
|
|
|
322
|
-
self.validate(**param_values)
|
|
323
|
-
return param_values
|
|
324
192
|
|
|
325
|
-
|
|
326
|
-
|
|
193
|
+
class XmlTemplateMixin:
|
|
194
|
+
"""Mixin to support methods that render using a template."""
|
|
327
195
|
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
context = self.get_context_data(**params)
|
|
331
|
-
if "outputFormat" in params:
|
|
332
|
-
output_format: OutputFormat = params["outputFormat"]
|
|
196
|
+
#: Default template to use for rendering
|
|
197
|
+
xml_template_name = None
|
|
333
198
|
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
renderer = output_format.renderer_class(self, **context)
|
|
337
|
-
return renderer.get_response()
|
|
199
|
+
#: Default content-type for render_xml()
|
|
200
|
+
xml_content_type = "text/xml; charset=utf-8"
|
|
338
201
|
|
|
339
|
-
|
|
202
|
+
def process_request(self, ows_request: ows.BaseOwsRequest):
|
|
203
|
+
context = self.get_context_data()
|
|
204
|
+
return self.render_xml(context, ows_request)
|
|
340
205
|
|
|
341
|
-
def get_context_data(self
|
|
206
|
+
def get_context_data(self):
|
|
342
207
|
"""Collect all arguments to use for rendering the XML template"""
|
|
343
208
|
return {}
|
|
344
209
|
|
|
345
|
-
def render_xml(self, context,
|
|
210
|
+
def render_xml(self, context, ows_request: ows.BaseOwsRequest):
|
|
346
211
|
"""Shortcut to render XML.
|
|
347
212
|
|
|
348
213
|
This is the default method when the OutputFormat class doesn't have a renderer_class
|
|
349
214
|
"""
|
|
350
215
|
return HttpResponse(
|
|
351
216
|
render_to_string(
|
|
352
|
-
self._get_xml_template_name(),
|
|
217
|
+
self._get_xml_template_name(ows_request),
|
|
353
218
|
context={
|
|
354
219
|
"view": self.view,
|
|
355
220
|
"app_xml_namespace": self.view.xml_namespace,
|
|
@@ -360,79 +225,38 @@ class WFSMethod:
|
|
|
360
225
|
content_type=self.xml_content_type,
|
|
361
226
|
)
|
|
362
227
|
|
|
363
|
-
def _get_xml_template_name(self):
|
|
364
|
-
"""Generate the XML template name for this operation, and check
|
|
365
|
-
service =
|
|
228
|
+
def _get_xml_template_name(self, ows_request: ows.BaseOwsRequest) -> str:
|
|
229
|
+
"""Generate the XML template name for this operation, and check its file pattern"""
|
|
230
|
+
service = ows_request.service.lower()
|
|
366
231
|
template_name = f"gisserver/{service}/{self.view.version}/{self.xml_template_name}"
|
|
367
232
|
|
|
368
233
|
# Since 'service' and 'version' are based on external input,
|
|
369
|
-
# these values are double
|
|
234
|
+
# these values are double-checked again to avoid remove file inclusion.
|
|
370
235
|
if not RE_SAFE_FILENAME.match(service) or not RE_SAFE_FILENAME.match(self.view.version):
|
|
371
236
|
raise SuspiciousOperation(f"Refusing to render template name {template_name}")
|
|
372
237
|
|
|
373
238
|
return template_name
|
|
374
239
|
|
|
375
240
|
|
|
376
|
-
class
|
|
377
|
-
"""
|
|
378
|
-
|
|
379
|
-
def __init__(self, *args, **kwargs):
|
|
380
|
-
super().__init__(*args, **kwargs)
|
|
381
|
-
|
|
382
|
-
# Retrieve the feature types directly, as these are needed during parsing.
|
|
383
|
-
# This is not delayed until _parse_type_names() as that wraps any TypeError
|
|
384
|
-
# from view.get_feature_types() as an InvalidParameterValue exception.
|
|
385
|
-
self.all_feature_types = self.view.get_feature_types()
|
|
386
|
-
self.all_feature_types_by_name = _get_feature_types_by_name(self.all_feature_types)
|
|
241
|
+
class OutputFormatMixin:
|
|
242
|
+
"""Mixin to support methods that handle different output formats."""
|
|
387
243
|
|
|
388
|
-
def
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
"typeNames",
|
|
394
|
-
alias="typeName", # WFS 1.0 name, but still needed for CITE tests
|
|
395
|
-
required=False, # sometimes required, depends on other parameters
|
|
396
|
-
parser=self._parse_type_names,
|
|
397
|
-
),
|
|
398
|
-
]
|
|
399
|
-
|
|
400
|
-
def _parse_type_names(self, type_names) -> list[FeatureType]:
|
|
401
|
-
"""Find the requested feature types by name"""
|
|
402
|
-
if "(" in type_names:
|
|
403
|
-
# This allows to perform multiple queries in a single request:
|
|
404
|
-
# TYPENAMES=(A)(B)&FILTER=(filter for A)(filter for B)
|
|
405
|
-
raise OperationParsingFailed(
|
|
406
|
-
"typenames",
|
|
407
|
-
"Parameter lists to perform multiple queries are not supported yet.",
|
|
408
|
-
)
|
|
409
|
-
|
|
410
|
-
return [self._parse_type_name(name, locator="typenames") for name in type_names.split(",")]
|
|
411
|
-
|
|
412
|
-
def _parse_type_name(self, name, locator="typename") -> FeatureType:
|
|
413
|
-
"""Find the requested feature type for a type name"""
|
|
414
|
-
app_prefix = self.namespaces[self.view.xml_namespace]
|
|
415
|
-
# strip our XML prefix
|
|
416
|
-
local_name = name[len(app_prefix) + 1 :] if name.startswith(f"{app_prefix}:") else name
|
|
417
|
-
|
|
418
|
-
try:
|
|
419
|
-
return self.all_feature_types_by_name[local_name]
|
|
420
|
-
except KeyError:
|
|
421
|
-
raise InvalidParameterValue(
|
|
422
|
-
locator,
|
|
423
|
-
f"Typename '{name}' doesn't exist in this server. "
|
|
424
|
-
f"Please check the capabilities and reformulate your request.",
|
|
425
|
-
) from None
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
def _get_feature_types_by_name(feature_types) -> dict[str, FeatureType]:
|
|
429
|
-
"""Create a lookup for feature types by name."""
|
|
430
|
-
features_by_name = {ft.name: ft for ft in feature_types}
|
|
431
|
-
|
|
432
|
-
# Check against bad configuration
|
|
433
|
-
if len(features_by_name) != len(feature_types):
|
|
434
|
-
all_names = [ft.name for ft in feature_types]
|
|
435
|
-
duplicates = ", ".join(sorted({n for n in all_names if all_names.count(n) > 1}))
|
|
436
|
-
raise ImproperlyConfigured(f"FeatureType names should be unique: {duplicates}")
|
|
244
|
+
def get_output_formats(self) -> list[OutputFormat]:
|
|
245
|
+
"""List all output formats. This is exposed in GetCapabilities,
|
|
246
|
+
and used for internal rendering.
|
|
247
|
+
"""
|
|
248
|
+
raise NotImplementedError()
|
|
437
249
|
|
|
438
|
-
|
|
250
|
+
def resolve_output_format(self, value, locator="outputFormat") -> OutputFormat:
|
|
251
|
+
"""Select the proper OutputFormat object based on the input value"""
|
|
252
|
+
# When using ?OUTPUTFORMAT=application/gml+xml", it is actually a URL-encoded space
|
|
253
|
+
# character. Hence, spaces are replaced back to a '+' character to allow such notation
|
|
254
|
+
# instead of forcing it to be ?OUTPUTFORMAT=application/gml%2bxml".
|
|
255
|
+
for v in {value, value.replace(" ", "+")}:
|
|
256
|
+
for o in self.get_output_formats():
|
|
257
|
+
if o.matches(v):
|
|
258
|
+
return o
|
|
259
|
+
raise InvalidParameterValue(
|
|
260
|
+
f"'{value}' is not a permitted output format for this operation.",
|
|
261
|
+
locator=locator or "outputFormat",
|
|
262
|
+
) from None
|