262 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from __future__ import annotations
 | |
| 
 | |
| import re
 | |
| import typing as t
 | |
| import uuid
 | |
| from urllib.parse import quote
 | |
| 
 | |
| if t.TYPE_CHECKING:
 | |
|     from .map import Map
 | |
| 
 | |
| 
 | |
| class ValidationError(ValueError):
 | |
|     """Validation error.  If a rule converter raises this exception the rule
 | |
|     does not match the current URL and the next URL is tried.
 | |
|     """
 | |
| 
 | |
| 
 | |
| class BaseConverter:
 | |
|     """Base class for all converters.
 | |
| 
 | |
|     .. versionchanged:: 2.3
 | |
|         ``part_isolating`` defaults to ``False`` if ``regex`` contains a ``/``.
 | |
|     """
 | |
| 
 | |
|     regex = "[^/]+"
 | |
|     weight = 100
 | |
|     part_isolating = True
 | |
| 
 | |
|     def __init_subclass__(cls, **kwargs: t.Any) -> None:
 | |
|         super().__init_subclass__(**kwargs)
 | |
| 
 | |
|         # If the converter isn't inheriting its regex, disable part_isolating by default
 | |
|         # if the regex contains a / character.
 | |
|         if "regex" in cls.__dict__ and "part_isolating" not in cls.__dict__:
 | |
|             cls.part_isolating = "/" not in cls.regex
 | |
| 
 | |
|     def __init__(self, map: Map, *args: t.Any, **kwargs: t.Any) -> None:
 | |
|         self.map = map
 | |
| 
 | |
|     def to_python(self, value: str) -> t.Any:
 | |
|         return value
 | |
| 
 | |
|     def to_url(self, value: t.Any) -> str:
 | |
|         # safe = https://url.spec.whatwg.org/#url-path-segment-string
 | |
|         return quote(str(value), safe="!$&'()*+,/:;=@")
 | |
| 
 | |
| 
 | |
| class UnicodeConverter(BaseConverter):
 | |
|     """This converter is the default converter and accepts any string but
 | |
|     only one path segment.  Thus the string can not include a slash.
 | |
| 
 | |
|     This is the default validator.
 | |
| 
 | |
|     Example::
 | |
| 
 | |
|         Rule('/pages/<page>'),
 | |
|         Rule('/<string(length=2):lang_code>')
 | |
| 
 | |
|     :param map: the :class:`Map`.
 | |
|     :param minlength: the minimum length of the string.  Must be greater
 | |
|                       or equal 1.
 | |
|     :param maxlength: the maximum length of the string.
 | |
|     :param length: the exact length of the string.
 | |
|     """
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         map: Map,
 | |
|         minlength: int = 1,
 | |
|         maxlength: int | None = None,
 | |
|         length: int | None = None,
 | |
|     ) -> None:
 | |
|         super().__init__(map)
 | |
|         if length is not None:
 | |
|             length_regex = f"{{{int(length)}}}"
 | |
|         else:
 | |
|             if maxlength is None:
 | |
|                 maxlength_value = ""
 | |
|             else:
 | |
|                 maxlength_value = str(int(maxlength))
 | |
|             length_regex = f"{{{int(minlength)},{maxlength_value}}}"
 | |
|         self.regex = f"[^/]{length_regex}"
 | |
| 
 | |
| 
 | |
| class AnyConverter(BaseConverter):
 | |
|     """Matches one of the items provided.  Items can either be Python
 | |
|     identifiers or strings::
 | |
| 
 | |
|         Rule('/<any(about, help, imprint, class, "foo,bar"):page_name>')
 | |
| 
 | |
|     :param map: the :class:`Map`.
 | |
|     :param items: this function accepts the possible items as positional
 | |
|                   arguments.
 | |
| 
 | |
|     .. versionchanged:: 2.2
 | |
|         Value is validated when building a URL.
 | |
|     """
 | |
| 
 | |
|     def __init__(self, map: Map, *items: str) -> None:
 | |
|         super().__init__(map)
 | |
|         self.items = set(items)
 | |
|         self.regex = f"(?:{'|'.join([re.escape(x) for x in items])})"
 | |
| 
 | |
|     def to_url(self, value: t.Any) -> str:
 | |
|         if value in self.items:
 | |
|             return str(value)
 | |
| 
 | |
|         valid_values = ", ".join(f"'{item}'" for item in sorted(self.items))
 | |
|         raise ValueError(f"'{value}' is not one of {valid_values}")
 | |
| 
 | |
| 
 | |
| class PathConverter(BaseConverter):
 | |
|     """Like the default :class:`UnicodeConverter`, but it also matches
 | |
|     slashes.  This is useful for wikis and similar applications::
 | |
| 
 | |
|         Rule('/<path:wikipage>')
 | |
|         Rule('/<path:wikipage>/edit')
 | |
| 
 | |
|     :param map: the :class:`Map`.
 | |
|     """
 | |
| 
 | |
|     part_isolating = False
 | |
|     regex = "[^/].*?"
 | |
|     weight = 200
 | |
| 
 | |
| 
 | |
| class NumberConverter(BaseConverter):
 | |
