From 4740a7b930a13fedf58c5ea48cd098b1b853dec2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Rub=C3=A9n=20De=20la=20Torre=20Vico?= Date: Tue, 13 Feb 2024 12:27:12 +0000 Subject: [PATCH] feat(azure): check related with App Insights service (#3395) --- poetry.lock | 64 +++++++++------ .../azure/services/appinsights/__init__.py | 0 .../appinsights/appinsights_client.py | 4 + .../__init__.py | 0 ...nsights_ensure_is_configured.metadata.json | 30 ++++++++ .../appinsights_ensure_is_configured.py | 25 ++++++ .../appinsights/appinsights_service.py | 44 +++++++++++ pyproject.toml | 1 + .../appinsights_ensure_is_configured_test.py | 77 +++++++++++++++++++ .../appinsights/appinsights_service_test.py | 50 ++++++++++++ 10 files changed, 271 insertions(+), 24 deletions(-) create mode 100644 prowler/providers/azure/services/appinsights/__init__.py create mode 100644 prowler/providers/azure/services/appinsights/appinsights_client.py create mode 100644 prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/__init__.py create mode 100644 prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.metadata.json create mode 100644 prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.py create mode 100644 prowler/providers/azure/services/appinsights/appinsights_service.py create mode 100644 tests/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured_test.py create mode 100644 tests/providers/azure/services/appinsights/appinsights_service_test.py diff --git a/poetry.lock b/poetry.lock index f0c09411..3469fac6 100644 --- a/poetry.lock +++ b/poetry.lock @@ -326,6 +326,22 @@ cryptography = ">=2.5" msal = ">=1.24.0,<2.0.0" msal-extensions = ">=0.3.0,<2.0.0" +[[package]] +name = "azure-mgmt-applicationinsights" +version = "4.0.0" +description = "Microsoft Azure Application Insights Management Client Library for Python" +optional = false +python-versions = ">=3.7" +files = [ + {file = "azure-mgmt-applicationinsights-4.0.0.zip", hash = "sha256:50c3db05573e0cc2d56314a0556fb346ef05ec489ac000f4d720d92c6b647e06"}, + {file = "azure_mgmt_applicationinsights-4.0.0-py3-none-any.whl", hash = "sha256:2b1ffd9a0114974455795c73a3a5d17c849e32b961d707d2db393b99254b576f"}, +] + +[package.dependencies] +azure-common = ">=1.1,<2.0" +azure-mgmt-core = ">=1.3.2,<2.0.0" +isodate = ">=0.6.1,<1.0.0" + [[package]] name = "azure-mgmt-authorization" version = "4.0.0" @@ -649,17 +665,17 @@ pycparser = "*" [[package]] name = "cfn-lint" -version = "0.85.0" +version = "0.85.1" description = "Checks CloudFormation templates for practices and behaviour that could potentially be improved" optional = false python-versions = ">=3.8, <=4.0, !=4.0" files = [ - {file = "cfn-lint-0.85.0.tar.gz", hash = "sha256:64d6e8d85cdc573b61add78f9ff95a142a1834edb4793d1291551f6d953f73fe"}, - {file = "cfn_lint-0.85.0-py3-none-any.whl", hash = "sha256:e4849e1779bd1a9f4543617372708a20519b6d7cad5f980e20c6deaa227361a2"}, + {file = "cfn-lint-0.85.1.tar.gz", hash = "sha256:f003603a6f13bcda125c60f5021fc19b96f18a27ebc44498947709cb7627d0d6"}, + {file = "cfn_lint-0.85.1-py3-none-any.whl", hash = "sha256:5d5b31609ded0bc513f1c57c0dc0017ec1613c2b33ef8e74802149bedb01a3de"}, ] [package.dependencies] -aws-sam-translator = ">=1.83.0" +aws-sam-translator = ">=1.84.0" jschema-to-python = ">=1.2.3,<1.3.0" jsonpatch = "*" jsonschema = ">=3.0,<5" @@ -1233,13 +1249,13 @@ dev = ["flake8", "markdown", "twine", "wheel"] [[package]] name = "google-api-core" -version = "2.16.2" +version = "2.17.0" description = "Google API client core library" optional = false python-versions = ">=3.7" files = [ - {file = "google-api-core-2.16.2.tar.gz", hash = "sha256:032d37b45d1d6bdaf68fb11ff621e2593263a239fa9246e2e94325f9c47876d2"}, - {file = "google_api_core-2.16.2-py3-none-any.whl", hash = "sha256:449ca0e3f14c179b4165b664256066c7861610f70b6ffe54bb01a04e9b466929"}, + {file = "google-api-core-2.17.0.tar.gz", hash = "sha256:de7ef0450faec7c75e0aea313f29ac870fdc44cfaec9d6499a9a17305980ef66"}, + {file = "google_api_core-2.17.0-py3-none-any.whl", hash = "sha256:08ed79ed8e93e329de5e3e7452746b734e6bf8438d8d64dd3319d21d3164890c"}, ] [package.dependencies] @@ -1901,13 +1917,13 @@ files = [ [[package]] name = "microsoft-kiota-abstractions" -version = "1.1.0" +version = "1.2.0" description = "Core abstractions for kiota generated libraries in Python" optional = false python-versions = "*" files = [ - {file = "microsoft_kiota_abstractions-1.1.0-py2.py3-none-any.whl", hash = "sha256:de076a12aed5d39d4eb2f4a36d483ba6a74086b1f4ef2d2ead54a6e1935d1af2"}, - {file = "microsoft_kiota_abstractions-1.1.0.tar.gz", hash = "sha256:41609aa57518b0fe470a97c874ed31cfe0630162e7bef294e8a3fb63ce6d0568"}, + {file = "microsoft_kiota_abstractions-1.2.0-py2.py3-none-any.whl", hash = "sha256:09061cec7e9c9da8bc3eced4488118c8b4a7ebd26aa8de2b46c43f07fee5610a"}, + {file = "microsoft_kiota_abstractions-1.2.0.tar.gz", hash = "sha256:2b38241402995952c5598923391dcbb84bd97e48c0d76f93e2817c69ba19e3f6"}, ] [package.dependencies] @@ -1935,13 +1951,13 @@ opentelemetry-sdk = ">=1.20.0" [[package]] name = "microsoft-kiota-http" -version = "1.2.1" +version = "1.3.0" description = "Kiota http request adapter implementation for httpx library" optional = false python-versions = "*" files = [ - {file = "microsoft_kiota_http-1.2.1-py2.py3-none-any.whl", hash = "sha256:9cadd2a36d0f933d59059814b4b8ba7fabaa28f7c2829e12951ce9e38d45f0cd"}, - {file = "microsoft_kiota_http-1.2.1.tar.gz", hash = "sha256:d0dcff25cf456b8a3b23e4fd1866ffd669aaa0b9b26ef562ea58f698e4b7ad50"}, + {file = "microsoft_kiota_http-1.3.0-py2.py3-none-any.whl", hash = "sha256:4be1fd892dcef31ecd8d616969e970437a0b6b98c3223980d6c4aa55582729aa"}, + {file = "microsoft_kiota_http-1.3.0.tar.gz", hash = "sha256:ae97bbfe4544c94e43591b44c8a60791689f31f7d16a9154df7738e829a28481"}, ] [package.dependencies] @@ -3660,18 +3676,18 @@ contextlib2 = ">=0.5.5" [[package]] name = "setuptools" -version = "69.0.3" +version = "69.1.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false python-versions = ">=3.8" files = [ - {file = "setuptools-69.0.3-py3-none-any.whl", hash = "sha256:385eb4edd9c9d5c17540511303e39a147ce2fc04bc55289c322b9e5904fe2c05"}, - {file = "setuptools-69.0.3.tar.gz", hash = "sha256:be1af57fc409f93647f2e8e4573a142ed38724b8cdd389706a867bb4efcf1e78"}, + {file = "setuptools-69.1.0-py3-none-any.whl", hash = "sha256:c054629b81b946d63a9c6e732bc8b2513a7c3ea645f11d0139a2191d735c60c6"}, + {file = "setuptools-69.1.0.tar.gz", hash = "sha256:850894c4195f09c4ed30dba56213bf7c3f21d86ed6bdaafb5df5972593bfc401"}, ] [package.extras] docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] -testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.1)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] [[package]] @@ -3745,13 +3761,13 @@ dev = ["twine", "wheel", "yapf"] [[package]] name = "std-uritemplate" -version = "0.0.50" +version = "0.0.52" description = "std-uritemplate implementation for Python" optional = false python-versions = ">=3.8,<4.0" files = [ - {file = "std_uritemplate-0.0.50-py3-none-any.whl", hash = "sha256:1b5ae735ef12cce939080d306673e58d1cff1fd6e8a80765288210d4f01e30c4"}, - {file = "std_uritemplate-0.0.50.tar.gz", hash = "sha256:65e30dc6d666fab96d5b8a43f7fe3a94711515d13148bedbe4dd364f701893e9"}, + {file = "std_uritemplate-0.0.52-py3-none-any.whl", hash = "sha256:33beeee9d74800a7d13f99f9913d8b0225af37f96caf8922de05185ab86a431a"}, + {file = "std_uritemplate-0.0.52.tar.gz", hash = "sha256:45b75ea1f7754f630a99590480d7c5473605a186cc0cef8d0f750d336a264f60"}, ] [[package]] @@ -3872,13 +3888,13 @@ files = [ [[package]] name = "tzdata" -version = "2023.4" +version = "2024.1" description = "Provider of IANA time zone data" optional = false python-versions = ">=2" files = [ - {file = "tzdata-2023.4-py2.py3-none-any.whl", hash = "sha256:aa3ace4329eeacda5b7beb7ea08ece826c28d761cda36e747cfbf97996d39bf3"}, - {file = "tzdata-2023.4.tar.gz", hash = "sha256:dd54c94f294765522c77399649b4fefd95522479a664a0cec87f41bebc6148c9"}, + {file = "tzdata-2024.1-py2.py3-none-any.whl", hash = "sha256:9068bc196136463f5245e51efda838afa15aaeca9903f49050dfa2679db4d252"}, + {file = "tzdata-2024.1.tar.gz", hash = "sha256:2674120f8d891909751c38abcdfd386ac0a5a1127954fbc332af6b5ceae07efd"}, ] [[package]] @@ -4205,4 +4221,4 @@ docs = ["mkdocs", "mkdocs-material"] [metadata] lock-version = "2.0" python-versions = ">=3.9,<3.13" -content-hash = "224f2518ad6d40eb4773d8812dec0480e8344c35b02db61c0f840d011788dc7a" +content-hash = "e07cca1a69753a598ad37052f19c8c839d52d6bffa7bdd9ee86e088a49870d61" diff --git a/prowler/providers/azure/services/appinsights/__init__.py b/prowler/providers/azure/services/appinsights/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/prowler/providers/azure/services/appinsights/appinsights_client.py b/prowler/providers/azure/services/appinsights/appinsights_client.py new file mode 100644 index 00000000..ed3cc15d --- /dev/null +++ b/prowler/providers/azure/services/appinsights/appinsights_client.py @@ -0,0 +1,4 @@ +from prowler.providers.azure.lib.audit_info.audit_info import azure_audit_info +from prowler.providers.azure.services.appinsights.appinsights_service import AppInsights + +appinsights_client = AppInsights(azure_audit_info) diff --git a/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/__init__.py b/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.metadata.json b/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.metadata.json new file mode 100644 index 00000000..badc97a6 --- /dev/null +++ b/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.metadata.json @@ -0,0 +1,30 @@ +{ + "Provider": "azure", + "CheckID": "appinsights_ensure_is_configured", + "CheckTitle": "Ensure Application Insights are Configured.", + "CheckType": [], + "ServiceName": "appinsights", + "SubServiceName": "", + "ResourceIdTemplate": "", + "Severity": "low", + "ResourceType": "Microsoft.Insights/components", + "Description": "Application Insights within Azure act as an Application Performance Monitoring solution providing valuable data into how well an application performs and additional information when performing incident response. The types of log data collected include application metrics, telemetry data, and application trace logging data providing organizations with detailed information about application activity and application transactions. Both data sets help organizations adopt a proactive and retroactive means to handle security and performance related metrics within their modern applications.", + "Risk": "Configuring Application Insights provides additional data not found elsewhere within Azure as part of a much larger logging and monitoring program within an organization's Information Security practice. The types and contents of these logs will act as both a potential cost saving measure (application performance) and a means to potentially confirm the source of a potential incident (trace logging). Metrics and Telemetry data provide organizations with a proactive approach to cost savings by monitoring an application's performance, while the trace logging data provides necessary details in a reactive incident response scenario by helping organizations identify the potential source of an incident within their application.", + "RelatedUrl": "https://learn.microsoft.com/en-us/azure/azure-monitor/app/app-insights-overview", + "Remediation": { + "Code": { + "CLI": "az monitor app-insights component create --app --resource-group --location --kind 'web' --retention-time --workspace -- subscription ", + "NativeIaC": "", + "Other": "https://www.tenable.com/audits/items/CIS_Microsoft_Azure_Foundations_v2.0.0_L2.audit:8a7a608d180042689ad9d3f16aa359f1", + "Terraform": "" + }, + "Recommendation": { + "Text": "1. Navigate to Application Insights 2. Under the Basics tab within the PROJECT DETAILS section, select the Subscription 3. Select the Resource group 4. Within the INSTANCE DETAILS, enter a Name 5. Select a Region 6. Next to Resource Mode, select Workspace-based 7. Within the WORKSPACE DETAILS, select the Subscription for the log analytics workspace 8. Select the appropriate Log Analytics Workspace 9. Click Next:Tags > 10. Enter the appropriate Tags as Name, Value pairs. 11. Click Next:Review+Create 12. Click Create.", + "Url": "" + } + }, + "Categories": [], + "DependsOn": [], + "RelatedTo": [], + "Notes": "Because Application Insights relies on a Log Analytics Workspace, an organization will incur additional expenses when using this service." +} diff --git a/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.py b/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.py new file mode 100644 index 00000000..10b6ca0c --- /dev/null +++ b/prowler/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured.py @@ -0,0 +1,25 @@ +from prowler.lib.check.models import Check, Check_Report_Azure +from prowler.providers.azure.services.appinsights.appinsights_client import ( + appinsights_client, +) + + +class appinsights_ensure_is_configured(Check): + def execute(self) -> Check_Report_Azure: + findings = [] + + for subscription_name, components in appinsights_client.components.items(): + report = Check_Report_Azure(self.metadata()) + report.status = "PASS" + report.subscription = subscription_name + report.resource_name = "AppInsights" + report.resource_id = "AppInsights" + report.status_extended = f"There is at least one AppInsight configured in susbscription {subscription_name}." + + if len(components) < 1: + report.status = "FAIL" + report.status_extended = f"There are no AppInsight configured in susbscription {subscription_name}." + + findings.append(report) + + return findings diff --git a/prowler/providers/azure/services/appinsights/appinsights_service.py b/prowler/providers/azure/services/appinsights/appinsights_service.py new file mode 100644 index 00000000..dc3140b0 --- /dev/null +++ b/prowler/providers/azure/services/appinsights/appinsights_service.py @@ -0,0 +1,44 @@ +from dataclasses import dataclass + +from azure.mgmt.applicationinsights import ApplicationInsightsManagementClient + +from prowler.lib.logger import logger +from prowler.providers.azure.lib.audit_info.models import Azure_Audit_Info +from prowler.providers.azure.lib.service.service import AzureService + + +########################## AppInsights +class AppInsights(AzureService): + def __init__(self, audit_info: Azure_Audit_Info): + super().__init__(ApplicationInsightsManagementClient, audit_info) + self.components = self.__get_components__() + + def __get_components__(self): + logger.info("AppInsights - Getting components...") + components = {} + + for subscription_name, client in self.clients.items(): + try: + components_list = client.components.list() + components.update({subscription_name: {}}) + + for component in components_list: + components[subscription_name].update( + { + component.app_id: Component( + resource_id=component.id, resource_name=component.name + ) + } + ) + except Exception as error: + logger.error( + f"Subscription name: {subscription_name} -- {error.__class__.__name__}[{error.__traceback__.tb_lineno}]: {error}" + ) + + return components + + +@dataclass +class Component: + resource_id: str + resource_name: str diff --git a/pyproject.toml b/pyproject.toml index 0239a7eb..44469df9 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -28,6 +28,7 @@ version = "3.13.0" alive-progress = "3.1.5" awsipranges = "0.3.3" azure-identity = "1.15.0" +azure-mgmt-applicationinsights = "4.0.0" azure-mgmt-authorization = "4.0.0" azure-mgmt-security = "6.0.0" azure-mgmt-sql = "3.0.1" diff --git a/tests/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured_test.py b/tests/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured_test.py new file mode 100644 index 00000000..9ccaf7e8 --- /dev/null +++ b/tests/providers/azure/services/appinsights/appinsights_ensure_is_configured/appinsights_ensure_is_configured_test.py @@ -0,0 +1,77 @@ +from unittest import mock + +from prowler.providers.azure.services.appinsights.appinsights_service import Component +from tests.providers.azure.azure_fixtures import AZURE_SUBSCRIPTION + + +class Test_appinsights_ensure_is_configured: + def test_appinsights_no_subscriptions(self): + appinsights_client = mock.MagicMock + appinsights_client.components = {} + + with mock.patch( + "prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured.appinsights_client", + new=appinsights_client, + ): + from prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured import ( + appinsights_ensure_is_configured, + ) + + check = appinsights_ensure_is_configured() + result = check.execute() + assert len(result) == 0 + + def test_no_appinsights(self): + appinsights_client = mock.MagicMock + appinsights_client.components = {AZURE_SUBSCRIPTION: {}} + + with mock.patch( + "prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured.appinsights_client", + new=appinsights_client, + ): + from prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured import ( + appinsights_ensure_is_configured, + ) + + check = appinsights_ensure_is_configured() + result = check.execute() + assert len(result) == 1 + assert result[0].subscription == AZURE_SUBSCRIPTION + assert result[0].status == "FAIL" + assert result[0].resource_id == "AppInsights" + assert result[0].resource_name == "AppInsights" + assert ( + result[0].status_extended + == f"There are no AppInsight configured in susbscription {AZURE_SUBSCRIPTION}." + ) + + def test_appinsights_configured(self): + appinsights_client = mock.MagicMock + appinsights_client.components = { + AZURE_SUBSCRIPTION: { + "app_id-1": Component( + resource_id="/subscriptions/resource_id", + resource_name="AppInsightsTest", + ) + } + } + + with mock.patch( + "prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured.appinsights_client", + new=appinsights_client, + ): + from prowler.providers.azure.services.appinsights.appinsights_ensure_is_configured.appinsights_ensure_is_configured import ( + appinsights_ensure_is_configured, + ) + + check = appinsights_ensure_is_configured() + result = check.execute() + assert len(result) == 1 + assert result[0].subscription == AZURE_SUBSCRIPTION + assert result[0].status == "PASS" + assert result[0].resource_id == "AppInsights" + assert result[0].resource_name == "AppInsights" + assert ( + result[0].status_extended + == f"There is at least one AppInsight configured in susbscription {AZURE_SUBSCRIPTION}." + ) diff --git a/tests/providers/azure/services/appinsights/appinsights_service_test.py b/tests/providers/azure/services/appinsights/appinsights_service_test.py new file mode 100644 index 00000000..04b6d836 --- /dev/null +++ b/tests/providers/azure/services/appinsights/appinsights_service_test.py @@ -0,0 +1,50 @@ +from unittest.mock import patch + +from prowler.providers.azure.services.appinsights.appinsights_service import ( + AppInsights, + Component, +) +from tests.providers.azure.azure_fixtures import ( + AZURE_SUBSCRIPTION, + set_mocked_azure_audit_info, +) + + +def mock_appinsights_get_components(_): + return { + AZURE_SUBSCRIPTION: { + "app_id-1": Component( + resource_id="/subscriptions/resource_id", + resource_name="AppInsightsTest", + ) + } + } + + +@patch( + "prowler.providers.azure.services.appinsights.appinsights_service.AppInsights.__get_components__", + new=mock_appinsights_get_components, +) +class Test_AppInsights_Service: + def test__get_client__(self): + app_insights = AppInsights(set_mocked_azure_audit_info()) + assert ( + app_insights.clients[AZURE_SUBSCRIPTION].__class__.__name__ + == "ApplicationInsightsManagementClient" + ) + + def test__get_subscriptions__(self): + app_insights = AppInsights(set_mocked_azure_audit_info()) + assert app_insights.subscriptions.__class__.__name__ == "dict" + + def test__get_componentes(self): + appinsights = AppInsights(set_mocked_azure_audit_info()) + assert len(appinsights.components) == 1 + assert ( + appinsights.components[AZURE_SUBSCRIPTION]["app_id-1"].resource_id + == "/subscriptions/resource_id" + ) + assert ( + appinsights.components[AZURE_SUBSCRIPTION]["app_id-1"].resource_name + == "AppInsightsTest" + )