Coverage for pydantic/aliases.py: 100.00%

52 statements  

« prev     ^ index     » next       coverage.py v7.5.4, created at 2024-07-03 19:29 +0000

1"""Support for alias configurations.""" 

2 

3from __future__ import annotations 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

4 

5import dataclasses 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

6from typing import Any, Callable, Literal 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

7 

8from pydantic_core import PydanticUndefined 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

9 

10from ._internal import _internal_dataclass 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

11 

12__all__ = ('AliasGenerator', 'AliasPath', 'AliasChoices') 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

13 

14 

15@dataclasses.dataclass(**_internal_dataclass.slots_true) 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

16class AliasPath: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

17 """Usage docs: https://docs.pydantic.dev/2.8/concepts/alias#aliaspath-and-aliaschoices 

18 

19 A data class used by `validation_alias` as a convenience to create aliases. 

20 

21 Attributes: 

22 path: A list of string or integer aliases. 

23 """ 

24 

25 path: list[int | str] 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

26 

27 def __init__(self, first_arg: str, *args: str | int) -> None: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

28 self.path = [first_arg] + list(args) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

29 

30 def convert_to_aliases(self) -> list[str | int]: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

31 """Converts arguments to a list of string or integer aliases. 

32 

33 Returns: 

34 The list of aliases. 

35 """ 

36 return self.path 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

37 

38 def search_dict_for_path(self, d: dict) -> Any: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

39 """Searches a dictionary for the path specified by the alias. 

40 

41 Returns: 

42 The value at the specified path, or `PydanticUndefined` if the path is not found. 

43 """ 

44 v = d 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

45 for k in self.path: 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

46 if isinstance(v, str): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

47 # disallow indexing into a str, like for AliasPath('x', 0) and x='abc' 

48 return PydanticUndefined 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

49 try: 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

50 v = v[k] 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

51 except (KeyError, IndexError, TypeError): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

52 return PydanticUndefined 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

53 return v 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

54 

55 

56@dataclasses.dataclass(**_internal_dataclass.slots_true) 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

57class AliasChoices: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

58 """Usage docs: https://docs.pydantic.dev/2.8/concepts/alias#aliaspath-and-aliaschoices 

59 

60 A data class used by `validation_alias` as a convenience to create aliases. 

61 

62 Attributes: 

63 choices: A list containing a string or `AliasPath`. 

64 """ 

65 

66 choices: list[str | AliasPath] 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

67 

68 def __init__(self, first_choice: str | AliasPath, *choices: str | AliasPath) -> None: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

69 self.choices = [first_choice] + list(choices) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

70 

71 def convert_to_aliases(self) -> list[list[str | int]]: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

72 """Converts arguments to a list of lists containing string or integer aliases. 

73 

74 Returns: 

75 The list of aliases. 

76 """ 

77 aliases: list[list[str | int]] = [] 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

78 for c in self.choices: 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

79 if isinstance(c, AliasPath): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

80 aliases.append(c.convert_to_aliases()) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

81 else: 

82 aliases.append([c]) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

83 return aliases 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

84 

85 

86@dataclasses.dataclass(**_internal_dataclass.slots_true) 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

87class AliasGenerator: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

88 """Usage docs: https://docs.pydantic.dev/2.8/concepts/alias#using-an-aliasgenerator 

89 

90 A data class used by `alias_generator` as a convenience to create various aliases. 

91 

92 Attributes: 

93 alias: A callable that takes a field name and returns an alias for it. 

94 validation_alias: A callable that takes a field name and returns a validation alias for it. 

95 serialization_alias: A callable that takes a field name and returns a serialization alias for it. 

96 """ 

97 

98 alias: Callable[[str], str] | None = None 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

99 validation_alias: Callable[[str], str | AliasPath | AliasChoices] | None = None 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

100 serialization_alias: Callable[[str], str] | None = None 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

101 

102 def _generate_alias( 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

103 self, 

104 alias_kind: Literal['alias', 'validation_alias', 'serialization_alias'], 

105 allowed_types: tuple[type[str] | type[AliasPath] | type[AliasChoices], ...], 

106 field_name: str, 

107 ) -> str | AliasPath | AliasChoices | None: 

108 """Generate an alias of the specified kind. Returns None if the alias generator is None. 

109 

110 Raises: 

111 TypeError: If the alias generator produces an invalid type. 

112 """ 

113 alias = None 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

114 if alias_generator := getattr(self, alias_kind): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

115 alias = alias_generator(field_name) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

116 if alias and not isinstance(alias, allowed_types): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

117 raise TypeError( 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

118 f'Invalid `{alias_kind}` type. `{alias_kind}` generator must produce one of `{allowed_types}`' 

119 ) 

120 return alias 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

121 

122 def generate_aliases(self, field_name: str) -> tuple[str | None, str | AliasPath | AliasChoices | None, str | None]: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL

123 """Generate `alias`, `validation_alias`, and `serialization_alias` for a field. 

124 

125 Returns: 

126 A tuple of three aliases - validation, alias, and serialization. 

127 """ 

128 alias = self._generate_alias('alias', (str,), field_name) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

129 validation_alias = self._generate_alias('validation_alias', (str, AliasChoices, AliasPath), field_name) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

130 serialization_alias = self._generate_alias('serialization_alias', (str,), field_name) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

131 

132 return alias, validation_alias, serialization_alias # type: ignore 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL