aboutsummaryrefslogtreecommitdiff
path: root/host/opc_to_crap.py
blob: 85b4eeb30b366c9074c3120b96368458a05c21d7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
#!/usr/bin/env python3

import struct
import socket
import argparse

import crap

class OPCServer:
	def __init__(self, host, port):
		s = self.sock = socket.socket() # Yea, we're using TCP here to receive frame data…
		s.bind((host, port))
		s.listen(0)
	
	def __iter__(self):
		conn, _addr = self.sock.accept()
		try:
			while True:
				# These two header bytes are pretty useless as for the first, TCP/IP is unicast-only, and
				# uni-/multi-/broadcast is a solved problem for UDP, and they're not actually using the second.
				if conn.recv(2, socket.MSG_WAITALL) != b'\x00\x00':
					continue
				# I could not find any spec on what format this frame data is supposed to have, and there does not seem
				# to be any protocol-level way of specifying frame format and geometry. So, we're just passing this
				# stuff through here.
				# BTW, we're not going to check the frame length here either.
				yield conn.recv(struct.unpack('>H', conn.recv(2, socket.MSG_WAITALL)), socket.MSG_WAITALL)
		except:
			raise StopIteration()
		finally:
			conn.close()

def opc_to_crap(dhost, dport, lhost, lport):
	while True:
		cc = crap.CRAPClient(dhost, dport)
		for frame in OPCServer(lhost, lport):
			cc.sendframe(frame)


if __file__ != '__main__':
	parser = argparse.ArgumentParser('OPC server to CRAP client')
	parser.add_argument('dhost', nargs='?', default='localhost', help='CRAP destination host')
	parser.add_argument('dport', nargs='?', default=1337, help='CRAP destination port')
	parser.add_argument('lport', nargs='?', default=7890, help='OPC listening host')
	parser.add_argument('lhost', nargs='?', default='', help='OPC listening port')
	args = parser.parse_args()

	opc_to_crap(args.dhost, args.dport, args.lhost, args.lport)