2019-07-04 15:56:03 +02:00
|
|
|
import logging
|
|
|
|
from uuid import uuid4
|
2019-07-22 17:42:38 +02:00
|
|
|
import json
|
2019-07-04 15:56:03 +02:00
|
|
|
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)
|
2019-07-23 11:34:43 +02:00
|
|
|
self.register_type(RepDeleteCommand, RepDeleteCommand)
|
|
|
|
|
2019-07-18 16:38:13 +02:00
|
|
|
|
|
|
|
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):
|
2019-07-17 17:59:42 +02:00
|
|
|
"""
|
|
|
|
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__:
|
2019-07-17 18:26:30 +02:00
|
|
|
return implementation
|
2019-07-18 16:38:13 +02:00
|
|
|
print("type not supported for replication")
|
|
|
|
raise NotImplementedError
|
2019-07-17 18:26:30 +02:00
|
|
|
|
2019-07-18 16:38:13 +02:00
|
|
|
def construct_from_dcc(self, data):
|
2019-07-17 17:59:42 +02:00
|
|
|
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
|
|
|
|
2019-07-17 17:59:42 +02:00
|
|
|
def construct_from_net(self, type_name):
|
|
|
|
"""
|
|
|
|
Reconstruct a new replicated value from serialized data
|
|
|
|
"""
|
2019-07-18 11:14:43 +02:00
|
|
|
return self.match_type_by_name(type_name)
|
2019-07-17 17:59:42 +02:00
|
|
|
|
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)
|
|
|
|
|
2019-07-22 17:42:38 +02:00
|
|
|
def __init__(self, owner=None, pointer=None, uuid=None, buffer=None):
|
2019-07-17 17:59:42 +02:00
|
|
|
self.uuid = uuid if uuid else str(uuid4())
|
2019-07-04 15:56:03 +02:00
|
|
|
assert(owner)
|
2019-07-17 17:59:42 +02:00
|
|
|
self.owner = owner
|
2019-07-18 18:15:01 +02:00
|
|
|
|
2019-07-22 17:42:38 +02:00
|
|
|
if pointer:
|
|
|
|
self.pointer = pointer
|
|
|
|
self.buffer = self.dump()
|
2019-07-18 18:15:01 +02:00
|
|
|
elif buffer:
|
2019-07-22 17:42:38 +02:00
|
|
|
self.buffer = buffer
|
2019-07-19 14:29:57 +02:00
|
|
|
|
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
|
2019-07-22 17:42:38 +02:00
|
|
|
- send them as a multipart frame thought the given socket
|
2019-07-04 15:56:03 +02:00
|
|
|
"""
|
2019-07-22 17:42:38 +02:00
|
|
|
assert(self.buffer)
|
|
|
|
|
|
|
|
data = self.serialize(self.buffer)
|
2019-07-05 18:07:16 +02:00
|
|
|
assert(isinstance(data, bytes))
|
2019-07-17 17:59:42 +02:00
|
|
|
owner = self.owner.encode()
|
2019-07-05 18:07:16 +02:00
|
|
|
key = self.uuid.encode()
|
2019-07-17 18:26:30 +02:00
|
|
|
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
|
2019-07-17 17:59:42 +02:00
|
|
|
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)
|
2019-07-17 17:59:42 +02:00
|
|
|
|
2019-07-18 11:14:43 +02:00
|
|
|
str_type = str_type.decode()
|
2019-07-18 16:38:13 +02:00
|
|
|
owner = owner.decode()
|
|
|
|
uuid = uuid.decode()
|
2019-07-18 11:14:43 +02:00
|
|
|
|
2019-07-22 17:42:38 +02:00
|
|
|
instance = factory.construct_from_net(str_type)(owner=owner, uuid=uuid)
|
|
|
|
instance.buffer = instance.deserialize(data)
|
|
|
|
|
2019-07-17 17:59:42 +02:00
|
|
|
return instance
|
|
|
|
|
2019-07-22 17:42:38 +02:00
|
|
|
|
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-22 17:42:38 +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
|
|
|
"""
|
2019-07-22 17:42:38 +02:00
|
|
|
JSON -> BUFFER
|
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-22 17:42:38 +02:00
|
|
|
def dump(self):
|
|
|
|
"""
|
|
|
|
DCC -> JSON
|
|
|
|
"""
|
|
|
|
assert(self.pointer)
|
|
|
|
|
|
|
|
return json.dumps(self.pointer)
|
|
|
|
|
|
|
|
|
|
|
|
def load(self,target=None):
|
2019-07-18 16:38:13 +02:00
|
|
|
"""
|
|
|
|
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-23 11:34:43 +02:00
|
|
|
def __repr__(self):
|
2019-07-23 11:41:36 +02:00
|
|
|
return "{uuid} - owner: {owner} - type: {type}".format(
|
2019-07-23 11:34:43 +02:00
|
|
|
uuid=self.uuid,
|
|
|
|
owner=self.owner,
|
2019-07-23 11:41:36 +02:00
|
|
|
type=self.str_type
|
2019-07-23 11:34:43 +02:00
|
|
|
)
|
2019-07-18 16:38:13 +02:00
|
|
|
|
|
|
|
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
|
|
|
|
2019-07-22 17:42:38 +02:00
|
|
|
def load(self,target):
|
2019-07-19 14:29:57 +02:00
|
|
|
target = self.pointer
|
2019-07-05 13:06:42 +02:00
|
|
|
|
2019-07-23 11:34:43 +02:00
|
|
|
class RepDeleteCommand(ReplicatedDatablock):
|
|
|
|
def serialize(self,data):
|
|
|
|
return pickle.dumps(data)
|
|
|
|
|
|
|
|
def deserialize(self,data):
|
|
|
|
return pickle.loads(data)
|
|
|
|
|
|
|
|
def store(self,rep_store):
|
|
|
|
assert(self.buffer)
|
|
|
|
|
|
|
|
if rep_store and self.buffer in rep_store.keys():
|
|
|
|
del rep_store[self.buffer]
|
|
|
|
|
|
|
|
|
|
|
|
|
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)
|