Source code for click_extra.types

# Copyright Kevin Deldycke <kevin@deldycke.com> and contributors.
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

from __future__ import annotations

from enum import Enum, StrEnum

from . import Choice

TYPE_CHECKING = False
if TYPE_CHECKING:
    from typing import Any

    from . import Context, Parameter


[docs] class ChoiceSource(StrEnum): """Source of choices for ``EnumChoice``.""" # KEY and NAME are synonyms. KEY = "key" NAME = "name" VALUE = "value" STR = "str"
[docs] class EnumChoice(Choice): """Choice type for ``Enum``. Allows to select which part of the members to use as choice strings, by setting the ``choice_source`` parameter to one of: - ``ChoiceSource.KEY`` or ``ChoiceSource.NAME`` to use the key (i.e. the ``name`` property), - ``ChoiceSource.VALUE`` to use the ``value``, or - ``ChoiceSource.STR`` to use the ``str()`` string representation. Default to ``ChoiceSource.STR``, which makes you to only have to define the ``__str__()`` method on your ``Enum`` to produce beautiful choice strings. """ choices: tuple[str, ...] """The available choice strings. .. hint:: Contrary to the parent ``Choice`` class, we store choices directly as strings, not the ``Enum`` members themselves. That way there is no surprises when displaying them to the user. This trick bypass ``Enum``-specific code path in the Click library. Because, after all, a terminal environment only deals with strings: arguments, parameters, parsing, help messages, environment variables, etc. """ _enum: Enum """The ``Enum`` class used for choices.""" _enum_map: dict[str, Enum] """Mapping of choice strings to ``Enum`` members.""" _choice_source: ChoiceSource """The source used to derive choice strings from Enum members.""" def __init__( self, choices: Enum, case_sensitive: bool = False, choice_source: ChoiceSource | str = ChoiceSource.STR, ) -> None: """Same as ``click.Choice``, but takes an ``Enum`` as ``choices``. Also defaults to case-insensitive matching. """ # Keep the Enum class around. assert issubclass(choices, Enum), ( f"choice_enum must be a subclass of Enum, got {choices!r}." ) self._enum = choices # Normalize choice_source to ChoiceSource. if isinstance(choice_source, str): choice_source = ChoiceSource[choice_source.upper()] self._choice_source = choice_source # Build the mapping of choice strings to Enum members. self._enum_map = {} for member in self._enum: choice = self.get_choice_string(member) if choice in self._enum_map: raise ValueError( f"{self._enum} has duplicated choice string {choice!r} for " f"members {self._enum_map[choice]!r} and {member!r} when using " f"{self._choice_source!r}." ) self._enum_map[choice] = member super().__init__(choices=self._enum_map, case_sensitive=case_sensitive)
[docs] def get_choice_string(self, member: Enum) -> str: """Derivate the choice string from the given ``Enum``'s ``member``.""" if self._choice_source in (ChoiceSource.KEY, ChoiceSource.NAME): choice = member.name elif self._choice_source == ChoiceSource.VALUE: choice = member.value elif self._choice_source == ChoiceSource.STR: choice = str(member) if not isinstance(choice, str): raise TypeError( f"{member!r} produced non-string choice {choice!r} when using " f"{self._choice_source!r}." ) return choice
[docs] def normalize_choice(self, choice: Enum | str, ctx: Context | None) -> str: """Expand the parent's ``normalize_choice()`` to accept ``Enum`` members as input. Parent method expects a string, but here we allow passing ``Enum`` members too. """ if isinstance(choice, Enum): choice = self.get_choice_string(choice) return super().normalize_choice(choice, ctx)
[docs] def convert(self, value: Any, param: Parameter | None, ctx: Context | None) -> Enum: """Convert the input value to the corresponding ``Enum`` member. The parent's ``convert()`` is going to return the choice string, which we then map back to the corresponding ``Enum`` member. """ choice_string = super().convert(value, param, ctx) return self._enum_map[choice_string]
def __repr__(self) -> str: return f"EnumChoice{self.choices!r}"