216 lines
6.9 KiB
Python
216 lines
6.9 KiB
Python
"""The sentry integration."""
|
|
from __future__ import annotations
|
|
|
|
import re
|
|
|
|
import sentry_sdk
|
|
from sentry_sdk.integrations.aiohttp import AioHttpIntegration
|
|
from sentry_sdk.integrations.logging import LoggingIntegration
|
|
from sentry_sdk.integrations.sqlalchemy import SqlalchemyIntegration
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
EVENT_HOMEASSISTANT_STARTED,
|
|
__version__ as current_version,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers import config_validation as cv, entity_platform
|
|
from homeassistant.loader import Integration, async_get_custom_components
|
|
|
|
from .const import (
|
|
CONF_DSN,
|
|
CONF_ENVIRONMENT,
|
|
CONF_EVENT_CUSTOM_COMPONENTS,
|
|
CONF_EVENT_HANDLED,
|
|
CONF_EVENT_THIRD_PARTY_PACKAGES,
|
|
CONF_LOGGING_EVENT_LEVEL,
|
|
CONF_LOGGING_LEVEL,
|
|
CONF_TRACING,
|
|
CONF_TRACING_SAMPLE_RATE,
|
|
DEFAULT_LOGGING_EVENT_LEVEL,
|
|
DEFAULT_LOGGING_LEVEL,
|
|
DEFAULT_TRACING_SAMPLE_RATE,
|
|
DOMAIN,
|
|
ENTITY_COMPONENTS,
|
|
)
|
|
|
|
CONFIG_SCHEMA = cv.deprecated(DOMAIN)
|
|
|
|
|
|
LOGGER_INFO_REGEX = re.compile(r"^(\w+)\.?(\w+)?\.?(\w+)?\.?(\w+)?(?:\..*)?$")
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: dict) -> bool:
|
|
"""Set up the Sentry component."""
|
|
return True
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up Sentry from a config entry."""
|
|
|
|
# Migrate environment from config entry data to config entry options
|
|
if (
|
|
CONF_ENVIRONMENT not in entry.options
|
|
and CONF_ENVIRONMENT in entry.data
|
|
and entry.data[CONF_ENVIRONMENT]
|
|
):
|
|
options = {**entry.options, CONF_ENVIRONMENT: entry.data[CONF_ENVIRONMENT]}
|
|
data = entry.data.copy()
|
|
data.pop(CONF_ENVIRONMENT)
|
|
hass.config_entries.async_update_entry(entry, data=data, options=options)
|
|
|
|
# https://docs.sentry.io/platforms/python/logging/
|
|
sentry_logging = LoggingIntegration(
|
|
level=entry.options.get(CONF_LOGGING_LEVEL, DEFAULT_LOGGING_LEVEL),
|
|
event_level=entry.options.get(
|
|
CONF_LOGGING_EVENT_LEVEL, DEFAULT_LOGGING_EVENT_LEVEL
|
|
),
|
|
)
|
|
|
|
# Additional/extra data collection
|
|
channel = get_channel(current_version)
|
|
huuid = await hass.helpers.instance_id.async_get()
|
|
system_info = await hass.helpers.system_info.async_get_system_info()
|
|
custom_components = await async_get_custom_components(hass)
|
|
|
|
tracing = {}
|
|
if entry.options.get(CONF_TRACING):
|
|
tracing = {
|
|
"traces_sample_rate": entry.options.get(
|
|
CONF_TRACING_SAMPLE_RATE, DEFAULT_TRACING_SAMPLE_RATE
|
|
),
|
|
}
|
|
|
|
sentry_sdk.init(
|
|
dsn=entry.data[CONF_DSN],
|
|
environment=entry.options.get(CONF_ENVIRONMENT),
|
|
integrations=[sentry_logging, AioHttpIntegration(), SqlalchemyIntegration()],
|
|
release=current_version,
|
|
before_send=lambda event, hint: process_before_send(
|
|
hass,
|
|
entry.options,
|
|
channel,
|
|
huuid,
|
|
system_info,
|
|
custom_components,
|
|
event,
|
|
hint,
|
|
),
|
|
**tracing,
|
|
)
|
|
|
|
async def update_system_info(now):
|
|
nonlocal system_info
|
|
system_info = await hass.helpers.system_info.async_get_system_info()
|
|
|
|
# Update system info every hour
|
|
hass.helpers.event.async_call_later(3600, update_system_info)
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STARTED, update_system_info)
|
|
|
|
return True
|
|
|
|
|
|
def get_channel(version: str) -> str:
|
|
"""Find channel based on version number."""
|
|
if "dev0" in version:
|
|
return "dev"
|
|
if "dev" in version:
|
|
return "nightly"
|
|
if "b" in version:
|
|
return "beta"
|
|
return "stable"
|
|
|
|
|
|
def process_before_send(
|
|
hass: HomeAssistant,
|
|
options,
|
|
channel: str,
|
|
huuid: str,
|
|
system_info: dict[str, bool | str],
|
|
custom_components: dict[str, Integration],
|
|
event,
|
|
hint,
|
|
):
|
|
"""Process a Sentry event before sending it to Sentry."""
|
|
# Filter out handled events by default
|
|
if (
|
|
"tags" in event
|
|
and event.tags.get("handled", "no") == "yes"
|
|
and not options.get(CONF_EVENT_HANDLED)
|
|
):
|
|
return None
|
|
|
|
# Additional tags to add to the event
|
|
additional_tags = {
|
|
"channel": channel,
|
|
"installation_type": system_info["installation_type"],
|
|
"uuid": huuid,
|
|
}
|
|
|
|
# Find out all integrations in use, filter "auth", because it
|
|
# triggers security rules, hiding all data.
|
|
integrations = [
|
|
integration
|
|
for integration in hass.config.components
|
|
if integration != "auth" and "." not in integration
|
|
]
|
|
|
|
# Add additional tags based on what caused the event.
|
|
platform = entity_platform.current_platform.get()
|
|
if platform is not None:
|
|
# This event happened in a platform
|
|
additional_tags["custom_component"] = "no"
|
|
additional_tags["integration"] = platform.platform_name
|
|
additional_tags["platform"] = platform.domain
|
|
elif "logger" in event:
|
|
# Logger event, try to get integration information from the logger name.
|
|
matches = LOGGER_INFO_REGEX.findall(event["logger"])
|
|
if matches:
|
|
group1, group2, group3, group4 = matches[0]
|
|
# Handle the "homeassistant." package differently
|
|
if group1 == "homeassistant" and group2 and group3:
|
|
if group2 == "components":
|
|
# This logger is from a component
|
|
additional_tags["custom_component"] = "no"
|
|
additional_tags["integration"] = group3
|
|
if group4 and group4 in ENTITY_COMPONENTS:
|
|
additional_tags["platform"] = group4
|
|
else:
|
|
# Not a component, could be helper, or something else.
|
|
additional_tags[group2] = group3
|
|
else:
|
|
# Not the "homeassistant" package, this third-party
|
|
if not options.get(CONF_EVENT_THIRD_PARTY_PACKAGES):
|
|
return None
|
|
additional_tags["package"] = group1
|
|
|
|
# If this event is caused by an integration, add a tag if this
|
|
# integration is custom or not.
|
|
if (
|
|
"integration" in additional_tags
|
|
and additional_tags["integration"] in custom_components
|
|
):
|
|
if not options.get(CONF_EVENT_CUSTOM_COMPONENTS):
|
|
return None
|
|
additional_tags["custom_component"] = "yes"
|
|
|
|
# Update event with the additional tags
|
|
event.setdefault("tags", {}).update(additional_tags)
|
|
|
|
# Set user context to the installation UUID
|
|
event.setdefault("user", {}).update({"id": huuid})
|
|
|
|
# Update event data with Home Assistant Context
|
|
event.setdefault("contexts", {}).update(
|
|
{
|
|
"Home Assistant": {
|
|
"channel": channel,
|
|
"custom_components": "\n".join(sorted(custom_components)),
|
|
"integrations": "\n".join(sorted(integrations)),
|
|
**system_info,
|
|
},
|
|
}
|
|
)
|
|
return event
|