multi-user/replication.py

228 lines
6.6 KiB
Python
Raw Normal View History

2019-07-04 15:56:03 +02:00
import logging
from uuid import uuid4
try:
from .libs import umsgpack
except:
# Server import
from libs import umsgpack
import zmq
2019-07-18 16:38:13 +02:00
import pickle
from enum import Enum
2019-07-04 15:56:03 +02:00
logger = logging.getLogger(__name__)
logging.basicConfig(level=logging.DEBUG)
2019-07-18 16:38:13 +02:00
class RepState(Enum):
ADDED = 0
COMMITED = 1
STAGED = 2
2019-07-05 13:06:42 +02:00
class ReplicatedDataFactory(object):
"""
2019-07-18 16:38:13 +02:00
Manage the data types implementations
2019-07-05 13:06:42 +02:00
"""
2019-07-18 16:38:13 +02:00
2019-07-05 13:06:42 +02:00
def __init__(self):
self.supported_types = []
2019-07-18 16:38:13 +02:00
# Default registered types
self.register_type(str,RepCommand)
def register_type(self, dtype, implementation):
2019-07-05 13:06:42 +02:00
"""
Register a new replicated datatype implementation
"""
2019-07-05 18:47:40 +02:00
self.supported_types.append((dtype, implementation))
2019-07-05 13:06:42 +02:00
2019-07-18 16:38:13 +02:00
def match_type_by_instance(self, data):
"""
Find corresponding type to the given datablock
"""
2019-07-18 16:38:13 +02:00
for stypes, implementation in self.supported_types:
2019-07-05 13:06:42 +02:00
if isinstance(data, stypes):
return implementation
2019-07-18 16:38:13 +02:00
2019-07-05 13:06:42 +02:00
print("type not supported for replication")
raise NotImplementedError
2019-07-04 15:56:03 +02:00
2019-07-18 16:38:13 +02:00
def match_type_by_name(self, type_name):
for stypes, implementation in self.supported_types:
if type_name == implementation.__name__:
return implementation
2019-07-18 16:38:13 +02:00
print("type not supported for replication")
raise NotImplementedError
2019-07-18 16:38:13 +02:00
def construct_from_dcc(self, data):
implementation = self.match_type_by_instance(data)
2019-07-05 18:07:16 +02:00
return implementation
2019-07-04 18:24:12 +02:00
def construct_from_net(self, type_name):
"""
Reconstruct a new replicated value from serialized data
"""
return self.match_type_by_name(type_name)
2019-07-18 16:38:13 +02:00
2019-07-04 15:56:03 +02:00
class ReplicatedDatablock(object):
"""
Datablock used for replication
"""
2019-07-18 16:38:13 +02:00
uuid = None # uuid used as key (string)
pointer = None # dcc data ref (DCC type)
2019-07-18 18:15:01 +02:00
buffer = None # raw data (json)
2019-07-18 16:38:13 +02:00
str_type = None # data type name (string)
deps = [None] # dependencies array (string)
owner = None # Data owner (string)
state = None # Data state (RepState)
def __init__(self, owner=None, data=None, uuid=None, buffer=None):
self.uuid = uuid if uuid else str(uuid4())
2019-07-04 15:56:03 +02:00
assert(owner)
self.owner = owner
2019-07-18 18:15:01 +02:00
if data:
self.pointer = data
elif buffer:
self.buffer = self.deserialize(buffer)
else:
raise ValueError("Not enought parameter in constructor")
2019-07-18 16:38:13 +02:00
self.str_type = type(self).__name__
2019-07-04 15:56:03 +02:00
def push(self, socket):
"""
2019-07-04 18:24:12 +02:00
Here send data over the wire:
- serialize the data
- send them as a multipart frame
2019-07-04 15:56:03 +02:00
"""
2019-07-05 18:07:16 +02:00
data = self.serialize(self.pointer)
assert(isinstance(data, bytes))
owner = self.owner.encode()
2019-07-05 18:07:16 +02:00
key = self.uuid.encode()
type = self.str_type.encode()
2019-07-18 16:38:13 +02:00
socket.send_multipart([key, owner, type, data])
2019-07-04 15:56:03 +02:00
@classmethod
def pull(cls, socket, factory):
2019-07-04 15:56:03 +02:00
"""
2019-07-04 18:24:12 +02:00
Here we reeceive data from the wire:
- read data from the socket
- reconstruct an instance
2019-07-04 15:56:03 +02:00
"""
2019-07-18 16:38:13 +02:00
uuid, owner, str_type, data = socket.recv_multipart(zmq.NOBLOCK)
str_type = str_type.decode()
2019-07-18 16:38:13 +02:00
owner = owner.decode()
uuid = uuid.decode()
2019-07-18 16:38:13 +02:00
instance = factory.construct_from_net(str_type)(owner=owner, uuid=uuid, buffer=data)
# instance.data = instance.deserialize(data)
return instance
2019-07-18 16:38:13 +02:00
def store(self, dict, persistent=False):
2019-07-04 15:56:03 +02:00
"""
I want to store my replicated data. Persistent means into the disk
If uuid is none we delete the key from the volume
"""
if self.uuid is not None:
2019-07-18 16:38:13 +02:00
if self.buffer == 'None':
logger.debug("erasing key {}".format(self.uuid))
2019-07-04 15:56:03 +02:00
del dict[self.uuid]
else:
dict[self.uuid] = self
2019-07-18 16:38:13 +02:00
2019-07-07 12:22:45 +02:00
return self.uuid
2019-07-04 15:56:03 +02:00
2019-07-18 16:38:13 +02:00
def deserialize(self, data):
2019-07-04 15:56:03 +02:00
"""
2019-07-18 16:38:13 +02:00
BUFFER -> JSON
2019-07-04 15:56:03 +02:00
"""
raise NotImplementedError
2019-07-18 16:38:13 +02:00
def serialize(self, data):
2019-07-04 15:56:03 +02:00
"""
I want to load data from DCC
2019-07-05 18:07:16 +02:00
2019-07-18 16:38:13 +02:00
DCC -> JSON
2019-07-05 18:07:16 +02:00
MUST RETURN A BYTE ARRAY
2019-07-04 15:56:03 +02:00
"""
raise NotImplementedError
2019-07-18 16:38:13 +02:00
2019-07-04 15:56:03 +02:00
2019-07-18 16:38:13 +02:00
def apply(self,data,target):
"""
JSON -> DCC
"""
raise NotImplementedError
def resolve(self):
"""
I want to resolve my orphan data to an existing one
= Assing my pointer
"""
raise NotImplementedError
2019-07-05 13:06:42 +02:00
2019-07-18 16:38:13 +02:00
def dump(self):
return self.deserialize(self.buffer)
class RepCommand(ReplicatedDatablock):
def serialize(self,data):
return pickle.dumps(data)
def deserialize(self,data):
2019-07-18 18:15:01 +02:00
return pickle.loads(data)
2019-07-18 16:38:13 +02:00
def apply(self,data,target):
target = data
2019-07-05 13:06:42 +02:00
2019-07-05 18:07:16 +02:00
# class RepObject(ReplicatedDatablock):
# def deserialize(self):
# try:
# if self.pointer is None:
# pointer = None
2019-07-18 16:38:13 +02:00
2019-07-05 18:07:16 +02:00
# # Object specific constructor...
# if self.data["data"] in bpy.data.meshes.keys():
# pointer = bpy.data.meshes[self.data["data"]]
# elif self.data["data"] in bpy.data.lights.keys():
# pointer = bpy.data.lights[self.data["data"]]
# elif self.data["data"] in bpy.data.cameras.keys():
# pointer = bpy.data.cameras[self.data["data"]]
# elif self.data["data"] in bpy.data.curves.keys():
# pointer = bpy.data.curves[self.data["data"]]
# elif self.data["data"] in bpy.data.armatures.keys():
# pointer = bpy.data.armatures[self.data["data"]]
# elif self.data["data"] in bpy.data.grease_pencils.keys():
# pointer = bpy.data.grease_pencils[self.data["data"]]
# elif self.data["data"] in bpy.data.curves.keys():
# pointer = bpy.data.curves[self.data["data"]]
# self.pointer = bpy.data.objects.new(self.data["name"], pointer)
# self.pointer.matrix_world = mathutils.Matrix(self.data["matrix_world"])
# self.pointer.id = self.data['id']
# client = bpy.context.window_manager.session.username
# if self.pointer.id == client or self.pointer.id == "Common":
# self.pointer.hide_select = False
# else:
# self.pointer.hide_select = True
2019-07-18 16:38:13 +02:00
2019-07-05 18:07:16 +02:00
# except Exception as e:
# logger.error("Object {} loading error: {} ".format(self.data["name"], e))
2019-07-04 15:56:03 +02:00
2019-07-05 18:07:16 +02:00
# def deserialize(self):
# self.data = dump_datablock(self.pointer, 1)