# -*- coding: utf-8 -*- ############################################################################## # # Copyright (c) 2009 Nexedi SA and Contributors. All Rights Reserved. # Danièle Vanbaelinghem <daniele@nexedi.com> # # WARNING: This program as such is intended to be used by professional # programmers who take the whole responsability of assessing all potential # consequences resulting from its eventual inadequacies and bugs # End users who are looking for a ready-to-use solution with commercial # garantees and support are strongly adviced to contract a Free Software # Service Company # # This program is Free Software; you can redistribute it and/or # modify it under the terms of the GNU General Public License # as published by the Free Software Foundation; either version 2 # of the License, or (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. # ############################################################################## from Products.ERP5Type.Globals import PersistentMapping from time import gmtime,strftime # for anchors from SyncCode import SyncCode from AccessControl import ClassSecurityInfo from Products.CMFCore.utils import getToolByName from Acquisition import Implicit, aq_base from Products.ERP5Type.Accessor.Constant import PropertyGetter as ConstantGetter from Products.ERP5Type.Core.Folder import Folder from Products.ERP5Type.Base import Base from Products.ERP5Type import Permissions from Products.ERP5Type import PropertySheet from Products.ERP5.Document import Document from DateTime import DateTime from zLOG import LOG, DEBUG, INFO import cStringIO from OFS.Image import Pdata from OFS.Image import File import md5 from base64 import b64encode, b64decode, b16encode, b16decode class Signature(Folder, SyncCode, File): """ status -- SENT, CONFLICT... md5_object -- An MD5 value of a given document #uid -- The UID of the document id -- the ID of the document gid -- the global id of the document rid -- the uid of the document on the remote database, only needed on the server. xml -- the xml of the object at the time where it was synchronized """ isIndexable = ConstantGetter('isIndexable', value=False) # Make sure RAD generated accessors at the class level # Constructor def __init__(self, id=None, rid=None, status=None, xml_string=None, object=None): Folder.__init__(self, id) File.__init__(self, id, '', '') if object is not None: self.setPath(object.getPhysicalPath()) self.setObjectId(object.getId()) else: self.setPath(None) self.setId(id) self.setRid(rid) self.status = status self.setXML(xml_string) self.setPartialXML(None) self.action = None self.setTempXML(None) self.resetConflictList() self.md5_string = None self.force = 0 self.setSubscriberXupdate(None) self.setPublisherXupdate(None) self.last_data_partial_xml = None def setStatus(self, status): """ set the Status (see SyncCode for numbers) """ self.status = status if status == self.SYNCHRONIZED: temp_xml = self.getTempXML() self.setForce(0) if temp_xml is not None: # This happens when we have sent the xml # and we just get the confirmation self.setXML(temp_xml) self.setTempXML(None) self.setPartialXML(None) self.setSubscriberXupdate(None) self.setPublisherXupdate(None) if len(self.getConflictList())>0: self.resetConflictList() # XXX This may be a problem, if the document is changed # during a synchronization self.setLastSynchronizationDate(DateTime()) self.getParentValue().removeRemainingObjectPath(self.getPath()) if status == self.NOT_SYNCHRONIZED: self.setTempXML(None) self.setPartialXML(None) elif status in (self.PUB_CONFLICT_MERGE, self.SENT): # We have a solution for the conflict, don't need to keep the list self.resetConflictList() def getStatus(self): """ get the Status (see SyncCode for numbers) """ return self.status def getPath(self): """ get the force value (if we need to force update or not) """ return getattr(self, 'path', None) def setPath(self, path): """ set the force value (if we need to force update or not) """ self.path = path def getForce(self): """ get the force value (if we need to force update or not) """ return self.force def setForce(self, force): """ set the force value (if we need to force update or not) """ self.force = force def getLastModificationDate(self): """ get the last modfication date, so that we don't always check the xml """ return getattr(self, 'modification_date', None) def setLastModificationDate(self,value): """ set the last modfication date, so that we don't always check the xml """ setattr(self, 'modification_date', value) def getLastSynchronizationDate(self): """ get the last modfication date, so that we don't always check the xml """ return getattr(self, 'synchronization_date', None) def setLastSynchronizationDate(self,value): """ set the last modfication date, so that we don't always check the xml """ setattr(self, 'synchronization_date', value) def hasXML(self): """ return True if the xml is available """ return bool(getattr(self, 'xml', None)) def setXML(self, xml): """ set the XML corresponding to the object """ if xml is not None: # convert the string to Pdata if the big size file = cStringIO.StringIO(xml) self.xml, size = self.getParentValue()._read_data(file) self.setTempXML(None) # We make sure that the xml will not be erased self.setMD5(xml) else: self.xml = None def getXML(self, default=None): """ get the XML corresponding to the object """ #Never return empty string if self.hasXML(): if isinstance(self.xml, Pdata): return str(self.xml) elif isinstance(self.xml, str): return self.xml else: raise ValueError, "the self.xml haven't good type" else: return default def hasTempXML(self): """ Return true if the temp_xml is available """ return bool(getattr(self, 'temp_xml', None)) def setTempXML(self, xml): """ This is the xml temporarily saved, it will be stored with setXML when we will receive the confirmation of synchronization """ if xml is not None: file = cStringIO.StringIO(xml) self.temp_xml, size = self.getParentValue()._read_data(file) else: self.temp_xml = None def getTempXML(self, default=None): """ get the temp xml """ if self.hasTempXML(): if isinstance(self.temp_xml, Pdata): return str(self.temp_xml) elif isinstance(self.temp_xml, str): return self.temp_xml else: raise ValueError, "the self.xml haven't good type" else: return default def setSubscriberXupdate(self, xupdate): """ set the full temp xupdate """ self.subscriber_xupdate = xupdate def getSubscriberXupdate(self): """ get the full temp xupdate """ return self.subscriber_xupdate def setPublisherXupdate(self, xupdate): """ set the full temp xupdate """ self.publisher_xupdate = xupdate def getPublisherXupdate(self): """ get the full temp xupdate """ return self.publisher_xupdate def setMD5(self, xml): """ set the MD5 object of this signature """ self.md5_string = md5.new(xml).digest() def getMD5(self): """ get the MD5 object of this signature """ return self.md5_string def checkMD5(self, xml_string): """ check if the given md5_object returns the same things as the one stored in this signature, this is very usefull if we want to know if an objects has changed or not Returns 1 if MD5 are equals, else it returns 0 """ return ((md5.new(xml_string).digest()) == self.getMD5()) def setRid(self, rid): """ set the rid """ if isinstance(rid, unicode): rid = rid.encode('utf-8') self.rid = rid def getRid(self): """ get the rid """ return getattr(self, 'rid', None) def setId(self, id): """ set the id """ if isinstance(id, unicode): id = id.encode('utf-8') self.id = id def getId(self): """ get the id """ return self.id def getGid(self): """ get the gid """ return self.getId() def setObjectId(self, id): """ set the id of the object associated to this signature """ if isinstance(id, unicode): id = id.encode('utf-8') self.object_id = id def getObjectId(self): """ get the id of the object associated to this signature """ return getattr(self, 'object_id', None) def hasPartialXML(self): """ Return true is the partial xml is available """ return bool(getattr(self, 'partial_xml', None)) def setPartialXML(self, xml): """ Set the partial string we will have to deliver in the future """ if xml is not None: # change encoding of xml to convert in file try: xml = xml.encode('utf-8') except UnicodeDecodeError: xml = xml.decode('utf-8').encode('ascii','xmlcharrefreplace') # convert the string to Pdata if the big size file = cStringIO.StringIO(xml) self.partial_xml, size = self.getParentValue()._read_data(file) if not isinstance(self.partial_xml, Pdata): self.partial_xml = Pdata(self.partial_xml) self.last_data_partial_xml = self.partial_xml.getLastPdata() else: self.partial_xml = None self.last_data_partial_xml = None def appendPartialXML(self, xml): """ Append the partial string we will have to deliver in the future """ if xml is not None: try: xml = xml.encode('utf-8') except UnicodeDecodeError: xml = xml.decode('utf-8').encode('ascii','xmlcharrefreplace') file = cStringIO.StringIO(xml) xml_append, size = self.getParentValue()._read_data(file) if not isinstance(xml_append, Pdata): xml_append = Pdata(xml_append) last_data = xml_append.getLastPdata() if self.last_data_partial_xml is not None: self.last_data_partial_xml.next = xml_append else: self.partial_xml = xml_append self.last_data_partial_xml = last_data def getFirstChunkPdata(self, size_lines): """ """ chunk = list() chunk.append(self.partial_xml.data) size = chunk[0].count('\n') current = self.partial_xml next = current.next while size < size_lines and next is not None: current = next size += current.data.count('\n') chunk.append(current.data) next = current.next if size == size_lines: self.partial_xml = next elif size > size_lines: overflow = size - size_lines data_list = chunk[-1].split('\n') chunk[-1] = '\n'.join(data_list[:-overflow]) if current is not None: current.data = '\n'.join(data_list[-overflow:]) self.partial_xml = current else: self.partial_xml.data = '\n'.join(data_list[-overflow:]) return ''.join(chunk) def getPartialXML(self, default=None): """ Set the partial string we will have to deliver in the future """ if self.hasPartialXML(): if isinstance(self.partial_xml, Pdata): return str(self.partial_xml) else: raise ValueError, "the self.xml haven't good type" else: return default def getAction(self): """ Return the actual action for a partial synchronization """ return self.action def setAction(self, action): """ Return the actual action for a partial synchronization """ self.action = action def getConflictList(self): """ Return the actual action for a partial synchronization """ returned_conflict_list = [] if len(self.conflict_list)>0: returned_conflict_list.extend(self.conflict_list) return returned_conflict_list def resetConflictList(self): """ Return the actual action for a partial synchronization """ self.conflict_list = PersistentMapping() def setConflictList(self, conflict_list): """ Return the actual action for a partial synchronization """ if conflict_list is None or conflict_list == []: self.resetConflictList() else: self.conflict_list = conflict_list def delConflict(self, conflict): """ Return the actual action for a partial synchronization """ conflict_list = [] for c in self.getConflictList(): #LOG('delConflict, c==conflict',0,c==aq_base(conflict)) if c != aq_base(conflict): conflict_list += [c] if conflict_list != []: self.setConflictList(conflict_list) else: self.resetConflictList() def getObject(self): """ Returns the object corresponding to this signature """ return self.getParentValue().getObjectFromGid(self.getObjectId())