fastgenerateapi 1.1.13__tar.gz → 1.1.16__tar.gz

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.

Potentially problematic release.


This version of fastgenerateapi might be problematic. Click here for more details.

Files changed (114) hide show
  1. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/PKG-INFO +5 -3
  2. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/README.md +4 -2
  3. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/__version__.py +1 -1
  4. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/base_view.py +19 -19
  5. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/get_all_view.py +15 -6
  6. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/get_one_view.py +11 -2
  7. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/get_relation_view.py +1 -1
  8. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/base_mixin.py +2 -1
  9. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/dbmodel_mixin.py +1 -1
  10. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/update_relation_view.py +1 -1
  11. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/update_view.py +1 -1
  12. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/filter_controller.py +5 -8
  13. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/rpc_controller.py +1 -1
  14. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/deps/paginator_deps.py +7 -1
  15. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/deps/tree_params_deps.py +7 -1
  16. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/fastapi_utils/param_utils.py +9 -0
  17. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/pydantic_utils/base_model.py +9 -2
  18. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/common_function.py +28 -12
  19. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/filter_schema_factory.py +6 -3
  20. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/app_settings.py +1 -1
  21. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi.egg-info/PKG-INFO +5 -3
  22. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/LICENSE +0 -0
  23. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/__init__.py +0 -0
  24. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/__init__.py +0 -0
  25. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/api_view.py +0 -0
  26. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/create_view.py +0 -0
  27. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/delete_filter_view.py +0 -0
  28. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/delete_tree_view.py +0 -0
  29. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/delete_view.py +0 -0
  30. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/get_tree_view.py +0 -0
  31. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/__init__.py +0 -0
  32. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/get_mixin.py +0 -0
  33. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/response_mixin.py +0 -0
  34. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/save_mixin.py +0 -0
  35. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/tool_mixin.py +0 -0
  36. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/__init__.py +0 -0
  37. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/docx_util.py +0 -0
  38. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/file_util.py +0 -0
  39. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/pdf_util.py +0 -0
  40. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/xlsx_util.py +0 -0
  41. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/mixin/utils/zip_util.py +0 -0
  42. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/sql_get_view.py +0 -0
  43. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/api_view/switch_view.py +0 -0
  44. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/cache/__init__.py +0 -0
  45. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/cache/cache_decorator.py +0 -0
  46. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/cache/key_builder.py +0 -0
  47. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/channel/__init__.py +0 -0
  48. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/channel/connection_manager.py +0 -0
  49. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/channel/consumer.py +0 -0
  50. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/channel/websocket_view.py +0 -0
  51. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/__init__.py +0 -0
  52. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/router_controller.py +0 -0
  53. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/search_controller.py +0 -0
  54. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/controller/ws_controller.py +0 -0
  55. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/data_type/__init__.py +0 -0
  56. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/data_type/data_type.py +0 -0
  57. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/data_type/mysql_data_type.py +0 -0
  58. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/deps/__init__.py +0 -0
  59. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/deps/filter_params_deps.py +0 -0
  60. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/example/__init__.py +0 -0
  61. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/example/models.py +0 -0
  62. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/example/routers.py +0 -0
  63. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/example/schemas.py +0 -0
  64. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/example/views.py +0 -0
  65. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/fastapi_utils/__init__.py +0 -0
  66. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/fastapi_utils/all.py +0 -0
  67. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/fastapi_utils/response_utils.py +0 -0
  68. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/model/__init__.py +0 -0
  69. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/model/base_model.py +0 -0
  70. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/__init__.py +0 -0
  71. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/aes_field.py +0 -0
  72. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/enum_field.py +0 -0
  73. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/pk_field.py +0 -0
  74. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/pwd_field.py +0 -0
  75. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/soft_delete_field.py +0 -0
  76. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/my_fields/validator.py +0 -0
  77. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/pydantic_utils/__init__.py +0 -0
  78. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/pydantic_utils/base_settings.py +0 -0
  79. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/pydantic_utils/json_encoders.py +0 -0
  80. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/__init__.py +0 -0
  81. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/common_schema_factory.py +0 -0
  82. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/create_schema_factory.py +0 -0
  83. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/get_all_schema_factory.py +0 -0
  84. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/get_one_schema_factory.py +0 -0
  85. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/get_relation_schema_factory.py +0 -0
  86. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/get_tree_schema_factory.py +0 -0
  87. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/response_factory.py +0 -0
  88. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/sql_get_all_schema_factory.py +0 -0
  89. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/schemas_factory/update_schema_factory.py +0 -0
  90. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/__init__.py +0 -0
  91. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/all_settings.py +0 -0
  92. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/db_settings.py +0 -0
  93. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/file_settings.py +0 -0
  94. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/jwt_settings.py +0 -0
  95. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/otlp_settings.py +0 -0
  96. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/redis_settings.py +0 -0
  97. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/sms_settings.py +0 -0
  98. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/settings/system_settings.py +0 -0
  99. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/__init__.py +0 -0
  100. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/aes.py +0 -0
  101. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/auto_discover.py +0 -0
  102. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/exception.py +0 -0
  103. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/file_utils.py +0 -0
  104. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/pwd_utils.py +0 -0
  105. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/ramdom_utils.py +0 -0
  106. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/snowflake.py +0 -0
  107. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/str_util.py +0 -0
  108. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi/utils/swagger_to_js.py +0 -0
  109. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi.egg-info/SOURCES.txt +0 -0
  110. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi.egg-info/dependency_links.txt +0 -0
  111. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/fastgenerateapi.egg-info/top_level.txt +0 -0
  112. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/script/__init__.py +0 -0
  113. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/setup.cfg +0 -0
  114. {fastgenerateapi-1.1.13 → fastgenerateapi-1.1.16}/setup.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: fastgenerateapi
