Slack bridge: Implement multiple channels bridges.

This commit is contained in:
rht 2021-11-28 00:56:56 -05:00 committed by Tim Abbott
parent 41ec1a9a29
commit eef02fbb76
2 changed files with 55 additions and 22 deletions

View file

@ -3,12 +3,20 @@ config = {
"email": "zulip-bot@email.com", "email": "zulip-bot@email.com",
"api_key": "put api key here", "api_key": "put api key here",
"site": "https://chat.zulip.org", "site": "https://chat.zulip.org",
"stream": "test here",
"topic": "<- slack-bridge",
}, },
"slack": { "slack": {
"username": "slack_username", "username": "slack_username",
"token": "xoxb-your-slack-token", "token": "xoxb-your-slack-token",
"channel": "C5Z5N7R8A -- must be channel id", },
# Mapping between Slack channels and Zulip stream-topic's.
# You can specify multiple pairs.
"channel_mapping": {
# Slack channel; must be channel ID
"C5Z5N7R8A": {
# Zulip stream
"stream": "test here",
# Zulip topic
"topic": "<- slack-bridge",
},
}, },
} }

View file

@ -5,7 +5,7 @@ import os
import sys import sys
import threading import threading
import traceback import traceback
from typing import Any, Callable, Dict from typing import Any, Callable, Dict, Optional, Tuple
import bridge_with_slack_config import bridge_with_slack_config
import slack_sdk import slack_sdk
@ -17,18 +17,28 @@ import zulip
ZULIP_MESSAGE_TEMPLATE = "**{username}**: {message}" ZULIP_MESSAGE_TEMPLATE = "**{username}**: {message}"
SLACK_MESSAGE_TEMPLATE = "<{username}> {message}" SLACK_MESSAGE_TEMPLATE = "<{username}> {message}"
StreamTopicT = Tuple[str, str]
def check_zulip_message_validity(msg: Dict[str, Any], config: Dict[str, Any]) -> bool:
def get_slack_channel_for_zulip_message(
msg: Dict[str, Any], zulip_to_slack_map: Dict[StreamTopicT, Any], bot_email: str
) -> Optional[str]:
is_a_stream = msg["type"] == "stream" is_a_stream = msg["type"] == "stream"
in_the_specified_stream = msg["display_recipient"] == config["stream"] if not is_a_stream:
at_the_specified_subject = msg["subject"] == config["topic"] return None
# We do this to identify the messages generated from Matrix -> Zulip stream_name = msg["display_recipient"]
# and we make sure we don't forward it again to the Matrix. topic_name = msg["subject"]
not_from_zulip_bot = msg["sender_email"] != config["email"] stream_topic: StreamTopicT = (stream_name, topic_name)
if is_a_stream and not_from_zulip_bot and in_the_specified_stream and at_the_specified_subject: if stream_topic not in zulip_to_slack_map:
return True return None
return False
# We do this to identify the messages generated from Slack -> Zulip
# and we make sure we don't forward it again to the Slack.
from_zulip_bot = msg["sender_email"] == bot_email
if from_zulip_bot:
return None
return zulip_to_slack_map[stream_topic]
class SlackBridge: class SlackBridge:
@ -37,14 +47,17 @@ class SlackBridge:
self.zulip_config = config["zulip"] self.zulip_config = config["zulip"]
self.slack_config = config["slack"] self.slack_config = config["slack"]
self.slack_to_zulip_map: Dict[str, Dict[str, str]] = config["channel_mapping"]
self.zulip_to_slack_map: Dict[StreamTopicT, str] = {
(z["stream"], z["topic"]): s for s, z in config["channel_mapping"].items()
}
# zulip-specific # zulip-specific
self.zulip_client = zulip.Client( self.zulip_client = zulip.Client(
email=self.zulip_config["email"], email=self.zulip_config["email"],
api_key=self.zulip_config["api_key"], api_key=self.zulip_config["api_key"],
site=self.zulip_config["site"], site=self.zulip_config["site"],
) )
self.zulip_stream = self.zulip_config["stream"]
self.zulip_subject = self.zulip_config["topic"]
# slack-specific # slack-specific
self.channel = self.slack_config["channel"] self.channel = self.slack_config["channel"]
@ -68,14 +81,16 @@ class SlackBridge:
def zulip_to_slack(self) -> Callable[[Dict[str, Any]], None]: def zulip_to_slack(self) -> Callable[[Dict[str, Any]], None]:
def _zulip_to_slack(msg: Dict[str, Any]) -> None: def _zulip_to_slack(msg: Dict[str, Any]) -> None:
message_valid = check_zulip_message_validity(msg, self.zulip_config) slack_channel = get_slack_channel_for_zulip_message(
if message_valid: msg, self.zulip_to_slack_map, self.zulip_config["email"]
)
if slack_channel is not None:
self.wrap_slack_mention_with_bracket(msg) self.wrap_slack_mention_with_bracket(msg)
slack_text = SLACK_MESSAGE_TEMPLATE.format( slack_text = SLACK_MESSAGE_TEMPLATE.format(
username=msg["sender_full_name"], message=msg["content"] username=msg["sender_full_name"], message=msg["content"]
) )
self.slack_webclient.chat_postMessage( self.slack_webclient.chat_postMessage(
channel=self.channel, channel=slack_channel,
text=slack_text, text=slack_text,
) )
@ -91,7 +106,7 @@ class SlackBridge:
@rtm.on("message") @rtm.on("message")
def slack_to_zulip(client: RTMClient, event: Dict[str, Any]) -> None: def slack_to_zulip(client: RTMClient, event: Dict[str, Any]) -> None:
if event["channel"] != self.channel: if event["channel"] not in self.slack_to_zulip_map:
return return
user_id = event["user"] user_id = event["user"]
user = self.slack_id_to_name[user_id] user = self.slack_id_to_name[user_id]
@ -100,8 +115,12 @@ class SlackBridge:
return return
self.replace_slack_id_with_name(event) self.replace_slack_id_with_name(event)
content = ZULIP_MESSAGE_TEMPLATE.format(username=user, message=event["text"]) content = ZULIP_MESSAGE_TEMPLATE.format(username=user, message=event["text"])
zulip_endpoint = self.slack_to_zulip_map[event["channel"]]
msg_data = dict( msg_data = dict(
type="stream", to=self.zulip_stream, subject=self.zulip_subject, content=content type="stream",
to=zulip_endpoint["stream"],
subject=zulip_endpoint["topic"],
content=content,
) )
self.zulip_client.send_message(msg_data) self.zulip_client.send_message(msg_data)
@ -118,11 +137,17 @@ if __name__ == "__main__":
sys.path.append(os.path.join(os.path.dirname(__file__), "..")) sys.path.append(os.path.join(os.path.dirname(__file__), ".."))
parser = argparse.ArgumentParser(usage=usage) parser = argparse.ArgumentParser(usage=usage)
config: Dict[str, Any] = bridge_with_slack_config.config
if "channel_mapping" not in config:
print(
'The key "channel_mapping" is not found in bridge_with_slack_config.py.\n'
"Your config file may be outdated."
)
exit(1)
print("Starting slack mirroring bot") print("Starting slack mirroring bot")
print("MAKE SURE THE BOT IS SUBSCRIBED TO THE RELEVANT ZULIP STREAM") print("MAKE SURE THE BOT IS SUBSCRIBED TO THE RELEVANT ZULIP STREAM")
config = bridge_with_slack_config.config
# We have to define rtm outside of SlackBridge because the rtm variable is used as a method decorator. # We have to define rtm outside of SlackBridge because the rtm variable is used as a method decorator.
rtm = RTMClient(token=config["slack"]["token"]) rtm = RTMClient(token=config["slack"]["token"])