|     """Baseclass for `IntegerConverter` and `FloatConverter`.
 | |
| 
 | |
|     :internal:
 | |
|     """
 | |
| 
 | |
|     weight = 50
 | |
|     num_convert: t.Callable[[t.Any], t.Any] = int
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         map: Map,
 | |
|         fixed_digits: int = 0,
 | |
|         min: int | None = None,
 | |
|         max: int | None = None,
 | |
|         signed: bool = False,
 | |
|     ) -> None:
 | |
|         if signed:
 | |
|             self.regex = self.signed_regex
 | |
|         super().__init__(map)
 | |
|         self.fixed_digits = fixed_digits
 | |
|         self.min = min
 | |
|         self.max = max
 | |
|         self.signed = signed
 | |
| 
 | |
|     def to_python(self, value: str) -> t.Any:
 | |
|         if self.fixed_digits and len(value) != self.fixed_digits:
 | |
|             raise ValidationError()
 | |
|         value_num = self.num_convert(value)
 | |
|         if (self.min is not None and value_num < self.min) or (
 | |
|             self.max is not None and value_num > self.max
 | |
|         ):
 | |
|             raise ValidationError()
 | |
|         return value_num
 | |
| 
 | |
|     def to_url(self, value: t.Any) -> str:
 | |
|         value_str = str(self.num_convert(value))
 | |
|         if self.fixed_digits:
 | |
|             value_str = value_str.zfill(self.fixed_digits)
 | |
|         return value_str
 | |
| 
 | |
|     @property
 | |
|     def signed_regex(self) -> str:
 | |
|         return f"-?{self.regex}"
 | |
| 
 | |
| 
 | |
| class IntegerConverter(NumberConverter):
 | |
|     """This converter only accepts integer values::
 | |
| 
 | |
|         Rule("/page/<int:page>")
 | |
| 
 | |
|     By default it only accepts unsigned, positive values. The ``signed``
 | |
|     parameter will enable signed, negative values. ::
 | |
| 
 | |
|         Rule("/page/<int(signed=True):page>")
 | |
| 
 | |
|     :param map: The :class:`Map`.
 | |
|     :param fixed_digits: The number of fixed digits in the URL. If you
 | |
|         set this to ``4`` for example, the rule will only match if the
 | |
|         URL looks like ``/0001/``. The default is variable length.
 | |
|     :param min: The minimal value.
 | |
|     :param max: The maximal value.
 | |
|     :param signed: Allow signed (negative) values.
 | |
| 
 | |
|     .. versionadded:: 0.15
 | |
|         The ``signed`` parameter.
 | |
|     """
 | |
| 
 | |
|     regex = r"\d+"
 | |
| 
 | |
| 
 | |
| class FloatConverter(NumberConverter):
 | |
|     """This converter only accepts floating point values::
 | |
| 
 | |
|         Rule("/probability/<float:probability>")
 | |
| 
 | |
|     By default it only accepts unsigned, positive values. The ``signed``
 | |
|     parameter will enable signed, negative values. ::
 | |
| 
 | |
|         Rule("/offset/<float(signed=True):offset>")
 | |
| 
 | |
|     :param map: The :class:`Map`.
 | |
|     :param min: The minimal value.
 | |
|     :param max: The maximal value.
 | |
|     :param signed: Allow signed (negative) values.
 | |
| 
 | |
|     .. versionadded:: 0.15
 | |
|         The ``signed`` parameter.
 | |
|     """
 | |
| 
 | |
|     regex = r"\d+\.\d+"
 | |
|     num_convert = float
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         map: Map,
 | |
|         min: float | None = None,
 | |
|         max: float | None = None,
 | |
|         signed: bool = False,
 | |
|     ) -> None:
 | |
|         super().__init__(map, min=min, max=max, signed=signed)  # type: ignore
 | |
| 
 | |
| 
 | |
| class UUIDConverter(BaseConverter):
 | |
|     """This converter only accepts UUID strings::
 | |
| 
 | |
|         Rule('/object/<uuid:identifier>')
 | |
| 
 | |
|     .. versionadded:: 0.10
 | |
| 
 | |
|     :param map: the :class:`Map`.
 | |
|     """
 | |
| 
 | |
|     regex = (
 | |
|         r"[A-Fa-f0-9]{8}-[A-Fa-f0-9]{4}-"
 | |
|         r"[A-Fa-f0-9]{4}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{12}"
 | |
|     )
 | |
| 
 | |
|     def to_python(self, value: str) -> uuid.UUID:
 | |
|         return uuid.UUID(value)
 | |
| 
 | |
|     def to_url(self, value: uuid.UUID) -> str:
 | |
|         return str(value)
 | |
| 
 | |
| 
 | |
| #: the default converter mapping for the map.
 | |
| DEFAULT_CONVERTERS: t.Mapping[str, type[BaseConverter]] = {
 | |
|     "default": UnicodeConverter,
 | |
|     "string": UnicodeConverter,
 | |
|     "any": AnyConverter,
 | |
|     "path": PathConverter,
 | |
|     "int": IntegerConverter,
 | |
|     "float": FloatConverter,
 | |
|     "uuid": UUIDConverter,
 | |
| }
 |