clear-skies 1.22.31__py3-none-any.whl → 2.0.1__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.

Potentially problematic release.


This version of clear-skies might be problematic. Click here for more details.

Files changed (345) hide show
  1. {clear_skies-1.22.31.dist-info → clear_skies-2.0.1.dist-info}/METADATA +12 -14
  2. clear_skies-2.0.1.dist-info/RECORD +249 -0
  3. {clear_skies-1.22.31.dist-info → clear_skies-2.0.1.dist-info}/WHEEL +1 -1
  4. clearskies/__init__.py +42 -25
  5. clearskies/action.py +7 -0
  6. clearskies/authentication/__init__.py +8 -41
  7. clearskies/authentication/authentication.py +46 -0
  8. clearskies/authentication/authorization.py +8 -9
  9. clearskies/authentication/authorization_pass_through.py +11 -9
  10. clearskies/authentication/jwks.py +133 -58
  11. clearskies/authentication/public.py +3 -38
  12. clearskies/authentication/secret_bearer.py +516 -54
  13. clearskies/autodoc/formats/oai3_json/__init__.py +1 -1
  14. clearskies/autodoc/formats/oai3_json/oai3_json.py +9 -7
  15. clearskies/autodoc/formats/oai3_json/parameter.py +6 -3
  16. clearskies/autodoc/formats/oai3_json/request.py +7 -5
  17. clearskies/autodoc/formats/oai3_json/response.py +7 -4
  18. clearskies/autodoc/formats/oai3_json/schema/object.py +4 -1
  19. clearskies/autodoc/request/__init__.py +2 -0
  20. clearskies/autodoc/request/header.py +4 -6
  21. clearskies/autodoc/request/json_body.py +4 -6
  22. clearskies/autodoc/request/parameter.py +8 -0
  23. clearskies/autodoc/request/request.py +7 -4
  24. clearskies/autodoc/request/url_parameter.py +4 -6
  25. clearskies/autodoc/request/url_path.py +4 -6
  26. clearskies/autodoc/schema/__init__.py +4 -2
  27. clearskies/autodoc/schema/array.py +5 -6
  28. clearskies/autodoc/schema/boolean.py +4 -10
  29. clearskies/autodoc/schema/date.py +0 -3
  30. clearskies/autodoc/schema/datetime.py +1 -4
  31. clearskies/autodoc/schema/double.py +0 -3
  32. clearskies/autodoc/schema/enum.py +4 -2
  33. clearskies/autodoc/schema/integer.py +4 -9
  34. clearskies/autodoc/schema/long.py +0 -3
  35. clearskies/autodoc/schema/number.py +4 -9
  36. clearskies/autodoc/schema/object.py +5 -7
  37. clearskies/autodoc/schema/password.py +0 -3
  38. clearskies/autodoc/schema/schema.py +11 -0
  39. clearskies/autodoc/schema/string.py +4 -10
  40. clearskies/backends/__init__.py +55 -20
  41. clearskies/backends/api_backend.py +1100 -284
  42. clearskies/backends/backend.py +53 -84
  43. clearskies/backends/cursor_backend.py +236 -186
  44. clearskies/backends/memory_backend.py +519 -226
  45. clearskies/backends/secrets_backend.py +75 -31
  46. clearskies/column.py +1229 -0
  47. clearskies/columns/__init__.py +71 -0
  48. clearskies/columns/audit.py +205 -0
  49. clearskies/columns/belongs_to_id.py +483 -0
  50. clearskies/columns/belongs_to_model.py +128 -0
  51. clearskies/columns/belongs_to_self.py +105 -0
  52. clearskies/columns/boolean.py +109 -0
  53. clearskies/columns/category_tree.py +275 -0
  54. clearskies/columns/category_tree_ancestors.py +51 -0
  55. clearskies/columns/category_tree_children.py +127 -0
  56. clearskies/columns/category_tree_descendants.py +48 -0
  57. clearskies/columns/created.py +94 -0
  58. clearskies/columns/created_by_authorization_data.py +116 -0
  59. clearskies/columns/created_by_header.py +99 -0
  60. clearskies/columns/created_by_ip.py +92 -0
  61. clearskies/columns/created_by_routing_data.py +96 -0
  62. clearskies/columns/created_by_user_agent.py +92 -0
  63. clearskies/columns/date.py +230 -0
  64. clearskies/columns/datetime.py +278 -0
  65. clearskies/columns/email.py +76 -0
  66. clearskies/columns/float.py +149 -0
  67. clearskies/columns/has_many.py +505 -0
  68. clearskies/columns/has_many_self.py +56 -0
  69. clearskies/columns/has_one.py +14 -0
  70. clearskies/columns/integer.py +156 -0
  71. clearskies/columns/json.py +122 -0
  72. clearskies/columns/many_to_many_ids.py +333 -0
  73. clearskies/columns/many_to_many_ids_with_data.py +270 -0
  74. clearskies/columns/many_to_many_models.py +154 -0
  75. clearskies/columns/many_to_many_pivots.py +133 -0
  76. clearskies/columns/phone.py +158 -0
  77. clearskies/columns/select.py +91 -0
  78. clearskies/columns/string.py +98 -0
  79. clearskies/columns/timestamp.py +160 -0
  80. clearskies/columns/updated.py +110 -0
  81. clearskies/columns/uuid.py +86 -0
  82. clearskies/configs/README.md +105 -0
  83. clearskies/configs/__init__.py +162 -0
  84. clearskies/configs/actions.py +43 -0
  85. clearskies/configs/any.py +13 -0
  86. clearskies/configs/any_dict.py +22 -0
  87. clearskies/configs/any_dict_or_callable.py +23 -0
  88. clearskies/configs/authentication.py +23 -0
  89. clearskies/configs/authorization.py +23 -0
  90. clearskies/configs/boolean.py +16 -0
  91. clearskies/configs/boolean_or_callable.py +18 -0
  92. clearskies/configs/callable_config.py +18 -0
  93. clearskies/configs/columns.py +34 -0
  94. clearskies/configs/conditions.py +30 -0
  95. clearskies/configs/config.py +24 -0
  96. clearskies/configs/datetime.py +18 -0
  97. clearskies/configs/datetime_or_callable.py +19 -0
  98. clearskies/configs/endpoint.py +23 -0
  99. clearskies/configs/endpoint_list.py +28 -0
  100. clearskies/configs/float.py +16 -0
  101. clearskies/configs/float_or_callable.py +18 -0
  102. clearskies/configs/integer.py +16 -0
  103. clearskies/configs/integer_or_callable.py +18 -0
  104. clearskies/configs/joins.py +30 -0
  105. clearskies/configs/list_any_dict.py +30 -0
  106. clearskies/configs/list_any_dict_or_callable.py +31 -0
  107. clearskies/configs/model_class.py +35 -0
  108. clearskies/configs/model_column.py +65 -0
  109. clearskies/configs/model_columns.py +56 -0
  110. clearskies/configs/model_destination_name.py +25 -0
  111. clearskies/configs/model_to_id_column.py +43 -0
  112. clearskies/configs/readable_model_column.py +9 -0
  113. clearskies/configs/readable_model_columns.py +9 -0
  114. clearskies/configs/schema.py +23 -0
  115. clearskies/configs/searchable_model_columns.py +9 -0
  116. clearskies/configs/security_headers.py +39 -0
  117. clearskies/configs/select.py +26 -0
  118. clearskies/configs/select_list.py +47 -0
  119. clearskies/configs/string.py +29 -0
  120. clearskies/configs/string_dict.py +32 -0
  121. clearskies/configs/string_list.py +32 -0
  122. clearskies/configs/string_list_or_callable.py +35 -0
  123. clearskies/configs/string_or_callable.py +18 -0
  124. clearskies/configs/timedelta.py +18 -0
  125. clearskies/configs/timezone.py +18 -0
  126. clearskies/configs/url.py +23 -0
  127. clearskies/configs/validators.py +45 -0
  128. clearskies/configs/writeable_model_column.py +9 -0
  129. clearskies/configs/writeable_model_columns.py +9 -0
  130. clearskies/configurable.py +76 -0
  131. clearskies/contexts/__init__.py +8 -8
  132. clearskies/contexts/cli.py +8 -41
  133. clearskies/contexts/context.py +91 -56
  134. clearskies/contexts/wsgi.py +16 -29
  135. clearskies/contexts/wsgi_ref.py +53 -0
  136. clearskies/di/__init__.py +10 -7
  137. clearskies/di/additional_config.py +115 -4
  138. clearskies/di/additional_config_auto_import.py +12 -0
  139. clearskies/di/di.py +742 -121
  140. clearskies/di/inject/__init__.py +23 -0
  141. clearskies/di/inject/by_class.py +21 -0
  142. clearskies/di/inject/by_name.py +18 -0
  143. clearskies/di/inject/di.py +13 -0
  144. clearskies/di/inject/environment.py +14 -0
  145. clearskies/di/inject/input_output.py +20 -0
  146. clearskies/di/inject/now.py +13 -0
  147. clearskies/di/inject/requests.py +13 -0
  148. clearskies/di/inject/secrets.py +14 -0
  149. clearskies/di/inject/utcnow.py +13 -0
  150. clearskies/di/inject/uuid.py +15 -0
  151. clearskies/di/injectable.py +29 -0
  152. clearskies/di/injectable_properties.py +131 -0
  153. clearskies/end.py +183 -0
  154. clearskies/endpoint.py +1310 -0
  155. clearskies/endpoint_group.py +310 -0
  156. clearskies/endpoints/__init__.py +23 -0
  157. clearskies/endpoints/advanced_search.py +526 -0
  158. clearskies/endpoints/callable.py +388 -0
  159. clearskies/endpoints/create.py +202 -0
  160. clearskies/endpoints/delete.py +139 -0
  161. clearskies/endpoints/get.py +275 -0
  162. clearskies/endpoints/health_check.py +181 -0
  163. clearskies/endpoints/list.py +573 -0
  164. clearskies/endpoints/restful_api.py +427 -0
  165. clearskies/endpoints/simple_search.py +286 -0
  166. clearskies/endpoints/update.py +190 -0
  167. clearskies/environment.py +5 -3
  168. clearskies/exceptions/__init__.py +17 -0
  169. clearskies/{handlers/exceptions/input_error.py → exceptions/input_errors.py} +1 -1
  170. clearskies/exceptions/moved_permanently.py +3 -0
  171. clearskies/exceptions/moved_temporarily.py +3 -0
  172. clearskies/exceptions/not_found.py +2 -0
  173. clearskies/functional/__init__.py +2 -2
  174. clearskies/functional/routing.py +92 -0
  175. clearskies/functional/string.py +19 -11
  176. clearskies/functional/validations.py +61 -9
  177. clearskies/input_outputs/__init__.py +9 -7
  178. clearskies/input_outputs/cli.py +130 -142
  179. clearskies/input_outputs/exceptions/__init__.py +1 -1
  180. clearskies/input_outputs/headers.py +45 -0
  181. clearskies/input_outputs/input_output.py +91 -122
  182. clearskies/input_outputs/programmatic.py +69 -0
  183. clearskies/input_outputs/wsgi.py +23 -38
  184. clearskies/model.py +984 -183
  185. clearskies/parameters_to_properties.py +31 -0
  186. clearskies/query/__init__.py +12 -0
  187. clearskies/query/condition.py +223 -0
  188. clearskies/query/join.py +136 -0
  189. clearskies/query/query.py +196 -0
  190. clearskies/query/sort.py +27 -0
  191. clearskies/schema.py +82 -0
  192. clearskies/secrets/__init__.py +3 -31
  193. clearskies/secrets/additional_configs/mysql_connection_dynamic_producer.py +15 -4
  194. clearskies/secrets/additional_configs/mysql_connection_dynamic_producer_via_ssh_cert_bastion.py +11 -5
  195. clearskies/secrets/akeyless.py +88 -147
  196. clearskies/secrets/secrets.py +8 -8
  197. clearskies/security_header.py +15 -0
  198. clearskies/security_headers/__init__.py +8 -8
  199. clearskies/security_headers/cache_control.py +47 -110
  200. clearskies/security_headers/cors.py +40 -95
  201. clearskies/security_headers/csp.py +76 -151
  202. clearskies/security_headers/hsts.py +14 -16
  203. clearskies/test_base.py +8 -0
  204. clearskies/typing.py +11 -0
  205. clearskies/validator.py +37 -0
  206. clearskies/validators/__init__.py +33 -0
  207. clearskies/validators/after_column.py +62 -0
  208. clearskies/validators/before_column.py +13 -0
  209. clearskies/validators/in_the_future.py +32 -0
  210. clearskies/validators/in_the_future_at_least.py +11 -0
  211. clearskies/validators/in_the_future_at_most.py +10 -0
  212. clearskies/validators/in_the_past.py +32 -0
  213. clearskies/validators/in_the_past_at_least.py +10 -0
  214. clearskies/validators/in_the_past_at_most.py +10 -0
  215. clearskies/validators/maximum_length.py +26 -0
  216. clearskies/validators/maximum_value.py +29 -0
  217. clearskies/validators/minimum_length.py +26 -0
  218. clearskies/validators/minimum_value.py +29 -0
  219. clearskies/validators/required.py +35 -0
  220. clearskies/validators/timedelta.py +59 -0
  221. clearskies/validators/unique.py +31 -0
  222. clear_skies-1.22.31.dist-info/RECORD +0 -214
  223. clearskies/application.py +0 -29
  224. clearskies/authentication/auth0_jwks.py +0 -118
  225. clearskies/authentication/auth_exception.py +0 -2
  226. clearskies/authentication/jwks_jwcrypto.py +0 -51
  227. clearskies/backends/api_get_only_backend.py +0 -48
  228. clearskies/backends/example_backend.py +0 -43
  229. clearskies/backends/file_backend.py +0 -48
  230. clearskies/backends/json_backend.py +0 -7
  231. clearskies/backends/restful_api_advanced_search_backend.py +0 -103
  232. clearskies/binding_config.py +0 -16
  233. clearskies/column_types/__init__.py +0 -203
  234. clearskies/column_types/audit.py +0 -249
  235. clearskies/column_types/belongs_to.py +0 -271
  236. clearskies/column_types/boolean.py +0 -60
  237. clearskies/column_types/category_tree.py +0 -304
  238. clearskies/column_types/column.py +0 -373
  239. clearskies/column_types/created.py +0 -26
  240. clearskies/column_types/created_by_authorization_data.py +0 -26
  241. clearskies/column_types/created_by_header.py +0 -24
  242. clearskies/column_types/created_by_ip.py +0 -17
  243. clearskies/column_types/created_by_routing_data.py +0 -25
  244. clearskies/column_types/created_by_user_agent.py +0 -17
  245. clearskies/column_types/created_micro.py +0 -26
  246. clearskies/column_types/datetime.py +0 -109
  247. clearskies/column_types/datetime_micro.py +0 -12
  248. clearskies/column_types/email.py +0 -18
  249. clearskies/column_types/float.py +0 -43
  250. clearskies/column_types/has_many.py +0 -179
  251. clearskies/column_types/has_one.py +0 -60
  252. clearskies/column_types/integer.py +0 -41
  253. clearskies/column_types/json.py +0 -25
  254. clearskies/column_types/many_to_many.py +0 -278
  255. clearskies/column_types/many_to_many_with_data.py +0 -162
  256. clearskies/column_types/phone.py +0 -48
  257. clearskies/column_types/select.py +0 -11
  258. clearskies/column_types/string.py +0 -24
  259. clearskies/column_types/timestamp.py +0 -73
  260. clearskies/column_types/updated.py +0 -26
  261. clearskies/column_types/updated_micro.py +0 -26
  262. clearskies/column_types/uuid.py +0 -25
  263. clearskies/columns.py +0 -123
  264. clearskies/condition_parser.py +0 -172
  265. clearskies/contexts/build_context.py +0 -54
  266. clearskies/contexts/convert_to_application.py +0 -190
  267. clearskies/contexts/extract_handler.py +0 -37
  268. clearskies/contexts/test.py +0 -94
  269. clearskies/decorators/__init__.py +0 -41
  270. clearskies/decorators/allow_non_json_bodies.py +0 -9
  271. clearskies/decorators/auth0_jwks.py +0 -22
  272. clearskies/decorators/authorization.py +0 -10
  273. clearskies/decorators/binding_classes.py +0 -9
  274. clearskies/decorators/binding_modules.py +0 -9
  275. clearskies/decorators/bindings.py +0 -9
  276. clearskies/decorators/create.py +0 -10
  277. clearskies/decorators/delete.py +0 -10
  278. clearskies/decorators/docs.py +0 -14
  279. clearskies/decorators/get.py +0 -10
  280. clearskies/decorators/jwks.py +0 -26
  281. clearskies/decorators/merge.py +0 -124
  282. clearskies/decorators/patch.py +0 -10
  283. clearskies/decorators/post.py +0 -10
  284. clearskies/decorators/public.py +0 -11
  285. clearskies/decorators/response_headers.py +0 -10
  286. clearskies/decorators/return_raw_response.py +0 -9
  287. clearskies/decorators/schema.py +0 -10
  288. clearskies/decorators/secret_bearer.py +0 -24
  289. clearskies/decorators/security_headers.py +0 -10
  290. clearskies/di/standard_dependencies.py +0 -151
  291. clearskies/handlers/__init__.py +0 -41
  292. clearskies/handlers/advanced_search.py +0 -271
  293. clearskies/handlers/base.py +0 -479
  294. clearskies/handlers/callable.py +0 -192
  295. clearskies/handlers/create.py +0 -35
  296. clearskies/handlers/crud_by_method.py +0 -18
  297. clearskies/handlers/database_connector.py +0 -32
  298. clearskies/handlers/delete.py +0 -61
  299. clearskies/handlers/exceptions/__init__.py +0 -5
  300. clearskies/handlers/exceptions/not_found.py +0 -3
  301. clearskies/handlers/get.py +0 -156
  302. clearskies/handlers/health_check.py +0 -59
  303. clearskies/handlers/input_processing.py +0 -79
  304. clearskies/handlers/list.py +0 -530
  305. clearskies/handlers/mygrations.py +0 -82
  306. clearskies/handlers/request_method_routing.py +0 -47
  307. clearskies/handlers/restful_api.py +0 -218
  308. clearskies/handlers/routing.py +0 -62
  309. clearskies/handlers/schema_helper.py +0 -128
  310. clearskies/handlers/simple_routing.py +0 -206
  311. clearskies/handlers/simple_routing_route.py +0 -197
  312. clearskies/handlers/simple_search.py +0 -136
  313. clearskies/handlers/update.py +0 -102
  314. clearskies/handlers/write.py +0 -193
  315. clearskies/input_requirements/__init__.py +0 -78
  316. clearskies/input_requirements/after.py +0 -36
  317. clearskies/input_requirements/before.py +0 -36
  318. clearskies/input_requirements/in_the_future_at_least.py +0 -19
  319. clearskies/input_requirements/in_the_future_at_most.py +0 -19
  320. clearskies/input_requirements/in_the_past_at_least.py +0 -19
  321. clearskies/input_requirements/in_the_past_at_most.py +0 -19
  322. clearskies/input_requirements/maximum_length.py +0 -19
  323. clearskies/input_requirements/maximum_value.py +0 -19
  324. clearskies/input_requirements/minimum_length.py +0 -22
  325. clearskies/input_requirements/minimum_value.py +0 -19
  326. clearskies/input_requirements/required.py +0 -23
  327. clearskies/input_requirements/requirement.py +0 -25
  328. clearskies/input_requirements/time_delta.py +0 -38
  329. clearskies/input_requirements/unique.py +0 -18
  330. clearskies/mocks/__init__.py +0 -7
  331. clearskies/mocks/input_output.py +0 -124
  332. clearskies/mocks/models.py +0 -142
  333. clearskies/models.py +0 -350
  334. clearskies/security_headers/base.py +0 -12
  335. clearskies/tests/simple_api/models/__init__.py +0 -2
  336. clearskies/tests/simple_api/models/status.py +0 -23
  337. clearskies/tests/simple_api/models/user.py +0 -21
  338. clearskies/tests/simple_api/users_api.py +0 -64
  339. {clear_skies-1.22.31.dist-info → clear_skies-2.0.1.dist-info}/LICENSE +0 -0
  340. /clearskies/{contexts/bash.py → autodoc/py.typed} +0 -0
  341. /clearskies/{handlers/exceptions → exceptions}/authentication.py +0 -0
  342. /clearskies/{handlers/exceptions → exceptions}/authorization.py +0 -0
  343. /clearskies/{handlers/exceptions → exceptions}/client_error.py +0 -0
  344. /clearskies/{tests/__init__.py → input_outputs/py.typed} +0 -0
  345. /clearskies/{tests/simple_api/__init__.py → py.typed} +0 -0
