# What this PR does It's part of work on https://github.com/grafana/oncall-gateway/issues/247. I added stack_slug to call to the chatops-proxy tenant/register API. On a side note I figured out that we didn't cleanup slack integration in chatops-proxy, once it's uninstalled on OnCall side, so it's [fixed](https://github.com/grafana/oncall/pull/4559/files#diff-1784f1d0d65fa477f4562e73aa23fe1c757b171f36e03f12600bdb021f121307R577) as well. Changes are validated locally.
164 lines
5.7 KiB
Python
164 lines
5.7 KiB
Python
"""
|
|
Set of utils to handle oncall and chatops-proxy interaction.
|
|
"""
|
|
import logging
|
|
import typing
|
|
|
|
from django.conf import settings
|
|
|
|
from .client import PROVIDER_TYPE_SLACK, SERVICE_TYPE_ONCALL, ChatopsProxyAPIClient, ChatopsProxyAPIException
|
|
from .tasks import (
|
|
link_slack_team_async,
|
|
register_oncall_tenant_async,
|
|
unlink_slack_team_async,
|
|
unregister_oncall_tenant_async,
|
|
)
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def get_installation_link_from_chatops_proxy(user) -> typing.Optional[str]:
|
|
"""
|
|
get_installation_link_from_chatops_proxy fetches slack installation link from chatops proxy.
|
|
If there is no existing slack installation - if returns link, If slack already installed, it returns None.
|
|
"""
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
org = user.organization
|
|
try:
|
|
link, _ = client.get_slack_oauth_link(
|
|
org.stack_id,
|
|
user.user_id,
|
|
org.web_link,
|
|
SERVICE_TYPE_ONCALL,
|
|
)
|
|
return link
|
|
except ChatopsProxyAPIException as api_exc:
|
|
if api_exc.status == 409:
|
|
return None
|
|
logger.exception(
|
|
"Error while getting installation link from chatops proxy: " "error=%s",
|
|
api_exc,
|
|
)
|
|
raise api_exc
|
|
|
|
|
|
def get_slack_oauth_response_from_chatops_proxy(stack_id) -> dict:
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
slack_installation, _ = client.get_oauth_installation(stack_id, PROVIDER_TYPE_SLACK)
|
|
return slack_installation.oauth_response
|
|
|
|
|
|
def register_oncall_tenant(service_tenant_id: str, cluster_slug: str, stack_id: int, stack_slug: str):
|
|
"""
|
|
register_oncall_tenant tries to register oncall tenant synchronously and fall back to task in case of any exceptions
|
|
to make sure that tenant is registered.
|
|
First attempt is synchronous to register tenant ASAP to not miss any chatops requests.
|
|
"""
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
try:
|
|
client.register_tenant(
|
|
service_tenant_id,
|
|
cluster_slug,
|
|
SERVICE_TYPE_ONCALL,
|
|
stack_id,
|
|
stack_slug,
|
|
)
|
|
except Exception as e:
|
|
logger.error(
|
|
f"create_oncall_connector: failed "
|
|
f"oncall_org_id={service_tenant_id} backend={cluster_slug} stack_id={stack_id} exc={e}"
|
|
)
|
|
register_oncall_tenant_async.apply_async(
|
|
kwargs={
|
|
"service_tenant_id": service_tenant_id,
|
|
"cluster_slug": cluster_slug,
|
|
"service_type": SERVICE_TYPE_ONCALL,
|
|
"stack_id": stack_id,
|
|
},
|
|
countdown=2,
|
|
)
|
|
|
|
|
|
def unregister_oncall_tenant(service_tenant_id: str, cluster_slug: str):
|
|
"""
|
|
unregister_oncall_tenant unregisters tenant asynchronously.
|
|
"""
|
|
unregister_oncall_tenant_async.apply_async(
|
|
kwargs={
|
|
"service_tenant_id": service_tenant_id,
|
|
"cluster_slug": cluster_slug,
|
|
"service_type": SERVICE_TYPE_ONCALL,
|
|
},
|
|
countdown=2,
|
|
)
|
|
|
|
|
|
def can_link_slack_team(
|
|
service_tenant_id: str,
|
|
slack_team_id: str,
|
|
cluster_slug: str,
|
|
) -> bool:
|
|
"""
|
|
can_link_slack_team checks if it's possible to link slack workspace to oncall tenant located in cluster.
|
|
All oncall tenants linked to same slack team should have same cluster.
|
|
"""
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
try:
|
|
response = client.can_slack_link(service_tenant_id, cluster_slug, slack_team_id, SERVICE_TYPE_ONCALL)
|
|
return response.status_code == 200
|
|
except Exception as e:
|
|
logger.error(
|
|
f"can_link_slack_team: slack installation impossible: {e} "
|
|
f"service_tenant_id={service_tenant_id} slack_team_id={slack_team_id} cluster_slug={cluster_slug}"
|
|
)
|
|
|
|
return False
|
|
|
|
|
|
def link_slack_team(service_tenant_id: str, slack_team_id: str):
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
try:
|
|
client.link_slack_team(service_tenant_id, slack_team_id, SERVICE_TYPE_ONCALL)
|
|
except Exception as e:
|
|
logger.error(
|
|
f'msg="Failed to link slack team: {e}"'
|
|
f"service_tenant_id={service_tenant_id} slack_team_id={slack_team_id}"
|
|
)
|
|
link_slack_team_async.apply_async(
|
|
kwargs={
|
|
"service_tenant_id": service_tenant_id,
|
|
"slack_team_id": slack_team_id,
|
|
"service_type": SERVICE_TYPE_ONCALL,
|
|
},
|
|
countdown=2,
|
|
)
|
|
|
|
|
|
def unlink_slack_team(service_tenant_id: str, slack_team_id: str):
|
|
unlink_slack_team_async.apply_async(
|
|
kwargs={
|
|
"service_tenant_id": service_tenant_id,
|
|
"slack_team_id": slack_team_id,
|
|
"service_type": SERVICE_TYPE_ONCALL,
|
|
}
|
|
)
|
|
|
|
|
|
def uninstall_slack(stack_id: int, grafana_user_id: int) -> bool:
|
|
"""
|
|
uninstall_slack uninstalls slack integration from chatops-proxy and returns bool indicating if it was removed.
|
|
If such installation does not exist - returns True as well.s
|
|
"""
|
|
client = ChatopsProxyAPIClient(settings.ONCALL_GATEWAY_URL, settings.ONCALL_GATEWAY_API_TOKEN)
|
|
try:
|
|
removed, response = client.delete_oauth_installation(stack_id, PROVIDER_TYPE_SLACK, grafana_user_id)
|
|
except ChatopsProxyAPIException as api_exc:
|
|
if api_exc.status == 404:
|
|
return True
|
|
logger.exception(
|
|
"uninstall_slack: error trying to install slack from chatops-proxy: " "error=%s",
|
|
api_exc,
|
|
)
|
|
return False
|
|
|
|
return removed is True
|