clear-skies 1.22.31__py3-none-any.whl → 2.0.0__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 (344) hide show
  1. {clear_skies-1.22.31.dist-info → clear_skies-2.0.0.dist-info}/METADATA +11 -13
  2. clear_skies-2.0.0.dist-info/RECORD +248 -0
  3. {clear_skies-1.22.31.dist-info → clear_skies-2.0.0.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 +42 -0
  8. clearskies/authentication/authorization.py +4 -9
  9. clearskies/authentication/authorization_pass_through.py +11 -9
  10. clearskies/authentication/jwks.py +128 -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 +40 -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 +1232 -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 +159 -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 +21 -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/float.py +16 -0
  100. clearskies/configs/float_or_callable.py +18 -0
  101. clearskies/configs/integer.py +16 -0
  102. clearskies/configs/integer_or_callable.py +18 -0
  103. clearskies/configs/joins.py +30 -0
  104. clearskies/configs/list_any_dict.py +30 -0
  105. clearskies/configs/list_any_dict_or_callable.py +31 -0
  106. clearskies/configs/model_class.py +35 -0
  107. clearskies/configs/model_column.py +65 -0
  108. clearskies/configs/model_columns.py +56 -0
  109. clearskies/configs/model_destination_name.py +25 -0
  110. clearskies/configs/model_to_id_column.py +43 -0
  111. clearskies/configs/readable_model_column.py +9 -0
  112. clearskies/configs/readable_model_columns.py +9 -0
  113. clearskies/configs/schema.py +23 -0
  114. clearskies/configs/searchable_model_columns.py +9 -0
  115. clearskies/configs/security_headers.py +39 -0
  116. clearskies/configs/select.py +26 -0
  117. clearskies/configs/select_list.py +47 -0
  118. clearskies/configs/string.py +29 -0
  119. clearskies/configs/string_dict.py +32 -0
  120. clearskies/configs/string_list.py +32 -0
  121. clearskies/configs/string_list_or_callable.py +35 -0
  122. clearskies/configs/string_or_callable.py +18 -0
  123. clearskies/configs/timedelta.py +18 -0
  124. clearskies/configs/timezone.py +18 -0
  125. clearskies/configs/url.py +23 -0
  126. clearskies/configs/validators.py +45 -0
  127. clearskies/configs/writeable_model_column.py +9 -0
  128. clearskies/configs/writeable_model_columns.py +9 -0
  129. clearskies/configurable.py +76 -0
  130. clearskies/contexts/__init__.py +8 -8
  131. clearskies/contexts/cli.py +5 -42
  132. clearskies/contexts/context.py +78 -56
  133. clearskies/contexts/wsgi.py +13 -30
  134. clearskies/contexts/wsgi_ref.py +49 -0
  135. clearskies/di/__init__.py +10 -7
  136. clearskies/di/additional_config.py +115 -4
  137. clearskies/di/additional_config_auto_import.py +12 -0
  138. clearskies/di/di.py +742 -121
  139. clearskies/di/inject/__init__.py +23 -0
  140. clearskies/di/inject/by_class.py +21 -0
  141. clearskies/di/inject/by_name.py +18 -0
  142. clearskies/di/inject/di.py +13 -0
  143. clearskies/di/inject/environment.py +14 -0
  144. clearskies/di/inject/input_output.py +20 -0
  145. clearskies/di/inject/now.py +13 -0
  146. clearskies/di/inject/requests.py +13 -0
  147. clearskies/di/inject/secrets.py +14 -0
  148. clearskies/di/inject/utcnow.py +13 -0
  149. clearskies/di/inject/uuid.py +15 -0
  150. clearskies/di/injectable.py +29 -0
  151. clearskies/di/injectable_properties.py +131 -0
  152. clearskies/end.py +183 -0
  153. clearskies/endpoint.py +1309 -0
  154. clearskies/endpoint_group.py +297 -0
  155. clearskies/endpoints/__init__.py +23 -0
  156. clearskies/endpoints/advanced_search.py +526 -0
  157. clearskies/endpoints/callable.py +387 -0
  158. clearskies/endpoints/create.py +202 -0
  159. clearskies/endpoints/delete.py +139 -0
  160. clearskies/endpoints/get.py +275 -0
  161. clearskies/endpoints/health_check.py +181 -0
  162. clearskies/endpoints/list.py +573 -0
  163. clearskies/endpoints/restful_api.py +427 -0
  164. clearskies/endpoints/simple_search.py +286 -0
  165. clearskies/endpoints/update.py +190 -0
  166. clearskies/environment.py +5 -3
  167. clearskies/exceptions/__init__.py +17 -0
  168. clearskies/{handlers/exceptions/input_error.py → exceptions/input_errors.py} +1 -1
  169. clearskies/exceptions/moved_permanently.py +3 -0
  170. clearskies/exceptions/moved_temporarily.py +3 -0
  171. clearskies/exceptions/not_found.py +2 -0
  172. clearskies/functional/__init__.py +2 -2
  173. clearskies/functional/routing.py +92 -0
  174. clearskies/functional/string.py +19 -11
  175. clearskies/functional/validations.py +61 -9
  176. clearskies/input_outputs/__init__.py +9 -7
  177. clearskies/input_outputs/cli.py +130 -142
  178. clearskies/input_outputs/exceptions/__init__.py +1 -1
  179. clearskies/input_outputs/headers.py +45 -0
  180. clearskies/input_outputs/input_output.py +91 -122
  181. clearskies/input_outputs/programmatic.py +69 -0
  182. clearskies/input_outputs/wsgi.py +23 -38
  183. clearskies/model.py +489 -184
  184. clearskies/parameters_to_properties.py +31 -0
  185. clearskies/query/__init__.py +12 -0
  186. clearskies/query/condition.py +223 -0
  187. clearskies/query/join.py +136 -0
  188. clearskies/query/query.py +196 -0
  189. clearskies/query/sort.py +27 -0
  190. clearskies/schema.py +82 -0
  191. clearskies/secrets/__init__.py +3 -31
  192. clearskies/secrets/additional_configs/mysql_connection_dynamic_producer.py +15 -4
  193. clearskies/secrets/additional_configs/mysql_connection_dynamic_producer_via_ssh_cert_bastion.py +11 -5
  194. clearskies/secrets/akeyless.py +88 -147
  195. clearskies/secrets/secrets.py +8 -8
  196. clearskies/security_header.py +8 -0
  197. clearskies/security_headers/__init__.py +8 -8
  198. clearskies/security_headers/cache_control.py +47 -110
  199. clearskies/security_headers/cors.py +40 -95
  200. clearskies/security_headers/csp.py +76 -151
  201. clearskies/security_headers/hsts.py +14 -16
  202. clearskies/test_base.py +8 -0
  203. clearskies/typing.py +11 -0
  204. clearskies/validator.py +25 -0
  205. clearskies/validators/__init__.py +33 -0
  206. clearskies/validators/after_column.py +62 -0
  207. clearskies/validators/before_column.py +13 -0
  208. clearskies/validators/in_the_future.py +32 -0
  209. clearskies/validators/in_the_future_at_least.py +11 -0
  210. clearskies/validators/in_the_future_at_most.py +10 -0
  211. clearskies/validators/in_the_past.py +32 -0
  212. clearskies/validators/in_the_past_at_least.py +10 -0
  213. clearskies/validators/in_the_past_at_most.py +10 -0
  214. clearskies/validators/maximum_length.py +26 -0
  215. clearskies/validators/maximum_value.py +29 -0
  216. clearskies/validators/minimum_length.py +26 -0
  217. clearskies/validators/minimum_value.py +29 -0
  218. clearskies/validators/required.py +35 -0
  219. clearskies/validators/timedelta.py +59 -0
  220. clearskies/validators/unique.py +31 -0
  221. clear_skies-1.22.31.dist-info/RECORD +0 -214
  222. clearskies/application.py +0 -29
  223. clearskies/authentication/auth0_jwks.py +0 -118
  224. clearskies/authentication/auth_exception.py +0 -2
  225. clearskies/authentication/jwks_jwcrypto.py +0 -51
  226. clearskies/backends/api_get_only_backend.py +0 -48
  227. clearskies/backends/example_backend.py +0 -43
  228. clearskies/backends/file_backend.py +0 -48
  229. clearskies/backends/json_backend.py +0 -7
  230. clearskies/backends/restful_api_advanced_search_backend.py +0 -103
  231. clearskies/binding_config.py +0 -16
  232. clearskies/column_types/__init__.py +0 -203
  233. clearskies/column_types/audit.py +0 -249
  234. clearskies/column_types/belongs_to.py +0 -271
  235. clearskies/column_types/boolean.py +0 -60
  236. clearskies/column_types/category_tree.py +0 -304
  237. clearskies/column_types/column.py +0 -373
  238. clearskies/column_types/created.py +0 -26
  239. clearskies/column_types/created_by_authorization_data.py +0 -26
  240. clearskies/column_types/created_by_header.py +0 -24
  241. clearskies/column_types/created_by_ip.py +0 -17
  242. clearskies/column_types/created_by_routing_data.py +0 -25
  243. clearskies/column_types/created_by_user_agent.py +0 -17
  244. clearskies/column_types/created_micro.py +0 -26
  245. clearskies/column_types/datetime.py +0 -109
  246. clearskies/column_types/datetime_micro.py +0 -12
  247. clearskies/column_types/email.py +0 -18
  248. clearskies/column_types/float.py +0 -43
  249. clearskies/column_types/has_many.py +0 -179
  250. clearskies/column_types/has_one.py +0 -60
  251. clearskies/column_types/integer.py +0 -41
  252. clearskies/column_types/json.py +0 -25
  253. clearskies/column_types/many_to_many.py +0 -278
  254. clearskies/column_types/many_to_many_with_data.py +0 -162
  255. clearskies/column_types/phone.py +0 -48
  256. clearskies/column_types/select.py +0 -11
  257. clearskies/column_types/string.py +0 -24
  258. clearskies/column_types/timestamp.py +0 -73
  259. clearskies/column_types/updated.py +0 -26
  260. clearskies/column_types/updated_micro.py +0 -26
  261. clearskies/column_types/uuid.py +0 -25
  262. clearskies/columns.py +0 -123
  263. clearskies/condition_parser.py +0 -172
  264. clearskies/contexts/build_context.py +0 -54
  265. clearskies/contexts/convert_to_application.py +0 -190
  266. clearskies/contexts/extract_handler.py +0 -37
  267. clearskies/contexts/test.py +0 -94
  268. clearskies/decorators/__init__.py +0 -41
  269. clearskies/decorators/allow_non_json_bodies.py +0 -9
  270. clearskies/decorators/auth0_jwks.py +0 -22
  271. clearskies/decorators/authorization.py +0 -10
  272. clearskies/decorators/binding_classes.py +0 -9
  273. clearskies/decorators/binding_modules.py +0 -9
  274. clearskies/decorators/bindings.py +0 -9
  275. clearskies/decorators/create.py +0 -10
  276. clearskies/decorators/delete.py +0 -10
  277. clearskies/decorators/docs.py +0 -14
  278. clearskies/decorators/get.py +0 -10
  279. clearskies/decorators/jwks.py +0 -26
  280. clearskies/decorators/merge.py +0 -124
  281. clearskies/decorators/patch.py +0 -10
  282. clearskies/decorators/post.py +0 -10
  283. clearskies/decorators/public.py +0 -11
  284. clearskies/decorators/response_headers.py +0 -10
  285. clearskies/decorators/return_raw_response.py +0 -9
  286. clearskies/decorators/schema.py +0 -10
  287. clearskies/decorators/secret_bearer.py +0 -24
  288. clearskies/decorators/security_headers.py +0 -10
  289. clearskies/di/standard_dependencies.py +0 -151
  290. clearskies/handlers/__init__.py +0 -41
  291. clearskies/handlers/advanced_search.py +0 -271
  292. clearskies/handlers/base.py +0 -479
  293. clearskies/handlers/callable.py +0 -192
  294. clearskies/handlers/create.py +0 -35
  295. clearskies/handlers/crud_by_method.py +0 -18
  296. clearskies/handlers/database_connector.py +0 -32
  297. clearskies/handlers/delete.py +0 -61
  298. clearskies/handlers/exceptions/__init__.py +0 -5
  299. clearskies/handlers/exceptions/not_found.py +0 -3
  300. clearskies/handlers/get.py +0 -156
  301. clearskies/handlers/health_check.py +0 -59
  302. clearskies/handlers/input_processing.py +0 -79
  303. clearskies/handlers/list.py +0 -530
  304. clearskies/handlers/mygrations.py +0 -82
  305. clearskies/handlers/request_method_routing.py +0 -47
  306. clearskies/handlers/restful_api.py +0 -218
  307. clearskies/handlers/routing.py +0 -62
  308. clearskies/handlers/schema_helper.py +0 -128
  309. clearskies/handlers/simple_routing.py +0 -206
  310. clearskies/handlers/simple_routing_route.py +0 -197
  311. clearskies/handlers/simple_search.py +0 -136
  312. clearskies/handlers/update.py +0 -102
  313. clearskies/handlers/write.py +0 -193
  314. clearskies/input_requirements/__init__.py +0 -78
  315. clearskies/input_requirements/after.py +0 -36
  316. clearskies/input_requirements/before.py +0 -36
  317. clearskies/input_requirements/in_the_future_at_least.py +0 -19
  318. clearskies/input_requirements/in_the_future_at_most.py +0 -19
  319. clearskies/input_requirements/in_the_past_at_least.py +0 -19
  320. clearskies/input_requirements/in_the_past_at_most.py +0 -19
  321. clearskies/input_requirements/maximum_length.py +0 -19
  322. clearskies/input_requirements/maximum_value.py +0 -19
  323. clearskies/input_requirements/minimum_length.py +0 -22
  324. clearskies/input_requirements/minimum_value.py +0 -19
  325. clearskies/input_requirements/required.py +0 -23
  326. clearskies/input_requirements/requirement.py +0 -25
  327. clearskies/input_requirements/time_delta.py +0 -38
  328. clearskies/input_requirements/unique.py +0 -18
  329. clearskies/mocks/__init__.py +0 -7
  330. clearskies/mocks/input_output.py +0 -124
  331. clearskies/mocks/models.py +0 -142
  332. clearskies/models.py +0 -350
  333. clearskies/security_headers/base.py +0 -12
  334. clearskies/tests/simple_api/models/__init__.py +0 -2
  335. clearskies/tests/simple_api/models/status.py +0 -23
  336. clearskies/tests/simple_api/models/user.py +0 -21
  337. clearskies/tests/simple_api/users_api.py +0 -64
  338. {clear_skies-1.22.31.dist-info → clear_skies-2.0.0.dist-info}/LICENSE +0 -0
  339. /clearskies/{contexts/bash.py → autodoc/py.typed} +0 -0
  340. /clearskies/{handlers/exceptions → exceptions}/authentication.py +0 -0
  341. /clearskies/{handlers/exceptions → exceptions}/authorization.py +0 -0
  342. /clearskies/{handlers/exceptions → exceptions}/client_error.py +0 -0
  343. /clearskies/{tests/__init__.py → input_outputs/py.typed} +0 -0
  344. /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)