@@ -0,0 +1,71 @@
1
+ from .audit import Audit
2
+ from .belongs_to_id import BelongsToId
3
+ from .belongs_to_model import BelongsToModel
4
+ from .belongs_to_self import BelongsToSelf
5
+ from .boolean import Boolean
6
+ from .category_tree import CategoryTree
7
+ from .category_tree_ancestors import CategoryTreeAncestors
8
+ from .category_tree_children import CategoryTreeChildren
9
+ from .category_tree_descendants import CategoryTreeDescendants
10
+ from .created import Created
11
+ from .created_by_authorization_data import CreatedByAuthorizationData
12
+ from .created_by_header import CreatedByHeader
13
+ from .created_by_ip import CreatedByIp
14
+ from .created_by_routing_data import CreatedByRoutingData
15
+ from .created_by_user_agent import CreatedByUserAgent
16
+ from .date import Date
17
+ from .datetime import Datetime
18
+ from .email import Email
19
+ from .float import Float
20
+ from .has_many import HasMany
21
+ from .has_many_self import HasManySelf
22
+ from .has_one import HasOne
23
+ from .integer import Integer
24
+ from .json import Json
25
+ from .many_to_many_ids import ManyToManyIds
26
+ from .many_to_many_ids_with_data import ManyToManyIdsWithData
27
+ from .many_to_many_models import ManyToManyModels
28
+ from .many_to_many_pivots import ManyToManyPivots
29
+ from .phone import Phone
30
+ from .select import Select
31
+ from .string import String
32
+ from .timestamp import Timestamp
33
+ from .updated import Updated
34
+ from .uuid import Uuid
35
+
36
+ __all__ = [
37
+ "Audit",
38
+ "BelongsToId",
39
+ "BelongsToModel",
40
+ "BelongsToSelf",
41
+ "Boolean",
42
+ "CategoryTree",
43
+ "CategoryTreeAncestors",
44
+ "CategoryTreeChildren",
45
+ "CategoryTreeDescendants",
46
+ "Created",
47
+ "CreatedByAuthorizationData",
48
+ "CreatedByHeader",
49
+ "CreatedByIp",
50
+ "CreatedByRoutingData",
51
+ "CreatedByUserAgent",
52
+ "Date",
53
+ "Datetime",
54
+ "Email",
55
+ "Float",
56
+ "HasMany",
57
+ "HasManySelf",
58
+ "HasOne",
59
+ "Integer",
60
+ "Json",
61
+ "ManyToManyIds",
62
+ "ManyToManyIdsWithData",
63
+ "ManyToManyModels",
64
+ "ManyToManyPivots",
65
+ "Phone",
66
+ "Select",
67
+ "String",
68
+ "Timestamp",
69
+ "Updated",
70
+ "Uuid",
71
+ ]
@@ -0,0 +1,205 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import Any
4
+
5
+ import clearskies.typing
6
+ from clearskies import configs, parameters_to_properties
7
+ from clearskies.column import Column
8
+ from clearskies.columns.has_many import HasMany
9
+
10
+
11
+ class Audit(HasMany):
12
+ """
13
+ Enables auditing for a model.
14
+
15
+ Specify the audit class to use and attach this column to your model. Everytime the model is created/updated/deleted,
16
+ the audit class will record the action and the changes. Your audit model must have the following columns:
17
+
18
+ | Name | type |
19
+ |-------------|----------|
20
+ | class | str |
21
+ | resource_id | str |
22
+ | action | str |
23
+ | data | json |
24
+ | created_at | created |
25
+
26
+ The names are not currently adjustable.
27
+
28
+ 1. Class is a string that records the name of the class that the action happened for. This allows you to use
29
+ the same audit class for multiple, different, resources.
30
+ 2. resource_id is the id of the record which the audit entry is for.
31
+ 3. Action is the actual action taken (create/update/delete)
32
+ 4. Data is a serialized record of what columns in the record were changed (both their previous and new values)
33
+ 5. The time the audit record was created
34
+ """
35
+
36
+ """ The model class for the destination that will store the audit data. """
37
+ audit_model_class = configs.ModelClass(required=True)
38
+
39
+ """
40
+ A list of columns that shouldn't be copied into the audit record.
41
+
42
+ To be clear, these are columns from the model class that the audit column is attached to.
43
+ If only excluded columns are updated then no audit record will be created.
44
+ """
45
+ exclude_columns = configs.ModelColumns()
46
+
47
+ """
48
+ A list of columns that should be masked when copied into the audit record.
49
+
50
+ With masked columns a generic value is placed in the audit record (e.g. XXXXX) which denotes that
51
+ the column was changed, but it does not record either old or new values.
52
+ """
53
+ mask_columns = configs.ModelColumns()
54
+
55
+ """ Columns from the child table that should be included when converting this column to JSON. """
56
+ readable_child_columns = configs.ReadableModelColumns(
57
+ "audit_model_class", default=["resource_id", "action", "data", "created_at"]
58
+ )
59
+
60
+ """
61
+ Since this column is always populated automatically, it is never directly writeable.
62
+ """
63
+ is_writeable = configs.Boolean(default=False)
64
+ is_searchable = configs.Boolean(default=False)
65
+ _descriptor_config_map = None
66
+ _parent_columns: dict[str, Column] | None
67
+
68
+ @parameters_to_properties.parameters_to_properties
69
+ def __init__(
70
+ self,
71
+ audit_model_class,
72
+ exclude_columns: list[str] = [],
73
+ mask_columns: list[str] = [],
74
+ foreign_column_name: str | None = None,
75
+ readable_child_columns: list[str] = [],
76
+ where: clearskies.typing.condition | list[clearskies.typing.condition] = [],
77
+ default: str | None = None,
78
+ is_readable: bool = True,
79
+ is_temporary: bool = False,
80
+ on_change_pre_save: clearskies.typing.action | list[clearskies.typing.action] = [],
81
+ on_change_post_save: clearskies.typing.action | list[clearskies.typing.action] = [],
82
+ on_change_save_finished: clearskies.typing.action | list[clearskies.typing.action] = [],
83
+ ):
84
+ self.child_model_class = self.audit_model_class
85
+
86
+ def save_finished(self, model):
87
+ super().save_finished(model)
88
+ old_data: dict[str, Any] = model._previous_data
89
+ new_data: dict[str, Any] = model.get_raw_data()
90
+ exclude_columns = self.exclude_columns
91
+ mask_columns = self.mask_columns
92
+ model_columns = self.get_model_columns()
93
+
94
+ if not old_data:
95
+ create_data: dict[str, Any] = {}
96
+ for key in new_data.keys():
97
+ if key in exclude_columns:
98
+ continue
99
+ if key in model_columns:
100
+ column_data = model_columns[key].to_json(model)
101
+ else:
102
+ column_data = {key: new_data[key]}
103
+
104
+ create_data = {
105
+ **create_data,
106
+ **column_data,
107
+ }
108
+ if key in mask_columns and key in create_data:
109
+ create_data[key] = "****"
110
+ self.record(model, "create", data=create_data)
111
+ return
112
+
113
+ # note that this is fairly simple logic to get started. It's not going to detect changes that happen
114
+ # in other "tables". For instance, disconnecting a record by deleting an entry in a many-to-many relationship
115
+ # won't be picked up by this.
116
+ old_model = model.empty_model()
117
+ old_model.data = old_data
118
+ from_data: dict[str, Any] = {}
119
+ to_data: dict[str, Any] = {}
120
+ for column, new_value in new_data.items():
121
+ if column in exclude_columns or column not in old_data:
122
+ continue
123
+ if old_data[column] == new_value:
124
+ continue
125
+ from_data = {
126
+ **from_data,
127
+ **(
128
+ model_columns[column].to_json(old_model)
129
+ if column in model_columns
130
+ else {column: old_data.get(column)}
131
+ ),
132
+ }
133
+ to_data = {
134
+ **to_data,
135
+ **(
136
+ model_columns[column].to_json(model)
137
+ if column in model_columns
138
+ else {column: model._data.get(column)}
139
+ ),
140
+ }
141
+ if column in mask_columns and column in to_data:
142
+ to_data[column] = "****"
143
+ from_data[column] = "****"
144
+ if not from_data and not to_data:
145
+ return
146
+
147
+ self.record(
148
+ model,
149
+ "update",
150
+ data={
151
+ "from": from_data,
152
+ "to": to_data,
153
+ },
154
+ )
155
+
156
+ def post_delete(self, model):
157
+ super().post_delete(model)
158
+ exclude_columns = self.exclude_columns
159
+ model_columns = self.get_model_columns()
160
+ mask_columns = self.mask_columns
161
+
162
+ final_data: dict[str, Any] = {}
163
+ for key in model._data.keys():
164
+ if key in exclude_columns:
165
+ continue
166
+ final_data = {
167
+ **final_data,
168
+ **(model_columns[key].to_json(model) if key in model_columns else {key: model.data.get(key)}),
169
+ }
170
+
171
+ for key in mask_columns:
172
+ if key not in final_data:
173
+ continue
174
+ final_data[key] = "****"
175
+
176
+ self.child_model.create(
177
+ {
178
+ "class": self.model_class.__name__,
179
+ "resource_id": model.get(self.model_class.id_column_name),
180
+ "action": "delete",
181
+ "data": final_data,
182
+ }
183
+ )
184
+
185
+ @property
186
+ def parent_columns(self) -> dict[str, Column]:
187
+ if self._parent_columns == None:
188
+ self._parent_columns = self.di.build(self.model_class, cache=True).columns()
189
+ return self._parent_columns
190
+
191
+ def record(self, model, action, data=None, record_data=None):
192
+ audit_data = {
193
+ "class": self.model_class.__name__,
194
+ "resource_id": model.get(self.model_class.id_column_name),
195
+ "action": action,
196
+ }
197
+ if data is not None:
198
+ audit_data["data"] = data
199
+ if record_data is not None:
200
+ audit_data = {
201
+ **audit_data,
202
+ **record_data,
203
+ }
204
+
205
+ self.child_model.create(audit_data)