3
- Version: 1.1.13
3
+ Version: 1.1.16
4
4
  Summary: FastAPIView Class View
5
5
  Author: ShiLiang
6
6
  Author-email: 2509144896@qq.com
@@ -24,15 +24,17 @@ License-File: LICENSE
24
24
  [Github](https://github.com/ShiLiangAPI)
25
25
 
26
26
  #### 版本说明
27
- 最新版本:v1.1.7+
27
+ [pydantic-v2]
28
+ 稳定版本:v1.1.15+
28
29
  建议python版本:3.9+
29
30
  推荐fastapi版本:0.103.2+
30
31
  推荐pydantic版本:2.11.5+
31
32
 
33
+ [pydantic-v1]
32
34
  稳定版本:v0.0.21
33
35
  建议python版本:3.9+
34
36
  推荐fastapi版本:0.97.0
35
-
37
+ 推荐pydantic版本:1.10.9+
36
38
 
37
39
 
38
40
  ###### FastGenerateApi 使用说明
@@ -6,15 +6,17 @@
6
6
  [Github](https://github.com/ShiLiangAPI)
7
7
 
8
8
  #### 版本说明
9
- 最新版本:v1.1.7+
9
+ [pydantic-v2]
10
+ 稳定版本:v1.1.15+
10
11
  建议python版本:3.9+
11
12
  推荐fastapi版本:0.103.2+
12
13
  推荐pydantic版本:2.11.5+
13
14
 
15
+ [pydantic-v1]
14
16
  稳定版本:v0.0.21
15
17
  建议python版本:3.9+
16
18
  推荐fastapi版本:0.97.0
17
-
19
+ 推荐pydantic版本:1.10.9+
18
20
 
19
21
 
20
22
  ###### FastGenerateApi 使用说明
@@ -8,7 +8,7 @@
8
8
  # d8888P
9
9
 
10
10
 
11
- VERSION = (1, 1, 13)
11
+ VERSION = (1, 1, 16)
12
12
 
13
13
  __version__ = '.'.join(map(str, VERSION))
14
14
 
@@ -58,11 +58,10 @@ class BaseView(BaseMixin, ResponseMixin, ToolMixin, DBModelMixin):
58
58
 
59
59
  return self.get_active_queryset(self.model_class)
60
60
 
61
- def get_active_queryset(self, model_class: Union[Type[Model], QuerySet, None] = None) -> QuerySet:
61
+ @staticmethod
62
+ def get_active_queryset(model_class: Union[Type[Model], QuerySet, None] = None) -> QuerySet:
62
63
  if not model_class:
63
- if not self.model_class:
64
- return self.error(msg="model_class not allow None")
65
- model_class = self.model_class
64
+ raise ResponseMixin.error(f"model_class not allow None")
66
65
  delete_filter_dict = {settings.app_settings.WHETHER_DELETE_FIELD: True}
67
66
  if settings.app_settings.DELETE_FIELD_TYPE == "time":
68
67
  delete_filter_dict = {settings.app_settings.WHETHER_DELETE_FIELD: 0}
@@ -76,26 +75,26 @@ class BaseView(BaseMixin, ResponseMixin, ToolMixin, DBModelMixin):
76
75
  return self.error(msg="relation_model_class not allow None")
77
76
  return self.get_active_queryset(self.relation_model_class)
78
77
 
79
- async def get_object(self, pk, model_class=None, is_with_prefetch=False):
80
- if model_class:
81
- queryset = self.get_active_queryset(model_class).filter(id=pk)
82
- else:
83
- queryset = self.queryset.filter(id=pk)
78
+ @classmethod
79
+ async def get_object(cls, pk, model_class, is_with_prefetch=False):
80
+ queryset = cls.get_active_queryset(model_class).filter(id=pk)
84
81
  if is_with_prefetch:
85
- queryset = queryset.prefetch_related(*self.prefetch_related_fields.keys())
82
+ queryset = queryset.prefetch_related(*cls.prefetch_related_fields.keys())
86
83
  model = await queryset.first()
87
84
  if model:
88
85
  return model
89
86
  else:
90
87
  raise NOT_FOUND
91
88
 
92
- def _delete_value(self):
89
+ @staticmethod
90
+ def _delete_value():
93
91
  result = False
94
92
  if settings.app_settings.DELETE_FIELD_TYPE == "time":
95
93
  result = int(time.time() * 1000)
96
94
  return result
97
95
 
98
- async def delete_queryset(self, queryset: QuerySet):
96
+ @staticmethod
97
+ async def delete_queryset(queryset: QuerySet):
99
98
  """
100
99
  考虑到不一定会集成已有的模型,删除根据是否存在字段来判断
101
100
  :param queryset:
@@ -103,7 +102,7 @@ class BaseView(BaseMixin, ResponseMixin, ToolMixin, DBModelMixin):
103
102
  """
104
103
  if settings.app_settings.WHETHER_DELETE_FIELD in queryset.fields:
105
104
  await queryset.update(**{
106
- settings.app_settings.WHETHER_DELETE_FIELD: self._delete_value()
105
+ settings.app_settings.WHETHER_DELETE_FIELD: BaseView._delete_value()
107
106
  })
108
107
  else:
109
108
  await queryset.delete()
@@ -155,20 +154,21 @@ class BaseView(BaseMixin, ResponseMixin, ToolMixin, DBModelMixin):
155
154
  )
156
155
  return schema.model_validate(model_dict)
157
156
 
157
+ @classmethod
158
158
  async def check_unique_field(
159
- self,
159
+ cls,
160
160
  request_data: Union[BaseModel, Model], # 创建或修改的参数数据
161
161
  model_class: Type[Model], # 数据库模型
162
162
  model: Union[Model, None] = None
163
163
  ):
164
- check_unique_fields = self._get_unique_fields(model_class) + self._get_foreign_key_fields(model_class)
165
- check_unique_together_fields = self._get_unique_together_fields(model_class)
164
+ check_unique_fields = cls._get_unique_fields(model_class) + cls._get_foreign_key_fields(model_class)
165
+ check_unique_together_fields = cls._get_unique_together_fields(model_class)
166
166
  for unique_field in check_unique_fields:
167
167
  if hasattr(request_data, unique_field):
168
168
  if model and getattr(request_data, unique_field) == getattr(model, unique_field):
169
169
  continue
170
170
  if await model_class.filter(**{unique_field: getattr(request_data, unique_field)}).first():
171
- self.fail(msg=f"{self.get_field_description(model_class, unique_field)}已存在相同值:{getattr(request_data, unique_field)}")
171
+ return cls.error(msg=f"{cls.get_field_description(model_class, unique_field)}已存在相同值:{getattr(request_data, unique_field)}")
172
172
  for unique_together_fields in check_unique_together_fields:
173
173
  filter_fields = {}
174
174
  is_equal = True
@@ -193,8 +193,8 @@ class BaseView(BaseMixin, ResponseMixin, ToolMixin, DBModelMixin):
193
193
  if await model_class.filter(**filter_fields).first():
194
194
  if settings.app_settings.WHETHER_DELETE_FIELD in description_fields:
195
195
  description_fields.remove(settings.app_settings.WHETHER_DELETE_FIELD)
196
- return self.error(
197
- msg=f"{self.get_field_description(model_class, description_fields)}已存在相同值:{[filter_fields.get(field) for field in description_fields]}"
196
+ return cls.error(
197
+ msg=f"{cls.get_field_description(model_class, description_fields)}已存在相同值:{[filter_fields.get(field) for field in description_fields]}"
198
198
  )
199
199
 
200
200
  # @staticmethod
@@ -33,6 +33,7 @@ class GetAllView(BaseView, GetMixin):
33
33
  filter_schema: Optional[Type[PYDANTIC_SCHEMA]] = None
34
34
  filter_fields: Union[None, list] = None
35
35
  order_by_fields: Union[None, list] = None
36
+ auto_add_id_order: bool = True
36
37
  """
37
38
  get_all_route: 获取详情路由开关,可以放依赖函数列表
38
39
  get_all_schema: 返回序列化
@@ -45,6 +46,7 @@ class GetAllView(BaseView, GetMixin):
45
46
  filter_fields: 筛选对应字段
46
47
  example: name__contains or (create_at__gt, datetime) or (create_at__gt, datetime, create_time)
47
48
  order_by_fields: 排序对应字段
49
+ auto_add_id_order: 是否自动在 order_by_fields 后面追加id排序
48
50
  """
49
51
 
50
52
  async def get_all(self, search: Optional[str], filters: dict, *args, **kwargs) -> Union[BaseModel, dict, None]:
@@ -60,10 +62,11 @@ class GetAllView(BaseView, GetMixin):
60
62
  queryset = await self.filter_queryset(queryset, filters, *args, **kwargs)
61
63
  queryset = self.filter_controller.query(queryset=queryset, values=filters)
62
64
  queryset = queryset.prefetch_related(*self.prefetch_related_fields.keys())
65
+ if self.auto_add_id_order:
66
+ self.order_by_fields = self.order_by_fields or []
67
+ self.order_by_fields.append("id")
63
68
  if self.order_by_fields:
64
- queryset = queryset.order_by(*self.order_by_fields, "id")
65
- else:
66
- queryset = queryset.order_by("id")
69
+ queryset = queryset.order_by(*self.order_by_fields)
67
70
 
68
71
  return queryset
69
72
 
@@ -143,7 +146,7 @@ class GetAllView(BaseView, GetMixin):
143
146
  key_builder=generate_key_builder))
