Created
October 7, 2020 00:02
-
-
Save busycalibrating/85e77a0f29104298726868ea6416df9b to your computer and use it in GitHub Desktop.
Dash plot.ly sending numpy with ZMQ pub/sub pyobj (Python 3)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# Largely a copy of the great gist by Danny Price (https://github.com/telegraphic) | |
# Minimal modifications to make the example work with Python 3, and to send | |
# python objects instead of using the msgpack library. | |
# | |
# Original gist can be found here: | |
# https://gist.github.com/telegraphic/2709b7e6edc3a0c39ed9b75452da205 | |
""" | |
# app.py - example Dash + ZMQ + msgpack + numpy monitor | |
This app receives data from zmq_pub.py, and plots it. | |
Run the app, browse to localhost:8085 in your web browser, and run zmq_pub.py | |
in a different terminal. | |
""" | |
import dash | |
import dash_core_components as dcc | |
import dash_html_components as html | |
from dash.dependencies import Input, Output | |
import numpy as np | |
import zmq | |
def create_zmq_socket(zmq_port: str="5556", topicfilter: str="data") -> zmq.Socket: | |
""" Create a ZMQ SUBSCRIBE socket """ | |
context = zmq.Context() | |
zmq_socket = context.socket(zmq.SUB) | |
zmq_socket.connect ("tcp://localhost:%s" % zmq_port) | |
zmq_socket.setsockopt_string(zmq.SUBSCRIBE, topicfilter) | |
return zmq_socket | |
def recv_zmq(topic: str='data') -> np.ndarray: | |
""" Receive data over ZMQ PubSub socket | |
Args: | |
topic: topic subscribed to | |
Returns numpy array data | |
""" | |
# Note - context managing socket as it can't be shared between threads | |
# This makes sure the socket is opened and closed by whatever thread Dash gives it | |
with create_zmq_socket() as socket: | |
# For pub/sub patterns, you must first send the topic as a string before | |
# you send a pyobj. | |
topic = socket.recv_string() | |
data = socket.recv_pyobj() | |
return data | |
external_stylesheets = ['https://codepen.io/chriddyp/pen/bWLwgP.css'] | |
app = dash.Dash(__name__, external_stylesheets=external_stylesheets) | |
app.layout = html.Div(children=[ | |
html.H1(children='Hello Dash'), | |
html.Div(children=''' | |
Dash: A web application framework for Python. | |
'''), | |
dcc.Graph( | |
id='example-graph', | |
figure={ | |
'data': [ | |
{'x': [1, 2, 3, 4, 5], 'y': [4, 1, 2, 1, 2], | |
'type': 'scatter', 'name': 'SF'}, | |
], | |
'layout': { | |
'title': 'Dash Data Visualization' | |
} | |
} | |
), | |
dcc.Interval( | |
id='interval-component', | |
interval=1*1000, # in milliseconds | |
n_intervals=0 | |
) | |
]) | |
# The updating is done with this callback | |
@app.callback( | |
Output('example-graph', 'figure'), | |
[Input('interval-component', 'n_intervals')]) | |
def update(n): | |
d = recv_zmq('data') | |
return {'data': [ | |
{'x': d['x'], 'y': d['y'], | |
'type': 'scatter', 'name': 'SF'}, | |
]} | |
if __name__ == '__main__': | |
app.run_server(debug=True) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# Largely a copy of the great gist by Danny Price (https://github.com/telegraphic) | |
# Minimal modifications to make the example work with Python 3.6, and to send | |
# python objects instead of using the msgpack library. | |
# | |
# Original gist can be found here: | |
# https://gist.github.com/telegraphic/2709b7e6edc3a0c39ed9b75452da205e | |
import zmq | |
import random | |
import sys | |
import time | |
import numpy as np | |
# Create ZMQ socket | |
port = "5556" | |
context = zmq.Context() | |
socket = context.socket(zmq.PUB) | |
socket.bind("tcp://*:%s" % port) | |
def send_msg(socket: zmq.Socket, d: np.ndarray, topic: str='data'): | |
""" Send data over ZMQ PubSub socket | |
Args: | |
socket: zmq.socket instance | |
d: np.ndarray containing the array to display | |
topic: topic to put in ZMQ topic field (str) | |
""" | |
# For pub/sub patterns, you must first send the topic as a string before | |
# you send a pyobj. | |
socket.send_string(topic, zmq.SNDMORE) | |
socket.send_pyobj(d) | |
return | |
while True: | |
topic = "data" | |
messagedata = {'x': np.arange(1024), 'y': np.random.randint(5, size=1024)} | |
print("%s %s" % (topic, messagedata)) | |
send_msg(socket, messagedata, topic) | |
time.sleep(1) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment