import binascii import collections import copy import logging import os import queue import sys import threading import time from enum import Enum from random import randint from . import draw, helpers, message from .libs import dump_anything, umsgpack, zmq # import zmq lock = threading.Lock() logger = logging.getLogger(__name__) logging.basicConfig(level=logging.INFO) CONNECT_TIMEOUT = 2 WAITING_TIME = 0.001 SERVER_MAX = 1 stop = False DUMP_AGENTS_NUMBER = 1 class State(Enum): INITIAL = 1 SYNCING = 2 ACTIVE = 3 def zpipe(ctx): """build inproc pipe for talking to threads mimic pipe used in czmq zthread_fork. Returns a pair of PAIRs connected via inproc """ a = ctx.socket(zmq.PAIR) b = ctx.socket(zmq.PAIR) a.linger = b.linger = 0 a.hwm = b.hwm = 1 iface = "inproc://%s" % binascii.hexlify(os.urandom(8)) a.bind(iface) b.connect(iface) return a, b class RCFClient(object): ctx = None pipe = None net_agent = None store = None def __init__(self): self.ctx = zmq.Context() self.pipe, peer = zpipe(self.ctx) self.store = {} self.serial_product = queue.Queue() self.serial_feed = queue.Queue() self.stop_event= threading.Event() # Database and connexion agent self.net_agent = threading.Thread( target=rcf_client_worker, args=(self.ctx, self.store, peer, self.serial_product,self.serial_feed, self.stop_event), name="net-agent") self.net_agent.daemon = True self.net_agent.start() # Local data translation agent self.serial_agents = [] for a in range(0, DUMP_AGENTS_NUMBER): serial_agent = threading.Thread( target=serial_worker, args=(self.serial_product, self.serial_feed), name="serial-agent") serial_agent.daemon = True serial_agent.start() self.serial_agents.append(serial_agent) # Database and connexion agent self.watchdog_agent = threading.Thread( target=watchdog_worker, args=(self.serial_feed, .5, self.stop_event), name="watchdog-agent") self.watchdog_agent.daemon = True self.watchdog_agent.start() def connect(self, id, address, port): self.pipe.send_multipart([b"CONNECT", (id.encode() if isinstance( id, str) else id), (address.encode() if isinstance( address, str) else address), b'%d' % port]) def init(self): """ Scene initialisation """ self.pipe.send_multipart( [b"INIT"]) def disconnect(self): """ Disconnect """ self.pipe.send_multipart( [b"DISCONNECT"]) def set(self, key, value=None, override=False): """Set new value in distributed hash table Sends [SET][key][value] to the agent """ if value: pass self.pipe.send_multipart( [b"SET", umsgpack.packb(key), (umsgpack.packb(value) if value else umsgpack.packb('None')),umsgpack.packb(override)]) else: self.serial_feed.put(('DUMP',key,None)) # self.serial_pipe.send_multipart( # [b"DUMP", umsgpack.packb(key)]) # self.pipe.send_multipart( # [b"SET", umsgpack.packb(key), (umsgpack.packb(value) if value else umsgpack.packb('None')),umsgpack.packb(override)]) def add(self, key, value=None): """Set new value in distributed hash table Sends [SET][key][value] to the agent """ self.serial_feed.put(key) # self.pipe.send_multipart( # [b"ADD", umsgpack.packb(key), (umsgpack.packb(value) if value else umsgpack.packb('None'))]) def is_busy(self): if self.serial_feed.qsize() == 0 and self.serial_product.qsize() == 0: return False else: return True def exit(self): if self.net_agent.is_alive(): self.disconnect() self.stop_event.set() for a in range(0,DUMP_AGENTS_NUMBER): self.serial_feed.put(('STOP',None,None)) # READ-ONLY FUNCTIONS def get(self, key): """Lookup value in distributed hash table Sends [GET][key] to the agent and waits for a value response If there is no clone available, will eventually return None. """ value = [] for k in self.store.keys(): if key in k: value.append([k, self.store.get(k).body]) return value def list(self): dump_list = [] for k,v in self.store.items(): if 'Client' in k: dump_list.append([k,v.id.decode()]) else: try: dump_list.append([k,v.body['id']]) except: pass return dump_list def state(self): if not self.is_busy(): self.pipe.send_multipart([b"STATE"]) try: reply = self.pipe.recv_multipart() except KeyboardInterrupt: return else: return umsgpack.unpackb(reply[0]) else: return None class RCFServer(object): address = None # Server address port = None # Server port snapshot = None # Snapshot socket subscriber = None # Incoming updates def __init__(self, ctx, address, port, id): self.address = address self.port = port self.snapshot = ctx.socket(zmq.DEALER) self.snapshot.linger = 0 self.snapshot.setsockopt(zmq.IDENTITY, id) self.snapshot.connect("tcp://{}:{}".format(address.decode(), port)) self.subscriber = ctx.socket(zmq.SUB) self.subscriber.setsockopt_string(zmq.SUBSCRIBE, '') self.subscriber.connect("tcp://{}:{}".format(address.decode(), port+1)) self.subscriber.linger = 0 print("connected on tcp://{}:{}".format(address.decode(), port)) class RCFClientAgent(object): ctx = None pipe = None property_map = None publisher = None id = None state = State.INITIAL server = None serial = None serialisation_agent = None def __init__(self, ctx, store, pipe): self.ctx = ctx self.pipe = pipe self.property_map = store self.id = b"test" self.state = State.INITIAL self.admin = False self.server = None self.publisher = self.ctx.socket(zmq.PUSH) # push update socket self.publisher.setsockopt(zmq.IDENTITY, self.id) self.publisher.setsockopt(zmq.SNDHWM, 60) self.publisher.linger = 0 # self.serial_agent = threading.Thread( # target=serialization_agent, args=(self.ctx, peer), name="serial-agent") # self.serial_agent.daemon = True # self.serial_agent.start() def _add(self,key, value='None'): if value == 'None': # try to dump from bpy # logging.info(key) value = helpers.dump(key) value['id'] = self.id.decode() if value: rcfmsg = message.RCFMessage( key=key, id=self.id, body=value) rcfmsg.store(self.property_map) rcfmsg.send(self.publisher) else: logger.error("Fail to dump ") def control_message(self): msg = self.pipe.recv_multipart() command = msg.pop(0) if command == b"CONNECT": self.id = msg.pop(0) address = msg.pop(0) port = int(msg.pop(0)) if self.server is None: if address == '127.0.0.1' or address == 'localhost' : self.admin = True self.server = RCFServer(self.ctx, address, port, self.id) self.publisher.connect( "tcp://{}:{}".format(address.decode(), port+2)) else: logger.error("E: too many servers (max. %i)", SERVER_MAX) elif command == b"DISCONNECT": if self.admin is False: uid = self.id.decode() delete_user = message.RCFMessage( key="Client/{}".format(uid), id=self.id, body=None) delete_user.send(self.publisher) # TODO: Do we need to pass every object rights to the moderator ? # for k,v in self.property_map.items(): # if v.body["id"] == uid: # delete_msg = message.RCFMessage( # key=k, id=self.id, body=None) # # delete_msg.store(self.property_map) # delete_msg.send(self.publisher) elif command == b"SET": key = umsgpack.unpackb(msg[0]) value = umsgpack.unpackb(msg[1]) override = umsgpack.unpackb(msg[2]) if key in self.property_map.keys(): if self.property_map[key].body['id'] == self.id.decode() or override: if value == 'None': value = helpers.dump(key) value['id'] = self.id.decode() if value: key_id = self.id # if override: # key_id = value['id'].encode() rcfmsg = message.RCFMessage( key=key, id=key_id, body=value) rcfmsg.store(self.property_map) if override: helpers.load(key,self.property_map[key].body) rcfmsg.send(self.publisher) else: logger.error("Fail to dump ") else: helpers.load(key,self.property_map[key].body) elif command == b"INIT": d = helpers.get_all_datablocks() for i in d: self._add(i) elif command == b"ADD": key = umsgpack.unpackb(msg[0]) value = umsgpack.unpackb(msg[1]) if value == 'None': # try to dump from bpy # logging.info(key) value = helpers.dump(key) value['id'] = self.id.decode() if value: rcfmsg = message.RCFMessage( key=key, id=self.id, body=value) rcfmsg.store(self.property_map) rcfmsg.send(self.publisher) else: logger.error("Fail to dump ") elif command == b"GET": value = [] key = umsgpack.unpackb(msg[0]) for k in self.property_map.keys(): if key in k: value.append([k, self.property_map.get(k).body]) # value = [self.property_map.get(key) for key in keys] # value = self.property_map.get(key) self.pipe.send(umsgpack.packb(value) if value else umsgpack.packb('')) elif command == b"LIST": dump_list = [] for k,v in self.property_map.items(): if 'Client' in k: dump_list.append([k,v.id.decode()]) else: try: dump_list.append([k,v.body['id']]) except: pass self.pipe.send(umsgpack.packb(dump_list) if dump_list else umsgpack.packb('')) elif command == b"STATE": self.pipe.send(umsgpack.packb(self.state.value)) def rcf_client_worker(ctx,store, pipe, serial_product, serial_feed, stop_event): agent = RCFClientAgent(ctx,store, pipe) server = None net_feed = serial_product net_product = serial_feed while not stop_event.is_set(): # logger.info("asdasd") poller = zmq.Poller() poller.register(agent.pipe, zmq.POLLIN) server_socket = None if agent.state == State.INITIAL: server = agent.server if agent.server: logger.info("%s: waiting for server at %s:%d...", agent.id.decode(), server.address, server.port) server.snapshot.send(b"SNAPSHOT_REQUEST") agent.state = State.SYNCING server_socket = server.snapshot elif agent.state == State.SYNCING: server_socket = server.snapshot elif agent.state == State.ACTIVE: server_socket = server.subscriber if server_socket: poller.register(server_socket, zmq.POLLIN) try: items = dict(poller.poll(1)) except: raise break if agent.pipe in items: agent.control_message() elif server_socket in items: rcfmsg = message.RCFMessage.recv(server_socket) if agent.state == State.SYNCING: # Store snapshot if rcfmsg.key == "SNAPSHOT_END": client_key = "Client/{}".format(agent.id.decode()) client_dict = {} with lock: client_dict = helpers.init_client(key=client_key) client_dict['id'] = agent.id.decode() client_store = message.RCFMessage( key=client_key, id=agent.id, body=client_dict) logger.info(client_store) client_store.store(agent.property_map) client_store.send(agent.publisher) logger.info("snapshot complete") agent.state = State.ACTIVE else: net_product.put(('LOAD',rcfmsg.key, rcfmsg.body)) # helpers.load(rcfmsg.key, rcfmsg.body) rcfmsg.store(agent.property_map) logger.info("snapshot from {} stored".format(rcfmsg.id)) elif agent.state == State.ACTIVE: # IN if rcfmsg.id != agent.id: # with lock: # helpers.load(rcfmsg.key, rcfmsg.body) net_product.put(('LOAD',rcfmsg.key, rcfmsg.body)) rcfmsg.store(agent.property_map) else: logger.debug("{} nothing to do".format(agent.id)) # Serialisation thread input if not net_feed.empty(): key, value = net_feed.get() if value: # Stamp with id value['id'] = agent.id.decode() # Format massage rcfmsg = message.RCFMessage( key=key, id=agent.id, body=value) rcfmsg.store(agent.property_map) rcfmsg.send(agent.publisher) else: logger.error("Fail to dump ") logger.info("exit thread") stop = False # else: else # agent.state = State.INITIAL def serial_worker(product, feed): logger.info("serial thread launched") while True: command,key,value = feed.get() if command == 'STOP': break elif command == 'DUMP': value = helpers.dump(key) if value: product.put((key,value)) elif command == 'LOAD': if value: helpers.load(key, value) logger.info("serial thread stopped") def watchdog_worker(feed,interval, stop_event): import bpy logger.info("watchdog thread launched with {} sec of interval".format(interval)) while not stop_event.is_set(): for datatype in helpers.SUPPORTED_TYPES: for item in getattr(bpy.data, helpers.CORRESPONDANCE[datatype]): if item.id == 'None': item.id = bpy.context.scene.session_settings.username key = "{}/{}".format(datatype, item.name) feed.put(('DUMP',key,None)) time.sleep(interval) logger.info("watchdog thread stopped")