144
147
  async def route(
145
148
  request: Request,
146
- paginator=Depends(paginator_deps()),
149
+ paginator = Depends(paginator_deps()),
147
150
  search: Optional[str] = Depends(search_params_deps(self.search_fields)),
148
151
  filters: dict = Depends(filter_params_deps(model_class=self.model_class, fields=self.filter_fields, schema=self.filter_schema)),
149
152
  token: Optional[str] = Depends(OAuth2PasswordBearer(tokenUrl="token", auto_error=False)),
@@ -166,9 +169,15 @@ class GetAllView(BaseView, GetMixin):
166
169
  return
167
170
  self.search_controller = SearchController(self.get_base_filter(self.search_fields))
168
171
  self.filter_controller = FilterController(self.get_base_filter(self.filter_fields, self.filter_schema))
172
+ get_all_schema_include = []
173
+ if self.prefetch_related_fields:
174
+ for key, value in self.prefetch_related_fields.items():
175
+ if value and type(value) in [list, tuple, set]:
176
+ for val in value:
177
+ get_all_schema_include.append(key + "__" + val)
169
178
  self.get_all_schema = self.get_all_schema or get_all_schema_factory(
170
- self.model_class) or self.get_one_schema if hasattr(self, "get_one_schema") else get_one_schema_factory(
171
- self.model_class)
179
+ self.model_class, include=get_all_schema_include
180
+ ) or self.get_one_schema if hasattr(self, "get_one_schema") else get_one_schema_factory(self.model_class)
172
181
  self.get_page_schema = get_page_schema_factory(self.get_all_schema)
