platypush/platypush/plugins/kafka.py

77 lines
2.2 KiB
Python
Raw Normal View History

2018-07-28 01:55:15 +02:00
import json
import logging
from platypush.context import get_backend
from platypush.plugins import Plugin, action
class KafkaPlugin(Plugin):
"""
Plugin to send messages to an Apache Kafka instance (https://kafka.apache.org/)
Triggers:
* :class:`platypush.message.event.kafka.KafkaMessageEvent` when a new message is received on the consumer topic.
Requires:
* **kafka** (``pip install kafka-python``)
"""
2021-04-05 00:58:44 +02:00
def __init__(self, server=None, port=9092, **kwargs):
2018-07-28 01:55:15 +02:00
"""
2021-04-05 00:58:44 +02:00
:param server: Default Kafka server name or address. If None (default), then it has to be specified upon
message sending.
2018-07-28 01:55:15 +02:00
:type server: str
2021-04-05 00:58:44 +02:00
:param port: Default Kafka server port (default: 9092).
:type port: int
2018-07-28 01:55:15 +02:00
"""
super().__init__(**kwargs)
self.server = '{server}:{port}'.format(server=server, port=port) \
if server else None
self.producer = None
# Kafka can be veryyyy noisy
logging.getLogger('kafka').setLevel(logging.ERROR)
@action
2021-04-05 00:58:44 +02:00
def send_message(self, msg, topic, server=None):
2018-07-28 01:55:15 +02:00
"""
2021-04-05 00:58:44 +02:00
:param msg: Message to send - as a string, bytes stream, JSON, Platypush message, dictionary, or anything
that implements ``__str__``
2018-07-28 01:55:15 +02:00
2021-04-05 00:58:44 +02:00
:param topic: Topic to send the message to.
:type topic: str
:param server: Kafka server name or address + port (format: ``host:port``). If None, then the default server
will be used
2018-07-28 01:55:15 +02:00
:type server: str
"""
from kafka import KafkaProducer
2018-07-28 01:55:15 +02:00
if not server:
if not self.server:
try:
kafka_backend = get_backend('kafka')
server = kafka_backend.server
2021-04-05 00:58:44 +02:00
except Exception as e:
raise RuntimeError(f'No Kafka server nor default server specified: {str(e)}')
2018-07-28 01:55:15 +02:00
else:
server = self.server
if isinstance(msg, dict) or isinstance(msg, list):
msg = json.dumps(msg)
msg = str(msg).encode('utf-8')
producer = KafkaProducer(bootstrap_servers=server)
producer.send(topic, msg)
producer.flush()
# vim:sw=4:ts=4:et: