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.
Files changed (73) hide show
  1. {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/METADATA +23 -13
  2. django_gisserver-2.0.dist-info/RECORD +66 -0
  3. {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/WHEEL +1 -1
  4. gisserver/__init__.py +1 -1
  5. gisserver/compat.py +23 -0
  6. gisserver/conf.py +7 -0
  7. gisserver/db.py +63 -60
  8. gisserver/exceptions.py +47 -9
  9. gisserver/extensions/__init__.py +4 -0
  10. gisserver/{parsers/fes20 → extensions}/functions.py +11 -5
  11. gisserver/extensions/queries.py +261 -0
  12. gisserver/features.py +267 -240
  13. gisserver/geometries.py +34 -39
  14. gisserver/management/__init__.py +0 -0
  15. gisserver/management/commands/__init__.py +0 -0
  16. gisserver/management/commands/loadgeojson.py +291 -0
  17. gisserver/operations/base.py +129 -305
  18. gisserver/operations/wfs20.py +428 -336
  19. gisserver/output/__init__.py +10 -48
  20. gisserver/output/base.py +198 -143
  21. gisserver/output/csv.py +81 -85
  22. gisserver/output/geojson.py +63 -72
  23. gisserver/output/gml32.py +310 -281
  24. gisserver/output/iters.py +207 -0
  25. gisserver/output/results.py +71 -30
  26. gisserver/output/stored.py +143 -0
  27. gisserver/output/utils.py +75 -154
  28. gisserver/output/xmlschema.py +86 -47
  29. gisserver/parsers/__init__.py +10 -10
  30. gisserver/parsers/ast.py +320 -0
  31. gisserver/parsers/fes20/__init__.py +15 -11
  32. gisserver/parsers/fes20/expressions.py +89 -50
  33. gisserver/parsers/fes20/filters.py +111 -43
  34. gisserver/parsers/fes20/identifiers.py +44 -26
  35. gisserver/parsers/fes20/lookups.py +144 -0
  36. gisserver/parsers/fes20/operators.py +336 -128
  37. gisserver/parsers/fes20/sorting.py +107 -34
  38. gisserver/parsers/gml/__init__.py +12 -11
  39. gisserver/parsers/gml/base.py +6 -3
  40. gisserver/parsers/gml/geometries.py +69 -35
  41. gisserver/parsers/ows/__init__.py +25 -0
  42. gisserver/parsers/ows/kvp.py +190 -0
  43. gisserver/parsers/ows/requests.py +158 -0
  44. gisserver/parsers/query.py +175 -0
  45. gisserver/parsers/values.py +26 -0
  46. gisserver/parsers/wfs20/__init__.py +37 -0
  47. gisserver/parsers/wfs20/adhoc.py +245 -0
  48. gisserver/parsers/wfs20/base.py +143 -0
  49. gisserver/parsers/wfs20/projection.py +103 -0
  50. gisserver/parsers/wfs20/requests.py +482 -0
  51. gisserver/parsers/wfs20/stored.py +192 -0
  52. gisserver/parsers/xml.py +249 -0
  53. gisserver/projection.py +357 -0
  54. gisserver/static/gisserver/index.css +12 -1
  55. gisserver/templates/gisserver/index.html +1 -1
  56. gisserver/templates/gisserver/service_description.html +2 -2
  57. gisserver/templates/gisserver/wfs/2.0.0/get_capabilities.xml +11 -11
  58. gisserver/templates/gisserver/wfs/feature_field.html +2 -2
  59. gisserver/templatetags/gisserver_tags.py +20 -0
  60. gisserver/types.py +375 -258
  61. gisserver/views.py +206 -75
  62. django_gisserver-1.4.1.dist-info/RECORD +0 -53
  63. gisserver/parsers/base.py +0 -149
  64. gisserver/parsers/fes20/query.py +0 -275
  65. gisserver/parsers/tags.py +0 -102
  66. gisserver/queries/__init__.py +0 -34
  67. gisserver/queries/adhoc.py +0 -181
  68. gisserver/queries/base.py +0 -146
  69. gisserver/queries/stored.py +0 -205
  70. gisserver/templates/gisserver/wfs/2.0.0/describe_stored_queries.xml +0 -20
  71. gisserver/templates/gisserver/wfs/2.0.0/list_stored_queries.xml +0 -14
  72. {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/LICENSE +0 -0
  73. {django_gisserver-1.4.1.dist-info → django_gisserver-2.0.dist-info}/top_level.txt +0 -0
@@ -1,155 +1,93 @@
1
1
  """The base protocol to implement an operation.
2
2
 
3
- All operations extend from an WFSMethod class.
4
- This defines the parameters and output formats of the method.
5
- This introspection data is also parsed by the GetCapabilities call.
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
- from collections.abc import Callable
13
- from dataclasses import dataclass, field
14
- from typing import Any
17
+ import typing
18
+ from dataclasses import dataclass
19
+ from functools import cached_property
15
20
 
16
- from django.core.exceptions import ImproperlyConfigured, SuspiciousOperation
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 definition of an parameter for an WFS method.
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 UnsupportedParameter(Parameter):
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 should be used in the ``output_formats`` section of the WFSMethod.
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
- renderer_class=None,
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 a HTML overview page.
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 WFSMethod:
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
- do_not_call_in_templates = True # avoid __call__ execution in Django templates
199
-
200
- #: List the suported parameters for this method, extended by get_parameters()
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
- #: List the supported output formats for this method.
204
- output_formats: list[OutputFormat] = []
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
- #: Default content-type for render_xml()
210
- xml_content_type = "text/xml; charset=utf-8"
211
-
212
- def __init__(self, view):
213
- self.view = view # an gisserver.views.GISView
214
- self.namespaces = {
215
- # Default namespaces for the incoming request:
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 _parse_output_format(self, value) -> OutputFormat:
260
- """Select the proper OutputFormat object based on the input value"""
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 _parse_namespaces(self, value) -> dict[str, str]:
274
- """Parse the namespaces definition.
157
+ def process_request(self, ows_request: ows.BaseOwsRequest):
158
+ """Default call implementation: render an XML template."""
159
+ raise NotImplementedError()
275
160
 
276
- The NAMESPACES parameter defines which namespaces are used in the KVP request.
277
- When this parameter is not given, the default namespaces are assumed.
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
- if not value:
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
- for param in self.get_parameters():
315
- param_values[param.name] = param.value_from_query(KVP)
316
-
317
- # Update version if requested.
318
- # This is stored on the view, so exceptions also use it.
319
- if param_values.get("version"):
320
- self.view.set_version(param_values["version"])
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
- def validate(self, **params):
326
- """Perform final request parameter validation before the method is called"""
193
+ class XmlTemplateMixin:
194
+ """Mixin to support methods that render using a template."""
327
195
 
328
- def __call__(self, **params):
329
- """Default call implementation: render an XML template."""
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
- if output_format.renderer_class is not None:
335
- # Streaming HTTP responses, e.g. GML32/GeoJSON output:
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
- return self.render_xml(context, **params)
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, **params):
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, **params):
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 it's file pattern"""
365
- service = self.view.KVP["SERVICE"].lower()
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 checked again to avoid remove file inclusion.
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 WFSTypeNamesMethod(WFSMethod):
377
- """A base method that also resolved the TYPENAMES parameter."""
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 get_parameters(self):
389
- return super().get_parameters() + [
390
- # QGis sends both TYPENAME (wfs 1.x) and TYPENAMES (wfs 2.0) for DescribeFeatureType
391
- # typeNames is not required when a ResourceID / GetFeatureById is specified.
392
- Parameter(
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
- return features_by_name
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