173
182
  self.get_list_schema = get_list_schema_factory(self.get_all_schema)
174
183
  self.get_all_response_schema = response_factory(self.get_page_schema, name="GetPage")
@@ -38,7 +38,7 @@ class GetOneView(BaseView, GetMixin):
38
38
  """
39
39
 
40
40
  async def get_one(self, pk: str, *args, **kwargs):
41
- model = await self.get_object(pk, is_with_prefetch=self.is_with_prefetch)
41
+ model = await self.get_object(pk, self.model_class, is_with_prefetch=self.is_with_prefetch)
42
42
  if self.is_with_prefetch:
43
43
  await self.setattr_model(model, prefetch_related_fields=self.prefetch_related_fields, *args, **kwargs)
44
44
 
@@ -66,7 +66,16 @@ class GetOneView(BaseView, GetMixin):
66
66
  def _handler_get_one_settings(self):
67
67
  if not self.get_one_route:
68
68
  return
69
- self.get_one_schema = self.schema or self.get_one_schema or get_one_schema_factory(model_class=self.model_class)
69
+ get_one_schema_include = []
70
+ if self.is_with_prefetch and self.prefetch_related_fields:
71
+ for key, value in self.prefetch_related_fields.items():
72
+ if value and type(value) in [list, tuple, set]:
73
+ for val in value:
74
+ get_one_schema_include.append(key + "__" + val)
75
+ self.get_one_schema = self.schema or self.get_one_schema or get_one_schema_factory(
76
+ model_class=self.model_class,
77
+ include=get_one_schema_include,
78
+ )
70
79
  self.get_one_response_schema = response_factory(self.get_one_schema, name="GetOne")
71
80
  doc = self.get_one.__doc__
72
81
  summary = doc.strip().split("\n")[0] if self.get_one.__doc__ else "Get One"
@@ -189,7 +189,7 @@ class GetRelationView(BaseView):
189
189
  self.get_relation_response_schema = response_factory(self.get_relation_page_schema, name="GetPage")
190
190
  doc = self.get_relation.__doc__
191
191
  summary = doc.strip().split("\n")[0] if doc else f"Get {self.model_class.__name__.title()}"
192
- path = f"/get-{self.relation_id_name.strip('_id')}-by-{self.path_id_name.strip('_id')}/{'{pk}'}" if settings.app_settings.ROUTER_WHETHER_ADD_SUFFIX else ""
192
+ path = f"/get-{self.relation_id_name.removesuffix('_id')}-by-{self.path_id_name.removesuffix('_id')}/{'{pk}'}" if settings.app_settings.ROUTER_WHETHER_ADD_SUFFIX else ""
193
193
  self._add_api_route(
194
194
  path=path,
195
195
  endpoint=self._get_relation_decorator(),
@@ -52,7 +52,8 @@ class BaseMixin(Generic[T], APIRouter, ABC):
52
52
  # if hasattr(self, "_handler_sql_get_all_settings"):
53
53
  # getattr(self, "_handler_sql_get_all_settings")()
54
54
 
55
- def get_base_filter(self, fields: list, schema: Optional[PYDANTIC_SCHEMA] = None) -> list:
55
+ @staticmethod
56
+ def get_base_filter(fields: list, schema: Optional[PYDANTIC_SCHEMA] = None) -> list:
56
57
  bast_filter_list = []
57
58
  if fields:
58
59
  bast_filter_list += [BaseFilter(field) if not isinstance(field, BaseFilter) else field for field in fields]
@@ -29,7 +29,7 @@ class DBModelMixin:
29
29
  if type(fields) == str:
30
30
  try:
31
31
  field_info = model_class._meta.fields_map.get(fields)
32
- field_info_fk = model_class._meta.fields_map.get(fields.rstrip("_id"))
32
+ field_info_fk = model_class._meta.fields_map.get(fields.removesuffix("_id"))
33
33
  if field_info:
34
34
  return field_info.description or ""
35
35
  elif fields.endswith("_id") and field_info_fk:
@@ -75,7 +75,7 @@ class UpdateRelationView(BaseView, SaveMixin):
75
75
  self.update_relation_response_schema = response_factory(name="UpdateRelation")
76
76
  doc = self.update_relation.__doc__
77
77
  summary = doc.strip().split("\n")[0] if doc else f"Update {self.model_class.__name__.title()}"
78
- path = f"/update-{self.relation_id_name.strip('_id')}-by-{self.path_id_name.strip('_id')}/{'{pk}'}" if settings.app_settings.ROUTER_WHETHER_ADD_SUFFIX else "/{pk}"
78
+ path = f"/update-{self.relation_id_name.removesuffix('_id')}-by-{self.path_id_name.removesuffix('_id')}/{'{pk}'}" if settings.app_settings.ROUTER_WHETHER_ADD_SUFFIX else "/{pk}"
79
79
  self._add_api_route(
80
80
  path=path,
81
81
  endpoint=self._update_relation_decorator(),
@@ -29,7 +29,7 @@ class UpdateView(BaseView, SaveMixin):
29
29
 
30
30
  @atomic()
31
31
  async def update(self, pk: str, request_data, *args, **kwargs):
32
- model = await self.get_object(pk)
32
+ model = await self.get_object(pk, self.model_class)
33
33
 
34
34
  request_data = await self.set_update_fields(request_data=request_data, *args, **kwargs)
35
35
 
@@ -69,17 +69,14 @@ class BaseFilter:
69
69
  for f in filter_str[1:]:
70
70
  if type(f) == type:
71
71
  field_type = f
72
- continue
73
- if type(f) == str:
72
+ elif type(f) == str:
74
73
  filter_field = f
75
- continue
76
- if callable(f):
74
+ elif callable(f):
77
75
  filter_func = f
78
76
  if not filter_field:
79
- if settings.app_settings.FILTER_UNDERLINE_WHETHER_DOUBLE_TO_SINGLE:
80
- filter_field = model_field.replace("__", "_")
81
- else:
82
- filter_field = model_field
77
+ filter_field = model_field
78
+ if settings.app_settings.FILTER_UNDERLINE_WHETHER_DOUBLE_TO_SINGLE:
79
+ filter_field = filter_field.replace("__", "_")
83
80
 
84
81
  self.field_type = field_type
85
82
  self.model_field = model_field
@@ -37,7 +37,7 @@ class RPCController(ResponseMixin):
37
37
  for response_param_value in request_param_value_list:
38
38
  if type(response_param_value) == str:
39
39
  response_field_name = response_param_value
40
- response_field_alias_name = model_field.rstrip("_id") + "_" + response_param_value
40
+ response_field_alias_name = model_field.removesuffix("_id") + "_" + response_param_value
41
41
  elif type(response_param_value) == tuple and len(response_param_value) > 1:
42
42
  response_field_name = response_param_value[0]
43
43
  response_field_alias_name = response_param_value[1]
@@ -5,6 +5,7 @@ from pydantic import create_model
5
5
  from pydantic.fields import FieldInfo
6
6
 
7
7
  from fastgenerateapi.pydantic_utils.base_model import BaseModel, model_config
8
+ from fastgenerateapi.schemas_factory.common_function import get_validate_dict
8
9
  from fastgenerateapi.settings.all_settings import settings
9
10
  from fastgenerateapi.utils.str_util import parse_str_to_int, parse_str_to_bool
10
11
 
@@ -42,7 +43,12 @@ def paginator_deps():
42
43
  FieldInfo(default=Query(settings.app_settings.DEFAULT_WHETHER_PAGE), title="是否分页", description="是否分页")
43
44
  ),
44
45
  }
45
- pagination_model: Type[BaseModel] = create_model("paginator", **fields, __config__=model_config)
46
+ pagination_model: Type[BaseModel] = create_model(
47
+ "paginator",
48
+ **fields,
49
+ __config__=model_config,
50
+ __validators__=get_validate_dict(),
51
+ )
46
52
 
47
53
  def pagination_deps(paginator: pagination_model = Depends(pagination_model)) -> pagination_model:
48
54
  current_page_value = parse_str_to_int(getattr(paginator, settings.app_settings.CURRENT_PAGE_FIELD))
@@ -5,6 +5,7 @@ from pydantic.fields import FieldInfo
5
5
 
6
6
  from fastgenerateapi.pydantic_utils.base_model import model_config
7
7
  from fastgenerateapi.pydantic_utils.base_model import ModelConfig
8
+ from fastgenerateapi.schemas_factory.common_function import get_validate_dict
8
9
  from fastgenerateapi.settings.all_settings import settings
9
10
 
10
11
  from pydantic import BaseModel, create_model
@@ -26,7 +27,12 @@ def tree_params_deps():
26
27
  )
27
28
  }
28
29
 
29
- filter_tree_params_model: Type[BaseModel] = create_model("TreeFilterParams", __config__=model_config, **model_fields)
30
+ filter_tree_params_model: Type[BaseModel] = create_model(
31
+ "TreeFilterParams",
32
+ **model_fields,
33
+ __config__=model_config,
34
+ __validators__=get_validate_dict(),
35
+ )
30
36
 
31
37
  def filter_query(filter_params: filter_tree_params_model = Depends(filter_tree_params_model)) -> Optional[str]:
32
38
  """
