infrahub-server 1.2.0rc0__py3-none-any.whl → 1.2.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.
Files changed (365) hide show
  1. infrahub/api/dependencies.py +6 -6
  2. infrahub/api/diff/validation_models.py +7 -7
  3. infrahub/api/schema.py +1 -1
  4. infrahub/artifacts/models.py +5 -3
  5. infrahub/artifacts/tasks.py +3 -5
  6. infrahub/cli/__init__.py +13 -9
  7. infrahub/cli/constants.py +3 -0
  8. infrahub/cli/db.py +165 -183
  9. infrahub/cli/upgrade.py +146 -0
  10. infrahub/computed_attribute/gather.py +185 -0
  11. infrahub/computed_attribute/models.py +240 -12
  12. infrahub/computed_attribute/tasks.py +77 -441
  13. infrahub/computed_attribute/triggers.py +13 -47
  14. infrahub/config.py +43 -32
  15. infrahub/context.py +14 -0
  16. infrahub/core/account.py +4 -4
  17. infrahub/core/attribute.py +58 -58
  18. infrahub/core/branch/tasks.py +74 -22
  19. infrahub/core/changelog/diff.py +95 -36
  20. infrahub/core/changelog/models.py +217 -43
  21. infrahub/core/constants/__init__.py +28 -0
  22. infrahub/core/constants/infrahubkind.py +2 -0
  23. infrahub/core/constants/schema.py +2 -0
  24. infrahub/core/constraint/node/runner.py +9 -8
  25. infrahub/core/diff/branch_differ.py +10 -10
  26. infrahub/core/diff/enricher/cardinality_one.py +5 -0
  27. infrahub/core/diff/enricher/hierarchy.py +17 -4
  28. infrahub/core/diff/enricher/labels.py +5 -0
  29. infrahub/core/diff/enricher/path_identifier.py +4 -0
  30. infrahub/core/diff/ipam_diff_parser.py +4 -5
  31. infrahub/core/diff/model/diff.py +27 -27
  32. infrahub/core/diff/model/path.py +32 -9
  33. infrahub/core/diff/parent_node_adder.py +78 -0
  34. infrahub/core/diff/payload_builder.py +13 -2
  35. infrahub/core/diff/query/filters.py +2 -2
  36. infrahub/core/diff/query/merge.py +20 -17
  37. infrahub/core/diff/query/save.py +188 -182
  38. infrahub/core/diff/query/summary_counts_enricher.py +51 -4
  39. infrahub/core/diff/query_parser.py +4 -4
  40. infrahub/core/diff/repository/deserializer.py +8 -3
  41. infrahub/core/diff/repository/repository.py +156 -38
  42. infrahub/core/diff/tasks.py +4 -4
  43. infrahub/core/graph/__init__.py +1 -1
  44. infrahub/core/graph/index.py +3 -0
  45. infrahub/core/initialization.py +1 -10
  46. infrahub/core/ipam/constants.py +3 -4
  47. infrahub/core/ipam/reconciler.py +12 -12
  48. infrahub/core/ipam/utilization.py +10 -13
  49. infrahub/core/manager.py +36 -36
  50. infrahub/core/merge.py +7 -7
  51. infrahub/core/migrations/__init__.py +2 -3
  52. infrahub/core/migrations/graph/__init__.py +12 -3
  53. infrahub/core/migrations/graph/m017_add_core_profile.py +1 -5
  54. infrahub/core/migrations/graph/m018_uniqueness_nulls.py +4 -4
  55. infrahub/core/migrations/graph/m019_restore_rels_to_time.py +256 -0
  56. infrahub/core/migrations/graph/m020_duplicate_edges.py +160 -0
  57. infrahub/core/migrations/graph/m021_missing_hierarchy_merge.py +51 -0
  58. infrahub/core/migrations/graph/m022_add_generate_template_attr.py +48 -0
  59. infrahub/core/migrations/graph/m023_deduplicate_cardinality_one_relationships.py +96 -0
  60. infrahub/core/migrations/query/attribute_add.py +2 -2
  61. infrahub/core/migrations/query/node_duplicate.py +43 -26
  62. infrahub/core/migrations/query/schema_attribute_update.py +2 -2
  63. infrahub/core/migrations/schema/models.py +19 -4
  64. infrahub/core/migrations/schema/node_remove.py +26 -12
  65. infrahub/core/migrations/schema/tasks.py +2 -2
  66. infrahub/core/migrations/shared.py +16 -16
  67. infrahub/core/models.py +15 -6
  68. infrahub/core/node/__init__.py +43 -39
  69. infrahub/core/node/base.py +2 -4
  70. infrahub/core/node/constraints/attribute_uniqueness.py +2 -2
  71. infrahub/core/node/constraints/grouped_uniqueness.py +99 -47
  72. infrahub/core/node/constraints/interface.py +1 -2
  73. infrahub/core/node/delete_validator.py +3 -5
  74. infrahub/core/node/ipam.py +4 -4
  75. infrahub/core/node/permissions.py +7 -7
  76. infrahub/core/node/resource_manager/ip_address_pool.py +6 -6
  77. infrahub/core/node/resource_manager/ip_prefix_pool.py +6 -6
  78. infrahub/core/node/resource_manager/number_pool.py +3 -3
  79. infrahub/core/path.py +12 -12
  80. infrahub/core/property.py +11 -11
  81. infrahub/core/protocols.py +7 -0
  82. infrahub/core/protocols_base.py +21 -21
  83. infrahub/core/query/__init__.py +33 -33
  84. infrahub/core/query/attribute.py +6 -4
  85. infrahub/core/query/diff.py +3 -3
  86. infrahub/core/query/node.py +82 -32
  87. infrahub/core/query/relationship.py +228 -40
  88. infrahub/core/query/resource_manager.py +2 -0
  89. infrahub/core/query/standard_node.py +3 -3
  90. infrahub/core/query/subquery.py +9 -9
  91. infrahub/core/registry.py +13 -15
  92. infrahub/core/relationship/constraints/count.py +3 -4
  93. infrahub/core/relationship/constraints/peer_kind.py +3 -4
  94. infrahub/core/relationship/constraints/profiles_kind.py +2 -2
  95. infrahub/core/relationship/model.py +51 -59
  96. infrahub/core/schema/attribute_schema.py +16 -8
  97. infrahub/core/schema/basenode_schema.py +105 -44
  98. infrahub/core/schema/computed_attribute.py +3 -3
  99. infrahub/core/schema/definitions/core/__init__.py +147 -0
  100. infrahub/core/schema/definitions/core/account.py +171 -0
  101. infrahub/core/schema/definitions/core/artifact.py +136 -0
  102. infrahub/core/schema/definitions/core/builtin.py +24 -0
  103. infrahub/core/schema/definitions/core/check.py +68 -0
  104. infrahub/core/schema/definitions/core/core.py +17 -0
  105. infrahub/core/schema/definitions/core/generator.py +100 -0
  106. infrahub/core/schema/definitions/core/graphql_query.py +79 -0
  107. infrahub/core/schema/definitions/core/group.py +108 -0
  108. infrahub/core/schema/definitions/core/ipam.py +193 -0
  109. infrahub/core/schema/definitions/core/lineage.py +19 -0
  110. infrahub/core/schema/definitions/core/menu.py +48 -0
  111. infrahub/core/schema/definitions/core/permission.py +163 -0
  112. infrahub/core/schema/definitions/core/profile.py +18 -0
  113. infrahub/core/schema/definitions/core/propose_change.py +97 -0
  114. infrahub/core/schema/definitions/core/propose_change_comment.py +193 -0
  115. infrahub/core/schema/definitions/core/propose_change_validator.py +328 -0
  116. infrahub/core/schema/definitions/core/repository.py +286 -0
  117. infrahub/core/schema/definitions/core/resource_pool.py +170 -0
  118. infrahub/core/schema/definitions/core/template.py +27 -0
  119. infrahub/core/schema/definitions/core/transform.py +96 -0
  120. infrahub/core/schema/definitions/core/webhook.py +134 -0
  121. infrahub/core/schema/definitions/internal.py +16 -16
  122. infrahub/core/schema/dropdown.py +3 -4
  123. infrahub/core/schema/generated/attribute_schema.py +15 -18
  124. infrahub/core/schema/generated/base_node_schema.py +12 -14
  125. infrahub/core/schema/generated/node_schema.py +3 -5
  126. infrahub/core/schema/generated/relationship_schema.py +9 -11
  127. infrahub/core/schema/generic_schema.py +2 -2
  128. infrahub/core/schema/manager.py +20 -9
  129. infrahub/core/schema/node_schema.py +4 -2
  130. infrahub/core/schema/relationship_schema.py +14 -6
  131. infrahub/core/schema/schema_branch.py +292 -144
  132. infrahub/core/schema/schema_branch_computed.py +41 -4
  133. infrahub/core/task/task.py +3 -3
  134. infrahub/core/task/user_task.py +15 -15
  135. infrahub/core/timestamp.py +3 -3
  136. infrahub/core/utils.py +20 -18
  137. infrahub/core/validators/__init__.py +1 -3
  138. infrahub/core/validators/aggregated_checker.py +2 -2
  139. infrahub/core/validators/attribute/choices.py +2 -2
  140. infrahub/core/validators/attribute/enum.py +2 -2
  141. infrahub/core/validators/attribute/kind.py +2 -2
  142. infrahub/core/validators/attribute/length.py +2 -2
  143. infrahub/core/validators/attribute/optional.py +2 -2
  144. infrahub/core/validators/attribute/regex.py +2 -2
  145. infrahub/core/validators/attribute/unique.py +2 -2
  146. infrahub/core/validators/checks_runner.py +25 -2
  147. infrahub/core/validators/determiner.py +1 -3
  148. infrahub/core/validators/interface.py +6 -2
  149. infrahub/core/validators/model.py +22 -3
  150. infrahub/core/validators/models/validate_migration.py +17 -4
  151. infrahub/core/validators/node/attribute.py +2 -2
  152. infrahub/core/validators/node/generate_profile.py +2 -2
  153. infrahub/core/validators/node/hierarchy.py +3 -5
  154. infrahub/core/validators/node/inherit_from.py +27 -5
  155. infrahub/core/validators/node/relationship.py +2 -2
  156. infrahub/core/validators/relationship/count.py +4 -4
  157. infrahub/core/validators/relationship/optional.py +2 -2
  158. infrahub/core/validators/relationship/peer.py +2 -2
  159. infrahub/core/validators/shared.py +2 -2
  160. infrahub/core/validators/tasks.py +8 -0
  161. infrahub/core/validators/uniqueness/checker.py +22 -21
  162. infrahub/core/validators/uniqueness/index.py +2 -2
  163. infrahub/core/validators/uniqueness/model.py +11 -11
  164. infrahub/database/__init__.py +27 -22
  165. infrahub/database/metrics.py +7 -1
  166. infrahub/dependencies/builder/constraint/grouped/node_runner.py +1 -3
  167. infrahub/dependencies/builder/diff/deserializer.py +3 -1
  168. infrahub/dependencies/builder/diff/enricher/hierarchy.py +3 -1
  169. infrahub/dependencies/builder/diff/parent_node_adder.py +8 -0
  170. infrahub/dependencies/component/registry.py +2 -2
  171. infrahub/events/__init__.py +25 -2
  172. infrahub/events/artifact_action.py +64 -0
  173. infrahub/events/branch_action.py +33 -22
  174. infrahub/events/generator.py +71 -0
  175. infrahub/events/group_action.py +51 -21
  176. infrahub/events/models.py +18 -19
  177. infrahub/events/node_action.py +88 -37
  178. infrahub/events/repository_action.py +5 -18
  179. infrahub/events/schema_action.py +4 -9
  180. infrahub/events/utils.py +16 -0
  181. infrahub/events/validator_action.py +55 -0
  182. infrahub/exceptions.py +32 -24
  183. infrahub/generators/models.py +2 -3
  184. infrahub/generators/tasks.py +24 -4
  185. infrahub/git/base.py +7 -7
  186. infrahub/git/integrator.py +48 -24
  187. infrahub/git/models.py +101 -9
  188. infrahub/git/repository.py +3 -3
  189. infrahub/git/tasks.py +408 -6
  190. infrahub/git/utils.py +48 -0
  191. infrahub/git/worktree.py +1 -2
  192. infrahub/git_credential/askpass.py +1 -2
  193. infrahub/graphql/analyzer.py +12 -0
  194. infrahub/graphql/app.py +13 -15
  195. infrahub/graphql/context.py +39 -0
  196. infrahub/graphql/initialization.py +3 -0
  197. infrahub/graphql/loaders/node.py +2 -12
  198. infrahub/graphql/loaders/peers.py +77 -0
  199. infrahub/graphql/loaders/shared.py +13 -0
  200. infrahub/graphql/manager.py +17 -19
  201. infrahub/graphql/mutations/artifact_definition.py +5 -5
  202. infrahub/graphql/mutations/branch.py +26 -1
  203. infrahub/graphql/mutations/computed_attribute.py +9 -5
  204. infrahub/graphql/mutations/diff.py +23 -11
  205. infrahub/graphql/mutations/diff_conflict.py +5 -0
  206. infrahub/graphql/mutations/generator.py +83 -0
  207. infrahub/graphql/mutations/graphql_query.py +5 -5
  208. infrahub/graphql/mutations/ipam.py +54 -74
  209. infrahub/graphql/mutations/main.py +195 -132
  210. infrahub/graphql/mutations/menu.py +7 -7
  211. infrahub/graphql/mutations/models.py +2 -4
  212. infrahub/graphql/mutations/node_getter/by_default_filter.py +10 -10
  213. infrahub/graphql/mutations/node_getter/by_hfid.py +1 -3
  214. infrahub/graphql/mutations/node_getter/by_id.py +1 -3
  215. infrahub/graphql/mutations/node_getter/interface.py +1 -2
  216. infrahub/graphql/mutations/proposed_change.py +7 -7
  217. infrahub/graphql/mutations/relationship.py +93 -19
  218. infrahub/graphql/mutations/repository.py +8 -8
  219. infrahub/graphql/mutations/resource_manager.py +3 -3
  220. infrahub/graphql/mutations/schema.py +19 -4
  221. infrahub/graphql/mutations/webhook.py +137 -0
  222. infrahub/graphql/parser.py +4 -4
  223. infrahub/graphql/permissions.py +1 -10
  224. infrahub/graphql/queries/diff/tree.py +19 -14
  225. infrahub/graphql/queries/event.py +5 -2
  226. infrahub/graphql/queries/ipam.py +2 -2
  227. infrahub/graphql/queries/relationship.py +2 -2
  228. infrahub/graphql/queries/search.py +2 -2
  229. infrahub/graphql/resolvers/many_relationship.py +264 -0
  230. infrahub/graphql/resolvers/resolver.py +13 -110
  231. infrahub/graphql/schema.py +2 -0
  232. infrahub/graphql/subscription/graphql_query.py +2 -0
  233. infrahub/graphql/types/context.py +12 -0
  234. infrahub/graphql/types/event.py +84 -17
  235. infrahub/graphql/types/node.py +2 -2
  236. infrahub/graphql/utils.py +2 -2
  237. infrahub/groups/ancestors.py +29 -0
  238. infrahub/groups/parsers.py +107 -0
  239. infrahub/lock.py +20 -20
  240. infrahub/menu/constants.py +0 -1
  241. infrahub/menu/generator.py +9 -21
  242. infrahub/menu/menu.py +17 -38
  243. infrahub/menu/models.py +117 -16
  244. infrahub/menu/repository.py +111 -0
  245. infrahub/menu/utils.py +5 -8
  246. infrahub/message_bus/__init__.py +11 -13
  247. infrahub/message_bus/messages/__init__.py +1 -21
  248. infrahub/message_bus/messages/check_generator_run.py +3 -3
  249. infrahub/message_bus/messages/finalize_validator_execution.py +3 -0
  250. infrahub/message_bus/messages/proposed_change/request_proposedchange_refreshartifacts.py +6 -0
  251. infrahub/message_bus/messages/request_generatordefinition_check.py +2 -0
  252. infrahub/message_bus/messages/send_echo_request.py +1 -1
  253. infrahub/message_bus/operations/__init__.py +1 -10
  254. infrahub/message_bus/operations/check/__init__.py +2 -2
  255. infrahub/message_bus/operations/check/generator.py +1 -0
  256. infrahub/message_bus/operations/event/__init__.py +2 -2
  257. infrahub/message_bus/operations/event/worker.py +0 -3
  258. infrahub/message_bus/operations/finalize/validator.py +51 -1
  259. infrahub/message_bus/operations/requests/__init__.py +0 -2
  260. infrahub/message_bus/operations/requests/generator_definition.py +21 -23
  261. infrahub/message_bus/operations/requests/proposed_change.py +14 -10
  262. infrahub/permissions/globals.py +15 -0
  263. infrahub/pools/number.py +2 -4
  264. infrahub/proposed_change/models.py +3 -0
  265. infrahub/proposed_change/tasks.py +58 -45
  266. infrahub/pytest_plugin.py +13 -10
  267. infrahub/server.py +2 -3
  268. infrahub/services/__init__.py +2 -2
  269. infrahub/services/adapters/cache/__init__.py +4 -6
  270. infrahub/services/adapters/cache/nats.py +4 -5
  271. infrahub/services/adapters/cache/redis.py +3 -7
  272. infrahub/services/adapters/event/__init__.py +1 -1
  273. infrahub/services/adapters/message_bus/__init__.py +3 -3
  274. infrahub/services/adapters/message_bus/local.py +2 -2
  275. infrahub/services/adapters/message_bus/nats.py +4 -4
  276. infrahub/services/adapters/message_bus/rabbitmq.py +4 -4
  277. infrahub/services/adapters/workflow/local.py +2 -2
  278. infrahub/services/component.py +5 -5
  279. infrahub/services/protocols.py +7 -7
  280. infrahub/services/scheduler.py +1 -3
  281. infrahub/task_manager/event.py +102 -9
  282. infrahub/task_manager/models.py +27 -7
  283. infrahub/tasks/artifact.py +7 -6
  284. infrahub/telemetry/__init__.py +0 -0
  285. infrahub/telemetry/constants.py +9 -0
  286. infrahub/telemetry/database.py +86 -0
  287. infrahub/telemetry/models.py +65 -0
  288. infrahub/telemetry/task_manager.py +77 -0
  289. infrahub/{tasks/telemetry.py → telemetry/tasks.py} +49 -56
  290. infrahub/telemetry/utils.py +11 -0
  291. infrahub/trace.py +4 -4
  292. infrahub/transformations/tasks.py +2 -2
  293. infrahub/trigger/catalogue.py +4 -6
  294. infrahub/trigger/constants.py +0 -8
  295. infrahub/trigger/models.py +54 -5
  296. infrahub/trigger/setup.py +90 -0
  297. infrahub/trigger/tasks.py +35 -84
  298. infrahub/utils.py +11 -1
  299. infrahub/validators/__init__.py +0 -0
  300. infrahub/validators/events.py +42 -0
  301. infrahub/validators/tasks.py +41 -0
  302. infrahub/webhook/gather.py +17 -0
  303. infrahub/webhook/models.py +176 -44
  304. infrahub/webhook/tasks.py +154 -155
  305. infrahub/webhook/triggers.py +31 -7
  306. infrahub/workers/infrahub_async.py +2 -2
  307. infrahub/workers/utils.py +2 -2
  308. infrahub/workflows/catalogue.py +86 -35
  309. infrahub/workflows/initialization.py +8 -2
  310. infrahub/workflows/models.py +27 -1
  311. infrahub/workflows/utils.py +10 -1
  312. infrahub_sdk/client.py +35 -8
  313. infrahub_sdk/config.py +3 -0
  314. infrahub_sdk/context.py +13 -0
  315. infrahub_sdk/ctl/branch.py +3 -2
  316. infrahub_sdk/ctl/cli_commands.py +5 -1
  317. infrahub_sdk/ctl/utils.py +0 -16
  318. infrahub_sdk/exceptions.py +12 -0
  319. infrahub_sdk/generator.py +4 -1
  320. infrahub_sdk/graphql.py +45 -13
  321. infrahub_sdk/node.py +71 -22
  322. infrahub_sdk/protocols.py +21 -8
  323. infrahub_sdk/protocols_base.py +32 -11
  324. infrahub_sdk/query_groups.py +6 -35
  325. infrahub_sdk/schema/__init__.py +55 -26
  326. infrahub_sdk/schema/main.py +8 -0
  327. infrahub_sdk/task/__init__.py +11 -0
  328. infrahub_sdk/task/constants.py +3 -0
  329. infrahub_sdk/task/exceptions.py +25 -0
  330. infrahub_sdk/task/manager.py +551 -0
  331. infrahub_sdk/task/models.py +74 -0
  332. infrahub_sdk/testing/schemas/animal.py +9 -0
  333. infrahub_sdk/timestamp.py +142 -33
  334. infrahub_sdk/utils.py +29 -1
  335. {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/METADATA +8 -6
  336. {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/RECORD +349 -293
  337. {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/entry_points.txt +1 -0
  338. infrahub_testcontainers/constants.py +2 -0
  339. infrahub_testcontainers/container.py +157 -12
  340. infrahub_testcontainers/docker-compose.test.yml +31 -6
  341. infrahub_testcontainers/helpers.py +18 -73
  342. infrahub_testcontainers/host.py +41 -0
  343. infrahub_testcontainers/measurements.py +93 -0
  344. infrahub_testcontainers/models.py +38 -0
  345. infrahub_testcontainers/performance_test.py +166 -0
  346. infrahub_testcontainers/plugin.py +136 -0
  347. infrahub_testcontainers/prometheus.yml +30 -0
  348. infrahub/core/schema/definitions/core.py +0 -2286
  349. infrahub/message_bus/messages/check_repository_checkdefinition.py +0 -20
  350. infrahub/message_bus/messages/check_repository_mergeconflicts.py +0 -16
  351. infrahub/message_bus/messages/check_repository_usercheck.py +0 -26
  352. infrahub/message_bus/messages/event_branch_create.py +0 -11
  353. infrahub/message_bus/messages/event_branch_delete.py +0 -11
  354. infrahub/message_bus/messages/event_branch_rebased.py +0 -9
  355. infrahub/message_bus/messages/event_node_mutated.py +0 -15
  356. infrahub/message_bus/messages/event_schema_update.py +0 -9
  357. infrahub/message_bus/messages/request_repository_checks.py +0 -12
  358. infrahub/message_bus/messages/request_repository_userchecks.py +0 -18
  359. infrahub/message_bus/operations/check/repository.py +0 -293
  360. infrahub/message_bus/operations/event/node.py +0 -20
  361. infrahub/message_bus/operations/event/schema.py +0 -17
  362. infrahub/message_bus/operations/requests/repository.py +0 -133
  363. infrahub/webhook/constants.py +0 -1
  364. {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/LICENSE.txt +0 -0
  365. {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/WHEEL +0 -0
@@ -0,0 +1,166 @@
1
+ import hashlib
2
+ import json
3
+ from datetime import UTC, datetime
4
+ from types import TracebackType
5
+ from typing import Any
6
+
7
+ import httpx
8
+ import pytest
9
+ from typing_extensions import Self
10
+
11
+ from .constants import PERFORMANCE_TEST_KIND, PERFORMANCE_TEST_VERSION
12
+ from .helpers import InfrahubDockerCompose
13
+ from .host import get_system_stats
14
+ from .models import (
15
+ ContextUnit,
16
+ InfrahubActiveMeasurementItem,
17
+ InfrahubMeasurementItem,
18
+ InfrahubResultContext,
19
+ MeasurementDefinition,
20
+ )
21
+
22
+
23
+ class InfrahubPerformanceTest:
24
+ context: dict[str, InfrahubResultContext]
25
+ measurements: list[InfrahubMeasurementItem]
26
+ active_measurements: InfrahubActiveMeasurementItem | None = None
27
+
28
+ def __init__(self, results_url: str) -> None:
29
+ self.name: str | None = None
30
+ self.infrahub_version: str | None = None
31
+ self.context = {}
32
+ self.measurements = []
33
+ self.metrics = {}
34
+ self.host = get_system_stats()
35
+ self.env_vars = {}
36
+ self.project_name = ""
37
+ self.test_info = {}
38
+ self.start_time = datetime.now(UTC)
39
+ self.end_time: datetime | None = None
40
+ self.results_url = results_url
41
+ self.scraper_endpoint = ""
42
+ self.initialized = False
43
+
44
+ def initialize(
45
+ self,
46
+ name: str,
47
+ compose: InfrahubDockerCompose | None = None,
48
+ client: Any | None = None, # noqa: ANN401
49
+ ) -> None:
50
+ self.name = name
51
+ if client:
52
+ self.infrahub_version = client.get_version()
53
+ if compose:
54
+ self.extract_compose_information(compose)
55
+
56
+ self.initialized = True
57
+
58
+ def finalize(self, session: pytest.Session) -> None:
59
+ if self.initialized:
60
+ self.end_time = datetime.now(UTC)
61
+ self.extract_test_session_information(session)
62
+ self.send_results()
63
+
64
+ def extract_compose_information(self, compose: InfrahubDockerCompose) -> None:
65
+ self.env_vars = compose.env_vars
66
+ self.project_name = compose.project_name
67
+ self.scraper_endpoint = (
68
+ f"http://127.0.0.1:{compose.get_service_host_and_port(service_name='scraper')[1]}/api/v1/export"
69
+ )
70
+
71
+ def extract_test_session_information(self, session: pytest.Session) -> None:
72
+ self.test_info = {
73
+ "summary": {
74
+ "exitstatus": session.exitstatus,
75
+ "nbr_tests": session.testscollected,
76
+ "nbr_errors": session.testsfailed,
77
+ }
78
+ }
79
+
80
+ def add_context(self, name: str, value: float | str = 0, unit: ContextUnit = ContextUnit.COUNT) -> None:
81
+ self.context[name] = InfrahubResultContext(name=name, value=value, unit=unit)
82
+
83
+ def add_measurement(
84
+ self,
85
+ definition: MeasurementDefinition,
86
+ value: float | str,
87
+ **kwargs: str | float,
88
+ ) -> None:
89
+ self.measurements.append(
90
+ InfrahubMeasurementItem(
91
+ name=definition.name,
92
+ value=value,
93
+ unit=definition.unit,
94
+ context=kwargs or {},
95
+ )
96
+ )
97
+
98
+ def start_measurement(self, definition: MeasurementDefinition, **kwargs: str | float) -> Self:
99
+ self.active_measurements = InfrahubActiveMeasurementItem(definition=definition, context=kwargs or {})
100
+ return self
101
+
102
+ def fetch_metrics(self) -> None:
103
+ with httpx.Client() as client:
104
+ # Get Infrahub metrics
105
+ response = client.post(
106
+ url=self.scraper_endpoint,
107
+ content='match[]={__name__=~"infrahub.*"}',
108
+ headers={"Content-Type": "application/x-www-form-urlencoded"},
109
+ )
110
+ self.metrics = [json.loads(line) for line in response.text.splitlines()]
111
+
112
+ # Get system metrics, filter by docker project name
113
+ response = client.post(
114
+ url=self.scraper_endpoint,
115
+ content=f'match[]={{__name__=~"container.*", container_label_com_docker_compose_project="{self.project_name}"}}',
116
+ headers={"Content-Type": "application/x-www-form-urlencoded"},
117
+ )
118
+ self.metrics += [json.loads(line) for line in response.text.splitlines()]
119
+
120
+ def __enter__(self) -> Self:
121
+ return self
122
+
123
+ def __exit__(
124
+ self,
125
+ exc_type: type[BaseException] | None,
126
+ exc_value: BaseException | None,
127
+ traceback: TracebackType | None,
128
+ ) -> None:
129
+ if not exc_type and self.active_measurements:
130
+ self.add_measurement(
131
+ definition=self.active_measurements.definition,
132
+ value=(datetime.now(UTC) - self.active_measurements.start_time).total_seconds() * 1000,
133
+ context=self.active_measurements.context,
134
+ )
135
+
136
+ # Don't record the measurement if the test failed
137
+ self.active_measurements = None
138
+
139
+ def _get_payload(self) -> dict[str, Any]:
140
+ return {
141
+ "name": self.name,
142
+ "infrahub_version": self.infrahub_version,
143
+ "start_time": self.start_time.isoformat(),
144
+ "end_time": self.end_time.isoformat() if self.end_time else None,
145
+ "duration": (self.end_time - self.start_time).total_seconds() * 1000 if self.end_time else None,
146
+ "context": [item.model_dump() for item in self.context.values()],
147
+ "measurements": [item.model_dump() for item in self.measurements],
148
+ "metrics": self.metrics,
149
+ "host": self.host,
150
+ "env_vars": self.env_vars,
151
+ "test_info": self.test_info,
152
+ }
153
+
154
+ def send_results(self) -> None:
155
+ data = self._get_payload()
156
+
157
+ payload = {
158
+ "kind": PERFORMANCE_TEST_KIND,
159
+ "payload_format": PERFORMANCE_TEST_VERSION,
160
+ "data": data,
161
+ "checksum": hashlib.sha256(json.dumps(data).encode()).hexdigest(),
162
+ }
163
+
164
+ with httpx.Client() as client:
165
+ response = client.post(self.results_url, json=payload)
166
+ response.raise_for_status()
@@ -0,0 +1,136 @@
1
+ from __future__ import annotations
2
+
3
+ from typing import TYPE_CHECKING
4
+
5
+ import pytest
6
+
7
+ from .performance_test import InfrahubPerformanceTest
8
+
9
+ if TYPE_CHECKING:
10
+ import _pytest
11
+
12
+
13
+ def pytest_addoption(parser: pytest.Parser) -> None:
14
+ group = parser.getgroup("infrahub-performance-test")
15
+
16
+ group.addoption(
17
+ "--performance-result-address",
18
+ action="store",
19
+ dest="infrahub_performance_test_result_address",
20
+ default="https://webhook.site/25802839-4b34-433e-9dc4-59623cd73c80",
21
+ metavar="INFRAHUB_PERFORMANCE_TEST_RESULT_ADDRESS",
22
+ help="Address to send the results of the performance test (default: %(default)s)",
23
+ )
24
+ group.addoption(
25
+ "--performance-backup",
26
+ action="store",
27
+ dest="infrahub_performance_backup",
28
+ default="neo4j_database.db",
29
+ metavar="INFRAHUB_PERFORMANCE_BACKUP",
30
+ help="Name of the backup to use (default: %(default)s)",
31
+ )
32
+ group.addoption(
33
+ "--performance-backup-location",
34
+ action="store",
35
+ dest="infrahub_performance_backup_location",
36
+ metavar="INFRAHUB_PERFORMANCE_BACKUP_LOCATION",
37
+ help="Location of the backup to use (default: %(default)s)",
38
+ )
39
+ group.addoption(
40
+ "--performance-create-backup",
41
+ action="store_true",
42
+ default=False,
43
+ dest="infrahub_performance_create_backup",
44
+ help="Generate a backup of the database (default: %(default)s)",
45
+ )
46
+ group.addoption(
47
+ "--performance-use-backup",
48
+ action="store_true",
49
+ default=False,
50
+ dest="infrahub_performance_use_backup",
51
+ help="Use a backup of the database and skip all tests that creates data (default: %(default)s)",
52
+ )
53
+ group.addoption(
54
+ "--performance-report",
55
+ action="store_true",
56
+ dest="infrahub_performance_report",
57
+ default=False,
58
+ help="Display performance report at the end of the test run",
59
+ )
60
+
61
+
62
+ def pytest_collection_modifyitems(config: pytest.Config, items: list[pytest.Item]) -> None:
63
+ use_backup_skip = pytest.mark.skip(reason="load data from backup")
64
+ no_create_backup_skip = pytest.mark.skip(reason="no need to create a backup")
65
+ no_load_backup_skip = pytest.mark.skip(reason="no need to load a backup")
66
+
67
+ use_backup = config.getoption("infrahub_performance_use_backup")
68
+ create_backup = config.getoption("infrahub_performance_create_backup")
69
+
70
+ for item in items:
71
+ has_create_backup_marker = bool(
72
+ [True for marker in item.own_markers if marker.name == "performance_create_backup"]
73
+ )
74
+ has_load_backup_marker = bool([True for marker in item.own_markers if marker.name == "performance_load_backup"])
75
+ has_load_data_marker = bool([True for marker in item.own_markers if marker.name == "performance_load_data"])
76
+
77
+ if has_create_backup_marker and not create_backup:
78
+ item.add_marker(no_create_backup_skip)
79
+ if has_load_backup_marker and not use_backup:
80
+ item.add_marker(no_load_backup_skip)
81
+ if has_load_data_marker and use_backup:
82
+ item.add_marker(use_backup_skip)
83
+
84
+
85
+ def pytest_sessionstart(session: pytest.Session) -> None:
86
+ session.infrahub_performance_test = InfrahubPerformanceTest(
87
+ results_url=session.config.getoption("infrahub_performance_test_result_address")
88
+ )
89
+
90
+
91
+ def pytest_sessionfinish(session: pytest.Session, exitstatus: int) -> None: # noqa: ARG001
92
+ """whole test run finishes."""
93
+ if not session.config.getoption("infrahub_performance_report"):
94
+ return
95
+
96
+ session.infrahub_performance_test.finalize(session=session)
97
+
98
+
99
+ def pytest_runtest_teardown(item: pytest.Item, nextitem: pytest.Item | None) -> None: # noqa: ARG001
100
+ """Fetch metrics at each test teardown because there's no better hook...
101
+ pytest_sessionfinish() is executed after fixtures has been finalized and pytest_fixture_post_finalizer() is too late"""
102
+ if not item.config.getoption("infrahub_performance_report"):
103
+ return
104
+
105
+ item.session.infrahub_performance_test.fetch_metrics()
106
+
107
+
108
+ def pytest_configure(config: pytest.Config) -> None:
109
+ if config.getoption("infrahub_performance_use_backup") and config.getoption("infrahub_performance_create_backup"):
110
+ raise pytest.UsageError("--performance-use-backup and --performance-create-backup are mutually exclusive")
111
+
112
+ config.addinivalue_line("markers", "performance_create_backup: Create a backup of the database")
113
+ config.addinivalue_line("markers", "performance_load_backup: Load the backup of the database")
114
+ config.addinivalue_line("markers", "performance_load_data: Load initial data into the database")
115
+
116
+
117
+ def pytest_terminal_summary(
118
+ terminalreporter: _pytest.terminal.TerminalReporter,
119
+ exitstatus: int, # noqa: ARG001
120
+ config: pytest.Config,
121
+ ) -> None:
122
+ if not config.getoption("infrahub_performance_report"):
123
+ return
124
+
125
+ performance_test = terminalreporter._session.infrahub_performance_test
126
+
127
+ report = [
128
+ f"{measurement.name}: {measurement.value} {measurement.unit.value}"
129
+ for measurement in performance_test.measurements
130
+ ]
131
+ terminalreporter.write("\n" + "\n".join(report) + "\n")
132
+
133
+
134
+ @pytest.fixture(scope="session")
135
+ def perf_test(request: pytest.FixtureRequest) -> InfrahubPerformanceTest:
136
+ return request.session.infrahub_performance_test
@@ -0,0 +1,30 @@
1
+ ---
2
+ global:
3
+ scrape_interval: 15s
4
+
5
+ scrape_configs:
6
+ - job_name: 'infrahub-server'
7
+ dns_sd_configs:
8
+ - names:
9
+ - server
10
+ type: A
11
+ port: 8000
12
+
13
+ - job_name: 'infrahub-worker'
14
+ dns_sd_configs:
15
+ - names:
16
+ - task-worker
17
+ type: A
18
+ port: 8000
19
+
20
+ - job_name: 'cadvisor'
21
+ dns_sd_configs:
22
+ - names:
23
+ - cadvisor
24
+ type: A
25
+ port: 8080
26
+ metric_relabel_configs:
27
+ - source_labels: [container_label_namespace]
28
+ target_label: namespace
29
+ - source_labels: [container_label_pod]
30
+ target_label: pod