2019-01-03 10:18:47 +01:00
|
|
|
#!/usr/bin/env python3
|
2020-12-31 18:50:11 +01:00
|
|
|
# Copyright (c) 2014-2020 The Bitcoin Core developers
|
2016-09-19 16:55:00 +02:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2014-11-18 18:06:32 +01:00
|
|
|
|
2019-01-03 10:18:47 +01:00
|
|
|
"""
|
|
|
|
ZMQ example using python3's asyncio
|
|
|
|
|
|
|
|
Dash should be started with the command line arguments:
|
2019-08-28 09:54:19 +02:00
|
|
|
dashd -testnet -daemon \
|
2019-01-03 10:18:47 +01:00
|
|
|
-zmqpubrawtx=tcp://127.0.0.1:28332 \
|
2017-11-30 01:17:24 +01:00
|
|
|
-zmqpubrawblock=tcp://127.0.0.1:28332 \
|
2019-01-03 10:18:47 +01:00
|
|
|
-zmqpubhashtx=tcp://127.0.0.1:28332 \
|
|
|
|
-zmqpubhashblock=tcp://127.0.0.1:28332
|
|
|
|
|
|
|
|
We use the asyncio library here. `self.handle()` installs itself as a
|
|
|
|
future at the end of the function. Since it never returns with the event
|
|
|
|
loop having an empty stack of futures, this creates an infinite loop. An
|
|
|
|
alternative is to wrap the contents of `handle` inside `while True`.
|
|
|
|
|
|
|
|
A blocking example using python 2.7 can be obtained from the git history:
|
|
|
|
https://github.com/bitcoin/bitcoin/blob/37a7fe9e440b83e2364d5498931253937abe9294/contrib/zmq/zmq_sub.py
|
|
|
|
"""
|
|
|
|
|
|
|
|
import asyncio
|
2014-11-18 18:06:32 +01:00
|
|
|
import zmq
|
2019-01-03 10:18:47 +01:00
|
|
|
import zmq.asyncio
|
|
|
|
import signal
|
2016-10-17 22:09:21 +02:00
|
|
|
import struct
|
2019-01-03 10:18:47 +01:00
|
|
|
import sys
|
|
|
|
|
2018-07-18 14:50:56 +02:00
|
|
|
if (sys.version_info.major, sys.version_info.minor) < (3, 5):
|
2019-01-03 10:18:47 +01:00
|
|
|
print("This example only works with Python 3.5 and greater")
|
2017-08-28 22:53:34 +02:00
|
|
|
sys.exit(1)
|
2014-11-18 18:06:32 +01:00
|
|
|
|
|
|
|
port = 28332
|
|
|
|
|
2019-01-03 10:18:47 +01:00
|
|
|
class ZMQHandler():
|
|
|
|
def __init__(self):
|
2018-03-06 19:02:57 +01:00
|
|
|
self.loop = asyncio.get_event_loop()
|
2019-01-03 10:18:47 +01:00
|
|
|
self.zmqContext = zmq.asyncio.Context()
|
|
|
|
|
|
|
|
self.zmqSubSocket = self.zmqContext.socket(zmq.SUB)
|
2021-09-08 18:39:06 +02:00
|
|
|
self.zmqSubSocket.setsockopt(zmq.RCVHWM, 0)
|
2019-01-03 10:18:47 +01:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashblock")
|
2019-05-08 11:12:54 +02:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashchainlock")
|
2019-01-03 10:18:47 +01:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashtx")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashtxlock")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashgovernancevote")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashgovernanceobject")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "hashinstantsenddoublespend")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawblock")
|
2019-05-08 11:12:54 +02:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawchainlock")
|
2019-05-23 11:13:58 +02:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawchainlocksig")
|
2019-01-03 10:18:47 +01:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawtx")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawtxlock")
|
2019-05-23 11:13:58 +02:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawtxlocksig")
|
2019-01-03 10:18:47 +01:00
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawgovernancevote")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawgovernanceobject")
|
|
|
|
self.zmqSubSocket.setsockopt_string(zmq.SUBSCRIBE, "rawinstantsenddoublespend")
|
|
|
|
self.zmqSubSocket.connect("tcp://127.0.0.1:%i" % port)
|
2014-11-18 18:06:32 +01:00
|
|
|
|
2019-01-03 10:18:47 +01:00
|
|
|
async def handle(self) :
|
|
|
|
msg = await self.zmqSubSocket.recv_multipart()
|
|
|
|
topic = msg[0]
|
2014-11-18 18:06:32 +01:00
|
|
|
body = msg[1]
|
2017-01-23 13:22:32 +01:00
|
|
|
sequence = "Unknown"
|
2016-10-17 22:09:21 +02:00
|
|
|
|
|
|
|
if len(msg[-1]) == 4:
|
|
|
|
msgSequence = struct.unpack('<I', msg[-1])[-1]
|
|
|
|
sequence = str(msgSequence)
|
2014-11-18 18:06:32 +01:00
|
|
|
|
2019-01-03 10:18:47 +01:00
|
|
|
if topic == b"hashblock":
|
2016-10-17 22:09:21 +02:00
|
|
|
print('- HASH BLOCK ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"hashchainlock":
|
|
|
|
print('- HASH CHAINLOCK ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"hashtx":
|
2016-10-17 22:09:21 +02:00
|
|
|
print ('- HASH TX ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"hashtxlock":
|
2016-10-17 22:09:21 +02:00
|
|
|
print('- HASH TX LOCK ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"hashgovernancevote":
|
|
|
|
print('- HASH GOVERNANCE VOTE ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"hashgovernanceobject":
|
|
|
|
print('- HASH GOVERNANCE OBJECT ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"hashinstantsenddoublespend":
|
|
|
|
print('- HASH IS DOUBLE SPEND ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"rawblock":
|
2016-10-17 22:09:21 +02:00
|
|
|
print('- RAW BLOCK HEADER ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body[:80].hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"rawchainlock":
|
|
|
|
print('- RAW CHAINLOCK ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body[:80].hex())
|
2019-05-23 11:13:58 +02:00
|
|
|
elif topic == b"rawchainlocksig":
|
|
|
|
print('- RAW CHAINLOCK SIG ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body[:80].hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"rawtx":
|
2016-10-17 22:09:21 +02:00
|
|
|
print('- RAW TX ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"rawtxlock":
|
2016-10-17 22:09:21 +02:00
|
|
|
print('- RAW TX LOCK ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-23 11:13:58 +02:00
|
|
|
elif topic == b"rawtxlocksig":
|
|
|
|
print('- RAW TX LOCK SIG ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"rawgovernancevote":
|
2018-07-12 11:06:30 +02:00
|
|
|
print('- RAW GOVERNANCE VOTE ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
elif topic == b"rawgovernanceobject":
|
2018-07-12 11:06:30 +02:00
|
|
|
print('- RAW GOVERNANCE OBJECT ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-05-08 11:12:54 +02:00
|
|
|
elif topic == b"rawinstantsenddoublespend":
|
|
|
|
print('- RAW IS DOUBLE SPEND ('+sequence+') -')
|
2021-08-27 21:03:02 +02:00
|
|
|
print(body.hex())
|
2019-01-03 10:18:47 +01:00
|
|
|
# schedule ourselves to receive the next message
|
|
|
|
asyncio.ensure_future(self.handle())
|
|
|
|
|
|
|
|
def start(self):
|
|
|
|
self.loop.add_signal_handler(signal.SIGINT, self.stop)
|
|
|
|
self.loop.create_task(self.handle())
|
|
|
|
self.loop.run_forever()
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
self.loop.stop()
|
|
|
|
self.zmqContext.destroy()
|
2014-11-18 18:06:32 +01:00
|
|
|
|
2019-01-03 10:18:47 +01:00
|
|
|
daemon = ZMQHandler()
|
|
|
|
daemon.start()
|