@@ -6,6 +6,7 @@ import fastapi
6
6
  from fastapi import params
7
7
  from fastapi.dependencies.models import Dependant
8
8
  from fastapi.dependencies.utils import get_sub_dependant
9
+ from pydantic.alias_generators import to_snake, to_camel
9
10
 
10
11
 
11
12
  def get_param_sub_dependant(
@@ -25,6 +26,14 @@ def get_param_sub_dependant(
25
26
  )
26
27
  for query_param in dependant.query_params:
27
28
  query_param_field = depends.dependency.model_fields.get(query_param.name)
29
+ if not query_param_field:
30
+ snake_name = to_snake(query_param.name)
31
+ if query_param.name != snake_name:
32
+ query_param_field = depends.dependency.model_fields.get(snake_name)
33
+ if not query_param_field:
34
+ camel_name = to_camel(query_param.name)
35
+ if query_param.name != camel_name:
36
+ query_param_field = depends.dependency.model_fields.get(camel_name)
28
37
  if query_param_field:
29
38
  query_param.field_info.description = query_param_field.description or query_param_field.title or ""
30
39
  return dependant
@@ -1,7 +1,7 @@
1
1
  import importlib
2
- from typing import List, Optional
2
+ from typing import List, Optional, Any
3
3
 
4
- from pydantic import BaseModel as PydanticBaseModel, BaseConfig, Field, ConfigDict
4
+ from pydantic import BaseModel as PydanticBaseModel, BaseConfig, Field, ConfigDict, field_validator
5
5
 
6
6
  from fastgenerateapi.pydantic_utils.json_encoders import JSON_ENCODERS
7
7
  from fastgenerateapi.settings.all_settings import settings
@@ -54,6 +54,13 @@ model_config = ConfigDict(
54
54
  class BaseModel(PydanticBaseModel):
55
55
  model_config = model_config
56
56
 
57
+ @field_validator('*', mode='before') # '*' 表示匹配所有字段
58
+ def empty_str_to_none(cls, v: Any) -> Any:
59
+ """将空字符串转换为None,其他值保持不变"""
60
+ if isinstance(v, str) and v.strip() == "": # 处理纯空字符串或仅含空格的字符串
61
+ return None
62
+ return v
63
+
57
64
 
58
65
  class IdList(BaseModel):
59
66
  id_list: List[str] = Field([], description="id数组")
@@ -1,6 +1,6 @@
1
1
  from typing import Type, Optional, Union
2
2
 
3
- from pydantic import validator
3
+ from pydantic import validator, field_validator
4
4
  from pydantic.fields import FieldInfo
5
5
  from tortoise import Model
6
6
 
@@ -11,18 +11,20 @@ from fastgenerateapi.settings.all_settings import settings
11
11
  def get_field_info(value, description="", default_field_type=None) -> (Type, FieldInfo):
12
12
  if value.pk:
13
13
  return str, FieldInfo()
14
+
15
+ required = False
14
16
  if value.field_type is None:
15
17
  if default_field_type:
16
- field_type = Optional[default_field_type]
18
+ value.field_type = default_field_type
17
19
  else:
18
- field_type = Optional[str]
20
+ value.field_type = str
21
+ if hasattr(value, "null") and value.null:
22
+ field_type = Optional[value.field_type]
19
23
  else:
20
- if hasattr(value, "null") and value.null:
21
- field_type = Optional[value.field_type]
22
- else:
23
- field_type = value.field_type
24
- field_info_dict = {}
25
- if hasattr(value, "default") and not hasattr(value.default, '__call__'):
24
+ field_type = value.field_type
25
+ required = True
26
+ field_info_dict = {"required": required}
27
+ if not required and hasattr(value, "default") and not hasattr(value.default, '__call__'):
26
28
  field_info_dict.setdefault("default", value.default)
27
29
  if hasattr(value, "description"):
28
30
  field_info_dict.setdefault("description", description + (value.description or ""))
@@ -111,13 +113,25 @@ def get_dict_from_pydanticmeta(model_class: Type[Model], data: Union[list, tuple
111
113
  return fields_info
112
114
 
113
115
 
116
+ def get_validate_dict() -> dict:
117
+
118
+ def empty_str_to_none(v, values):
119
+ if isinstance(v, str) and v.strip() == "":
120
+ return None
121
+ return v
122
+
123
+ validator_dict = {
124
+ "empty_str_to_none": field_validator('*', mode='before')(empty_str_to_none)
125
+ }
126
+
127
+ return validator_dict
128
+
129
+
114
130
  def get_validate_dict_from_fields(fields_info: dict) -> dict:
115
131
  validator_dict = {}
116
132
 
117
133
  def remove_blank_strings(v, values):
118
- if isinstance(v, str):
119
- v = v.strip()
120
- if v == "":
134
+ if isinstance(v, str) and v.strip() == "":
121
135
  return None
122
136
  return v
123
137
 
@@ -129,3 +143,5 @@ def get_validate_dict_from_fields(fields_info: dict) -> dict:
129
143
  return validator_dict
130
144
 
131
145
 
146
+
147
+
@@ -1,4 +1,4 @@
1
- from typing import Type, Union
1
+ from typing import Type, Union, Optional
2
2
 
3
3
  from fastapi import Query
4
4
  from pydantic import BaseModel, create_model
@@ -9,6 +9,8 @@ from fastgenerateapi.api_view.mixin.dbmodel_mixin import DBModelMixin
9
9
  from fastgenerateapi.controller.filter_controller import BaseFilter
10
10
  from tortoise import Model
11
11
 
12
+ from fastgenerateapi.schemas_factory.common_function import get_validate_dict
13
+
12
14
 
13
15
  def filter_schema_factory(model_class: Type[Model], fields: list[str, tuple[str, Type], BaseFilter] = None):
14
16
  """
@@ -24,7 +26,7 @@ def filter_schema_factory(model_class: Type[Model], fields: list[str, tuple[str,
24
26
  description = DBModelMixin.get_field_description(model_class, field_info.model_field)
25
27
  model_fields.update({
26
28
  f: (
27
- Union[t, str],
29
+ Optional[t],
28
30
  FieldInfo(
29
31
  title=f"{description}",
30
32
  default=Query("", description=description),
@@ -35,7 +37,8 @@ def filter_schema_factory(model_class: Type[Model], fields: list[str, tuple[str,
35
37
  filter_params_model: Type[BaseModel] = create_model(
36
38
  model_class.__name__+"CommonFilterParams",
37
39
  **model_fields,
38
- __config__=model_config
40
+ __config__=model_config,
41
+ __validators__=get_validate_dict(),
39
42
  )
40
43
 
41
44
  return filter_params_model
@@ -7,7 +7,7 @@ from fastgenerateapi.pydantic_utils.base_settings import BaseSettings
7
7
 
8
8
 
9
9
  class AppSettings(BaseSettings):
10
- # 字段配置 老版本(pydantic.utils.to_lower_camel)
10
+ # 字段配置 驼峰格式:新版本(pydantic.alias_generators:to_camel)、老版本(pydantic.utils.to_lower_camel)
11
11
  ALIAS_GENERATOR: Optional[str] = Field(default="pydantic.alias_generators.to_snake", description="序列化参数命名方法路径")
12
12
 
13
13
  # 分页对应字段以及配置默认值
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: fastgenerateapi
3
- Version: 1.1.13
3
+ Version: 1.1.16
4
4
  Summary: FastAPIView Class View
5
5
  Author: ShiLiang
6
6
  Author-email: 2509144896@qq.com
@@ -24,15 +24,17 @@ License-File: LICENSE
24
24
  [Github](https://github.com/ShiLiangAPI)
25
25
 
26
26
  #### 版本说明
27
- 最新版本:v1.1.7+
27
+ [pydantic-v2]
28
+ 稳定版本:v1.1.15+
28
29
  建议python版本:3.9+
29
30
  推荐fastapi版本:0.103.2+
30
31
  推荐pydantic版本:2.11.5+
31
32
 
33
+ [pydantic-v1]
32
34
  稳定版本:v0.0.21
33
35
  建议python版本:3.9+
34
36
  推荐fastapi版本:0.97.0
35
-
37
+ 推荐pydantic版本:1.10.9+
36
38
 
37
39
 
38
40
  ###### FastGenerateApi 使用说明