Skip to content

Commit ba4a63d

Browse files
authored
♻️ ref: refactor notification building utils to prevent circular dependencies (#88030)
making my aci life easier i am running into a lot of circular dependency issues around these methods since we are importing slack specific utils and methods in a generic `messaging` class.
1 parent 26e0b74 commit ba4a63d

File tree

13 files changed

+72
-36
lines changed

13 files changed

+72
-36
lines changed

src/sentry/integrations/discord/message_builder/issues.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,12 @@ def build(self, notification_uuid: str | None = None) -> dict[str, object]:
7373
color=LEVEL_TO_COLOR[get_color(event_for_tags, self.notification, self.group)],
7474
# We can't embed urls in Discord embed footers.
7575
footer=DiscordMessageEmbedFooter(
76-
build_footer(self.group, project, self.rules, "{text}")
76+
build_footer(
77+
group=self.group,
78+
project=project,
79+
url_format="{text}",
80+
rules=self.rules,
81+
)
7782
),
7883
fields=build_tag_fields(event_for_tags, self.tags),
7984
timestamp=timestamp,

src/sentry/integrations/messaging/message_builder.py

+3-6
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55

66
from sentry import features
77
from sentry.eventstore.models import Event, GroupEvent
8-
from sentry.integrations.slack.message_builder.types import LEVEL_TO_COLOR, SLACK_URL_FORMAT
8+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
99
from sentry.integrations.types import EXTERNAL_PROVIDERS, ExternalProviders
1010
from sentry.issues.grouptype import GroupCategory
1111
from sentry.models.environment import Environment
@@ -173,7 +173,7 @@ def build_attachment_text(group: Group, event: Event | GroupEvent | None = None)
173173

174174

175175
def build_attachment_replay_link(
176-
group: Group, event: Event | GroupEvent | None = None, url_format: str = SLACK_URL_FORMAT
176+
group: Group, url_format: str, event: Event | GroupEvent | None = None
177177
) -> str | None:
178178
has_replay = features.has("organizations:session-replay", group.organization)
179179
has_slack_links = features.has(
@@ -199,8 +199,8 @@ def build_rule_url(rule: Any, group: Group, project: Project) -> str:
199199
def build_footer(
200200
group: Group,
201201
project: Project,
202+
url_format: str,
202203
rules: Sequence[Rule] | None = None,
203-
url_format: str = SLACK_URL_FORMAT,
204204
) -> str:
205205
footer = f"{group.qualified_short_id}"
206206
if rules:
@@ -210,9 +210,6 @@ def build_footer(
210210
text = rules[0].label if rules[0].label else "Test Alert"
211211
footer += f" via {url_format.format(text=text, url=rule_url)}"
212212

213-
if url_format == SLACK_URL_FORMAT:
214-
footer = url_format.format(text=text, url=rule_url)
215-
216213
if len(rules) > 1:
217214
footer += f" (+{len(rules) - 1} other)"
218215

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
# Attachment colors used for issues with no actions take.
2+
3+
LEVEL_TO_COLOR = {
4+
"_actioned_issue": "#EDEEEF",
5+
"_incident_resolved": "#4DC771",
6+
"debug": "#FBE14F",
7+
"error": "#E03E2F",
8+
"fatal": "#FA4747",
9+
"info": "#2788CE",
10+
"warning": "#FFC227",
11+
}

src/sentry/integrations/msteams/card_builder/issues.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -129,7 +129,12 @@ def build_group_footer(self) -> ColumnSetBlock:
129129

130130
image_column = create_footer_logo_block()
131131

132-
text = build_footer(self.group, project, self.rules, MSTEAMS_URL_FORMAT)
132+
text = build_footer(
133+
group=self.group,
134+
project=project,
135+
url_format=MSTEAMS_URL_FORMAT,
136+
rules=self.rules,
137+
)
133138

134139
text_column = create_footer_column_block(create_footer_text_block(text))
135140

src/sentry/integrations/slack/message_builder/incidents.py

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,10 @@
11
from datetime import datetime
22

33
from sentry.incidents.typings.metric_detector import AlertContext, MetricIssueContext
4+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
45
from sentry.integrations.metric_alerts import incident_attachment_info
56
from sentry.integrations.slack.message_builder.base.block import BlockSlackMessageBuilder
6-
from sentry.integrations.slack.message_builder.types import (
7-
INCIDENT_COLOR_MAPPING,
8-
LEVEL_TO_COLOR,
9-
SlackBody,
10-
)
7+
from sentry.integrations.slack.message_builder.types import INCIDENT_COLOR_MAPPING, SlackBody
118
from sentry.integrations.slack.utils.escape import escape_slack_text
129
from sentry.models.organization import Organization
1310

src/sentry/integrations/slack/message_builder/issues.py

+11-3
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@
1717
build_attachment_replay_link,
1818
build_attachment_text,
1919
build_attachment_title,
20-
build_footer,
2120
format_actor_option_slack,
2221
format_actor_options_slack,
2322
get_title_link,
@@ -32,6 +31,7 @@
3231
SLACK_URL_FORMAT,
3332
SlackBlock,
3433
)
34+
from sentry.integrations.slack.message_builder.util import build_slack_footer
3535
from sentry.integrations.slack.utils.escape import escape_slack_markdown_text, escape_slack_text
3636
from sentry.integrations.time_utils import get_approx_start_time, time_since
3737
from sentry.integrations.types import ExternalProviders
@@ -506,7 +506,11 @@ def get_suggested_assignees_block(self, suggested_assignees: list[str]) -> Slack
506506

507507
def get_footer(self) -> SlackBlock:
508508
# This link does not contain user input (it's a static label and a url), must not escape it.
509-
replay_link = build_attachment_replay_link(self.group, self.event)
509+
replay_link = build_attachment_replay_link(
510+
group=self.group,
511+
url_format=SLACK_URL_FORMAT,
512+
event=self.event,
513+
)
510514

511515
timestamp = None
512516
if not self.issue_details:
@@ -517,7 +521,11 @@ def get_footer(self) -> SlackBlock:
517521
footer = (
518522
self.notification.build_notification_footer(self.recipient, ExternalProviders.SLACK)
519523
if self.notification and self.recipient
520-
else build_footer(self.group, project, self.rules, SLACK_URL_FORMAT)
524+
else build_slack_footer(
525+
group=self.group,
526+
project=project,
527+
rules=self.rules,
528+
)
521529
)
522530

523531
if not self.notification:

src/sentry/integrations/slack/message_builder/metric_alerts.py

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,9 @@
11
from sentry.incidents.models.alert_rule import AlertRule
22
from sentry.incidents.models.incident import Incident, IncidentStatus
3+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
34
from sentry.integrations.metric_alerts import metric_alert_unfurl_attachment_info
45
from sentry.integrations.slack.message_builder.base.block import BlockSlackMessageBuilder
5-
from sentry.integrations.slack.message_builder.types import (
6-
INCIDENT_COLOR_MAPPING,
7-
LEVEL_TO_COLOR,
8-
SlackBody,
9-
)
6+
from sentry.integrations.slack.message_builder.types import INCIDENT_COLOR_MAPPING, SlackBody
107
from sentry.integrations.slack.utils.escape import escape_slack_text
118

129

src/sentry/integrations/slack/message_builder/types.py

-11
Original file line numberDiff line numberDiff line change
@@ -7,17 +7,6 @@
77
SlackBlock = dict[str, Any]
88
SlackBody = Union[SlackAttachment, SlackBlock]
99

10-
# Attachment colors used for issues with no actions take.
11-
LEVEL_TO_COLOR = {
12-
"_actioned_issue": "#EDEEEF",
13-
"_incident_resolved": "#4DC771",
14-
"debug": "#FBE14F",
15-
"error": "#E03E2F",
16-
"fatal": "#FA4747",
17-
"info": "#2788CE",
18-
"warning": "#FFC227",
19-
}
20-
2110
INCIDENT_COLOR_MAPPING = {
2211
"Resolved": "_incident_resolved",
2312
"Warning": "warning",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
from collections.abc import Sequence
2+
3+
from sentry.integrations.messaging.message_builder import build_rule_url
4+
from sentry.integrations.slack.message_builder.types import SLACK_URL_FORMAT
5+
from sentry.models.group import Group
6+
from sentry.models.project import Project
7+
from sentry.models.rule import Rule
8+
9+
10+
def build_slack_footer(
11+
group: Group,
12+
project: Project,
13+
rules: Sequence[Rule] | None = None,
14+
) -> str:
15+
footer = f"{group.qualified_short_id}"
16+
17+
if rules:
18+
rule_url = build_rule_url(rules[0], group, project)
19+
# If this notification is triggered via the "Send Test Notification"
20+
# button then the label is not defined, but the url works.
21+
text = rules[0].label if rules[0].label else "Test Alert"
22+
footer = SLACK_URL_FORMAT.format(text=text, url=rule_url)
23+
24+
if len(rules) > 1:
25+
footer += f" (+{len(rules) - 1} other)"
26+
27+
return footer

src/sentry_plugins/slack/plugin.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
from sentry import tagstore
44
from sentry.integrations.base import FeatureDescription, IntegrationFeatures
5-
from sentry.integrations.slack.message_builder.types import LEVEL_TO_COLOR
5+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
66
from sentry.plugins.base.structs import Notification
77
from sentry.plugins.bases import notify
88
from sentry.shared_integrations.exceptions import ApiError

tests/sentry/integrations/discord/test_issue_alert.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -140,7 +140,7 @@ def test_basic(self, mock_record):
140140
notification_uuid=notification_uuid,
141141
),
142142
"color": LEVEL_TO_COLOR["error"],
143-
"footer": {"text": build_footer(self.event.group, self.event.project, None, "{text}")},
143+
"footer": {"text": build_footer(self.event.group, self.event.project, "{text}", None)},
144144
"fields": [],
145145
"timestamp": self.event.timestamp,
146146
}

tests/sentry/integrations/slack/test_message_builder.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
build_attachment_text,
2323
build_attachment_title,
2424
)
25+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
2526
from sentry.integrations.slack.message_builder.incidents import SlackIncidentsMessageBuilder
2627
from sentry.integrations.slack.message_builder.issues import (
2728
SlackIssuesMessageBuilder,
@@ -32,7 +33,6 @@
3233
get_tags,
3334
)
3435
from sentry.integrations.slack.message_builder.metric_alerts import SlackMetricAlertMessageBuilder
35-
from sentry.integrations.slack.message_builder.types import LEVEL_TO_COLOR
3636
from sentry.integrations.time_utils import time_since
3737
from sentry.issues.grouptype import (
3838
FeedbackGroup,

tests/sentry_plugins/slack/test_plugin.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
import pytest
66
import responses
77

8-
from sentry.integrations.slack.message_builder.types import LEVEL_TO_COLOR
8+
from sentry.integrations.messaging.types import LEVEL_TO_COLOR
99
from sentry.models.rule import Rule
1010
from sentry.plugins.base import Notification
1111
from sentry.shared_integrations.exceptions import ApiError

0 commit comments

Comments
 (0)