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.
- infrahub/api/dependencies.py +6 -6
- infrahub/api/diff/validation_models.py +7 -7
- infrahub/api/schema.py +1 -1
- infrahub/artifacts/models.py +5 -3
- infrahub/artifacts/tasks.py +3 -5
- infrahub/cli/__init__.py +13 -9
- infrahub/cli/constants.py +3 -0
- infrahub/cli/db.py +165 -183
- infrahub/cli/upgrade.py +146 -0
- infrahub/computed_attribute/gather.py +185 -0
- infrahub/computed_attribute/models.py +240 -12
- infrahub/computed_attribute/tasks.py +77 -441
- infrahub/computed_attribute/triggers.py +13 -47
- infrahub/config.py +43 -32
- infrahub/context.py +14 -0
- infrahub/core/account.py +4 -4
- infrahub/core/attribute.py +58 -58
- infrahub/core/branch/tasks.py +74 -22
- infrahub/core/changelog/diff.py +95 -36
- infrahub/core/changelog/models.py +217 -43
- infrahub/core/constants/__init__.py +28 -0
- infrahub/core/constants/infrahubkind.py +2 -0
- infrahub/core/constants/schema.py +2 -0
- infrahub/core/constraint/node/runner.py +9 -8
- infrahub/core/diff/branch_differ.py +10 -10
- infrahub/core/diff/enricher/cardinality_one.py +5 -0
- infrahub/core/diff/enricher/hierarchy.py +17 -4
- infrahub/core/diff/enricher/labels.py +5 -0
- infrahub/core/diff/enricher/path_identifier.py +4 -0
- infrahub/core/diff/ipam_diff_parser.py +4 -5
- infrahub/core/diff/model/diff.py +27 -27
- infrahub/core/diff/model/path.py +32 -9
- infrahub/core/diff/parent_node_adder.py +78 -0
- infrahub/core/diff/payload_builder.py +13 -2
- infrahub/core/diff/query/filters.py +2 -2
- infrahub/core/diff/query/merge.py +20 -17
- infrahub/core/diff/query/save.py +188 -182
- infrahub/core/diff/query/summary_counts_enricher.py +51 -4
- infrahub/core/diff/query_parser.py +4 -4
- infrahub/core/diff/repository/deserializer.py +8 -3
- infrahub/core/diff/repository/repository.py +156 -38
- infrahub/core/diff/tasks.py +4 -4
- infrahub/core/graph/__init__.py +1 -1
- infrahub/core/graph/index.py +3 -0
- infrahub/core/initialization.py +1 -10
- infrahub/core/ipam/constants.py +3 -4
- infrahub/core/ipam/reconciler.py +12 -12
- infrahub/core/ipam/utilization.py +10 -13
- infrahub/core/manager.py +36 -36
- infrahub/core/merge.py +7 -7
- infrahub/core/migrations/__init__.py +2 -3
- infrahub/core/migrations/graph/__init__.py +12 -3
- infrahub/core/migrations/graph/m017_add_core_profile.py +1 -5
- infrahub/core/migrations/graph/m018_uniqueness_nulls.py +4 -4
- infrahub/core/migrations/graph/m019_restore_rels_to_time.py +256 -0
- infrahub/core/migrations/graph/m020_duplicate_edges.py +160 -0
- infrahub/core/migrations/graph/m021_missing_hierarchy_merge.py +51 -0
- infrahub/core/migrations/graph/m022_add_generate_template_attr.py +48 -0
- infrahub/core/migrations/graph/m023_deduplicate_cardinality_one_relationships.py +96 -0
- infrahub/core/migrations/query/attribute_add.py +2 -2
- infrahub/core/migrations/query/node_duplicate.py +43 -26
- infrahub/core/migrations/query/schema_attribute_update.py +2 -2
- infrahub/core/migrations/schema/models.py +19 -4
- infrahub/core/migrations/schema/node_remove.py +26 -12
- infrahub/core/migrations/schema/tasks.py +2 -2
- infrahub/core/migrations/shared.py +16 -16
- infrahub/core/models.py +15 -6
- infrahub/core/node/__init__.py +43 -39
- infrahub/core/node/base.py +2 -4
- infrahub/core/node/constraints/attribute_uniqueness.py +2 -2
- infrahub/core/node/constraints/grouped_uniqueness.py +99 -47
- infrahub/core/node/constraints/interface.py +1 -2
- infrahub/core/node/delete_validator.py +3 -5
- infrahub/core/node/ipam.py +4 -4
- infrahub/core/node/permissions.py +7 -7
- infrahub/core/node/resource_manager/ip_address_pool.py +6 -6
- infrahub/core/node/resource_manager/ip_prefix_pool.py +6 -6
- infrahub/core/node/resource_manager/number_pool.py +3 -3
- infrahub/core/path.py +12 -12
- infrahub/core/property.py +11 -11
- infrahub/core/protocols.py +7 -0
- infrahub/core/protocols_base.py +21 -21
- infrahub/core/query/__init__.py +33 -33
- infrahub/core/query/attribute.py +6 -4
- infrahub/core/query/diff.py +3 -3
- infrahub/core/query/node.py +82 -32
- infrahub/core/query/relationship.py +228 -40
- infrahub/core/query/resource_manager.py +2 -0
- infrahub/core/query/standard_node.py +3 -3
- infrahub/core/query/subquery.py +9 -9
- infrahub/core/registry.py +13 -15
- infrahub/core/relationship/constraints/count.py +3 -4
- infrahub/core/relationship/constraints/peer_kind.py +3 -4
- infrahub/core/relationship/constraints/profiles_kind.py +2 -2
- infrahub/core/relationship/model.py +51 -59
- infrahub/core/schema/attribute_schema.py +16 -8
- infrahub/core/schema/basenode_schema.py +105 -44
- infrahub/core/schema/computed_attribute.py +3 -3
- infrahub/core/schema/definitions/core/__init__.py +147 -0
- infrahub/core/schema/definitions/core/account.py +171 -0
- infrahub/core/schema/definitions/core/artifact.py +136 -0
- infrahub/core/schema/definitions/core/builtin.py +24 -0
- infrahub/core/schema/definitions/core/check.py +68 -0
- infrahub/core/schema/definitions/core/core.py +17 -0
- infrahub/core/schema/definitions/core/generator.py +100 -0
- infrahub/core/schema/definitions/core/graphql_query.py +79 -0
- infrahub/core/schema/definitions/core/group.py +108 -0
- infrahub/core/schema/definitions/core/ipam.py +193 -0
- infrahub/core/schema/definitions/core/lineage.py +19 -0
- infrahub/core/schema/definitions/core/menu.py +48 -0
- infrahub/core/schema/definitions/core/permission.py +163 -0
- infrahub/core/schema/definitions/core/profile.py +18 -0
- infrahub/core/schema/definitions/core/propose_change.py +97 -0
- infrahub/core/schema/definitions/core/propose_change_comment.py +193 -0
- infrahub/core/schema/definitions/core/propose_change_validator.py +328 -0
- infrahub/core/schema/definitions/core/repository.py +286 -0
- infrahub/core/schema/definitions/core/resource_pool.py +170 -0
- infrahub/core/schema/definitions/core/template.py +27 -0
- infrahub/core/schema/definitions/core/transform.py +96 -0
- infrahub/core/schema/definitions/core/webhook.py +134 -0
- infrahub/core/schema/definitions/internal.py +16 -16
- infrahub/core/schema/dropdown.py +3 -4
- infrahub/core/schema/generated/attribute_schema.py +15 -18
- infrahub/core/schema/generated/base_node_schema.py +12 -14
- infrahub/core/schema/generated/node_schema.py +3 -5
- infrahub/core/schema/generated/relationship_schema.py +9 -11
- infrahub/core/schema/generic_schema.py +2 -2
- infrahub/core/schema/manager.py +20 -9
- infrahub/core/schema/node_schema.py +4 -2
- infrahub/core/schema/relationship_schema.py +14 -6
- infrahub/core/schema/schema_branch.py +292 -144
- infrahub/core/schema/schema_branch_computed.py +41 -4
- infrahub/core/task/task.py +3 -3
- infrahub/core/task/user_task.py +15 -15
- infrahub/core/timestamp.py +3 -3
- infrahub/core/utils.py +20 -18
- infrahub/core/validators/__init__.py +1 -3
- infrahub/core/validators/aggregated_checker.py +2 -2
- infrahub/core/validators/attribute/choices.py +2 -2
- infrahub/core/validators/attribute/enum.py +2 -2
- infrahub/core/validators/attribute/kind.py +2 -2
- infrahub/core/validators/attribute/length.py +2 -2
- infrahub/core/validators/attribute/optional.py +2 -2
- infrahub/core/validators/attribute/regex.py +2 -2
- infrahub/core/validators/attribute/unique.py +2 -2
- infrahub/core/validators/checks_runner.py +25 -2
- infrahub/core/validators/determiner.py +1 -3
- infrahub/core/validators/interface.py +6 -2
- infrahub/core/validators/model.py +22 -3
- infrahub/core/validators/models/validate_migration.py +17 -4
- infrahub/core/validators/node/attribute.py +2 -2
- infrahub/core/validators/node/generate_profile.py +2 -2
- infrahub/core/validators/node/hierarchy.py +3 -5
- infrahub/core/validators/node/inherit_from.py +27 -5
- infrahub/core/validators/node/relationship.py +2 -2
- infrahub/core/validators/relationship/count.py +4 -4
- infrahub/core/validators/relationship/optional.py +2 -2
- infrahub/core/validators/relationship/peer.py +2 -2
- infrahub/core/validators/shared.py +2 -2
- infrahub/core/validators/tasks.py +8 -0
- infrahub/core/validators/uniqueness/checker.py +22 -21
- infrahub/core/validators/uniqueness/index.py +2 -2
- infrahub/core/validators/uniqueness/model.py +11 -11
- infrahub/database/__init__.py +27 -22
- infrahub/database/metrics.py +7 -1
- infrahub/dependencies/builder/constraint/grouped/node_runner.py +1 -3
- infrahub/dependencies/builder/diff/deserializer.py +3 -1
- infrahub/dependencies/builder/diff/enricher/hierarchy.py +3 -1
- infrahub/dependencies/builder/diff/parent_node_adder.py +8 -0
- infrahub/dependencies/component/registry.py +2 -2
- infrahub/events/__init__.py +25 -2
- infrahub/events/artifact_action.py +64 -0
- infrahub/events/branch_action.py +33 -22
- infrahub/events/generator.py +71 -0
- infrahub/events/group_action.py +51 -21
- infrahub/events/models.py +18 -19
- infrahub/events/node_action.py +88 -37
- infrahub/events/repository_action.py +5 -18
- infrahub/events/schema_action.py +4 -9
- infrahub/events/utils.py +16 -0
- infrahub/events/validator_action.py +55 -0
- infrahub/exceptions.py +32 -24
- infrahub/generators/models.py +2 -3
- infrahub/generators/tasks.py +24 -4
- infrahub/git/base.py +7 -7
- infrahub/git/integrator.py +48 -24
- infrahub/git/models.py +101 -9
- infrahub/git/repository.py +3 -3
- infrahub/git/tasks.py +408 -6
- infrahub/git/utils.py +48 -0
- infrahub/git/worktree.py +1 -2
- infrahub/git_credential/askpass.py +1 -2
- infrahub/graphql/analyzer.py +12 -0
- infrahub/graphql/app.py +13 -15
- infrahub/graphql/context.py +39 -0
- infrahub/graphql/initialization.py +3 -0
- infrahub/graphql/loaders/node.py +2 -12
- infrahub/graphql/loaders/peers.py +77 -0
- infrahub/graphql/loaders/shared.py +13 -0
- infrahub/graphql/manager.py +17 -19
- infrahub/graphql/mutations/artifact_definition.py +5 -5
- infrahub/graphql/mutations/branch.py +26 -1
- infrahub/graphql/mutations/computed_attribute.py +9 -5
- infrahub/graphql/mutations/diff.py +23 -11
- infrahub/graphql/mutations/diff_conflict.py +5 -0
- infrahub/graphql/mutations/generator.py +83 -0
- infrahub/graphql/mutations/graphql_query.py +5 -5
- infrahub/graphql/mutations/ipam.py +54 -74
- infrahub/graphql/mutations/main.py +195 -132
- infrahub/graphql/mutations/menu.py +7 -7
- infrahub/graphql/mutations/models.py +2 -4
- infrahub/graphql/mutations/node_getter/by_default_filter.py +10 -10
- infrahub/graphql/mutations/node_getter/by_hfid.py +1 -3
- infrahub/graphql/mutations/node_getter/by_id.py +1 -3
- infrahub/graphql/mutations/node_getter/interface.py +1 -2
- infrahub/graphql/mutations/proposed_change.py +7 -7
- infrahub/graphql/mutations/relationship.py +93 -19
- infrahub/graphql/mutations/repository.py +8 -8
- infrahub/graphql/mutations/resource_manager.py +3 -3
- infrahub/graphql/mutations/schema.py +19 -4
- infrahub/graphql/mutations/webhook.py +137 -0
- infrahub/graphql/parser.py +4 -4
- infrahub/graphql/permissions.py +1 -10
- infrahub/graphql/queries/diff/tree.py +19 -14
- infrahub/graphql/queries/event.py +5 -2
- infrahub/graphql/queries/ipam.py +2 -2
- infrahub/graphql/queries/relationship.py +2 -2
- infrahub/graphql/queries/search.py +2 -2
- infrahub/graphql/resolvers/many_relationship.py +264 -0
- infrahub/graphql/resolvers/resolver.py +13 -110
- infrahub/graphql/schema.py +2 -0
- infrahub/graphql/subscription/graphql_query.py +2 -0
- infrahub/graphql/types/context.py +12 -0
- infrahub/graphql/types/event.py +84 -17
- infrahub/graphql/types/node.py +2 -2
- infrahub/graphql/utils.py +2 -2
- infrahub/groups/ancestors.py +29 -0
- infrahub/groups/parsers.py +107 -0
- infrahub/lock.py +20 -20
- infrahub/menu/constants.py +0 -1
- infrahub/menu/generator.py +9 -21
- infrahub/menu/menu.py +17 -38
- infrahub/menu/models.py +117 -16
- infrahub/menu/repository.py +111 -0
- infrahub/menu/utils.py +5 -8
- infrahub/message_bus/__init__.py +11 -13
- infrahub/message_bus/messages/__init__.py +1 -21
- infrahub/message_bus/messages/check_generator_run.py +3 -3
- infrahub/message_bus/messages/finalize_validator_execution.py +3 -0
- infrahub/message_bus/messages/proposed_change/request_proposedchange_refreshartifacts.py +6 -0
- infrahub/message_bus/messages/request_generatordefinition_check.py +2 -0
- infrahub/message_bus/messages/send_echo_request.py +1 -1
- infrahub/message_bus/operations/__init__.py +1 -10
- infrahub/message_bus/operations/check/__init__.py +2 -2
- infrahub/message_bus/operations/check/generator.py +1 -0
- infrahub/message_bus/operations/event/__init__.py +2 -2
- infrahub/message_bus/operations/event/worker.py +0 -3
- infrahub/message_bus/operations/finalize/validator.py +51 -1
- infrahub/message_bus/operations/requests/__init__.py +0 -2
- infrahub/message_bus/operations/requests/generator_definition.py +21 -23
- infrahub/message_bus/operations/requests/proposed_change.py +14 -10
- infrahub/permissions/globals.py +15 -0
- infrahub/pools/number.py +2 -4
- infrahub/proposed_change/models.py +3 -0
- infrahub/proposed_change/tasks.py +58 -45
- infrahub/pytest_plugin.py +13 -10
- infrahub/server.py +2 -3
- infrahub/services/__init__.py +2 -2
- infrahub/services/adapters/cache/__init__.py +4 -6
- infrahub/services/adapters/cache/nats.py +4 -5
- infrahub/services/adapters/cache/redis.py +3 -7
- infrahub/services/adapters/event/__init__.py +1 -1
- infrahub/services/adapters/message_bus/__init__.py +3 -3
- infrahub/services/adapters/message_bus/local.py +2 -2
- infrahub/services/adapters/message_bus/nats.py +4 -4
- infrahub/services/adapters/message_bus/rabbitmq.py +4 -4
- infrahub/services/adapters/workflow/local.py +2 -2
- infrahub/services/component.py +5 -5
- infrahub/services/protocols.py +7 -7
- infrahub/services/scheduler.py +1 -3
- infrahub/task_manager/event.py +102 -9
- infrahub/task_manager/models.py +27 -7
- infrahub/tasks/artifact.py +7 -6
- infrahub/telemetry/__init__.py +0 -0
- infrahub/telemetry/constants.py +9 -0
- infrahub/telemetry/database.py +86 -0
- infrahub/telemetry/models.py +65 -0
- infrahub/telemetry/task_manager.py +77 -0
- infrahub/{tasks/telemetry.py → telemetry/tasks.py} +49 -56
- infrahub/telemetry/utils.py +11 -0
- infrahub/trace.py +4 -4
- infrahub/transformations/tasks.py +2 -2
- infrahub/trigger/catalogue.py +4 -6
- infrahub/trigger/constants.py +0 -8
- infrahub/trigger/models.py +54 -5
- infrahub/trigger/setup.py +90 -0
- infrahub/trigger/tasks.py +35 -84
- infrahub/utils.py +11 -1
- infrahub/validators/__init__.py +0 -0
- infrahub/validators/events.py +42 -0
- infrahub/validators/tasks.py +41 -0
- infrahub/webhook/gather.py +17 -0
- infrahub/webhook/models.py +176 -44
- infrahub/webhook/tasks.py +154 -155
- infrahub/webhook/triggers.py +31 -7
- infrahub/workers/infrahub_async.py +2 -2
- infrahub/workers/utils.py +2 -2
- infrahub/workflows/catalogue.py +86 -35
- infrahub/workflows/initialization.py +8 -2
- infrahub/workflows/models.py +27 -1
- infrahub/workflows/utils.py +10 -1
- infrahub_sdk/client.py +35 -8
- infrahub_sdk/config.py +3 -0
- infrahub_sdk/context.py +13 -0
- infrahub_sdk/ctl/branch.py +3 -2
- infrahub_sdk/ctl/cli_commands.py +5 -1
- infrahub_sdk/ctl/utils.py +0 -16
- infrahub_sdk/exceptions.py +12 -0
- infrahub_sdk/generator.py +4 -1
- infrahub_sdk/graphql.py +45 -13
- infrahub_sdk/node.py +71 -22
- infrahub_sdk/protocols.py +21 -8
- infrahub_sdk/protocols_base.py +32 -11
- infrahub_sdk/query_groups.py +6 -35
- infrahub_sdk/schema/__init__.py +55 -26
- infrahub_sdk/schema/main.py +8 -0
- infrahub_sdk/task/__init__.py +11 -0
- infrahub_sdk/task/constants.py +3 -0
- infrahub_sdk/task/exceptions.py +25 -0
- infrahub_sdk/task/manager.py +551 -0
- infrahub_sdk/task/models.py +74 -0
- infrahub_sdk/testing/schemas/animal.py +9 -0
- infrahub_sdk/timestamp.py +142 -33
- infrahub_sdk/utils.py +29 -1
- {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/METADATA +8 -6
- {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/RECORD +349 -293
- {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/entry_points.txt +1 -0
- infrahub_testcontainers/constants.py +2 -0
- infrahub_testcontainers/container.py +157 -12
- infrahub_testcontainers/docker-compose.test.yml +31 -6
- infrahub_testcontainers/helpers.py +18 -73
- infrahub_testcontainers/host.py +41 -0
- infrahub_testcontainers/measurements.py +93 -0
- infrahub_testcontainers/models.py +38 -0
- infrahub_testcontainers/performance_test.py +166 -0
- infrahub_testcontainers/plugin.py +136 -0
- infrahub_testcontainers/prometheus.yml +30 -0
- infrahub/core/schema/definitions/core.py +0 -2286
- infrahub/message_bus/messages/check_repository_checkdefinition.py +0 -20
- infrahub/message_bus/messages/check_repository_mergeconflicts.py +0 -16
- infrahub/message_bus/messages/check_repository_usercheck.py +0 -26
- infrahub/message_bus/messages/event_branch_create.py +0 -11
- infrahub/message_bus/messages/event_branch_delete.py +0 -11
- infrahub/message_bus/messages/event_branch_rebased.py +0 -9
- infrahub/message_bus/messages/event_node_mutated.py +0 -15
- infrahub/message_bus/messages/event_schema_update.py +0 -9
- infrahub/message_bus/messages/request_repository_checks.py +0 -12
- infrahub/message_bus/messages/request_repository_userchecks.py +0 -18
- infrahub/message_bus/operations/check/repository.py +0 -293
- infrahub/message_bus/operations/event/node.py +0 -20
- infrahub/message_bus/operations/event/schema.py +0 -17
- infrahub/message_bus/operations/requests/repository.py +0 -133
- infrahub/webhook/constants.py +0 -1
- {infrahub_server-1.2.0rc0.dist-info → infrahub_server-1.2.1.dist-info}/LICENSE.txt +0 -0
- {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
|