TrashTool.py 8.18 KB
Newer Older
Aurel's avatar
Aurel committed
1 2 3
##############################################################################
#
# Copyright (c) 2002 Nexedi SARL and Contributors. All Rights Reserved.
4
#                    Aurelien Calonne <aurel@nexedi.com>
Aurel's avatar
Aurel committed
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
#
# 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 AccessControl import ClassSecurityInfo
31
from Products.ERP5Type.Accessor.Constant import PropertyGetter as ConstantGetter
32
from Products.ERP5Type.Globals import InitializeClass, DTMLFile
Aurel's avatar
Aurel committed
33 34 35
from Products.ERP5Type.Tool.BaseTool import BaseTool
from Products.ERP5Type import Permissions
from Products.ERP5 import _dtmldir
36
from Products.ERP5.Document.BusinessTemplate import getChainByType
37
from zLOG import LOG, WARNING
Aurel's avatar
Aurel committed
38 39 40 41 42
from DateTime import DateTime
from Acquisition import aq_base

class TrashTool(BaseTool):
  """
Vincent Pelletier's avatar
Vincent Pelletier committed
43
    TrashTool contains objects removed/replaced during installation of business templates.
Aurel's avatar
Aurel committed
44 45 46 47 48 49 50 51 52
  """
  title = 'Trash Tool'
  id = 'portal_trash'
  meta_type = 'ERP5 Trash Tool'
  portal_type = 'Trash Tool'
  allowed_types = ('ERP5 Trash Bin',)

  # Declarative Security
  security = ClassSecurityInfo()
53

Aurel's avatar
Aurel committed
54
  security.declareProtected(Permissions.ManagePortal, 'manage_overview' )
Aurel's avatar
Aurel committed
55
  manage_overview = DTMLFile( 'explainTrashTool', _dtmldir )
Aurel's avatar
Aurel committed
56 57 58 59 60 61 62 63 64 65 66 67 68

  def backupObject(self, trashbin, container_path, object_id, save, **kw):
    """
      Backup an object in a trash bin
    """
#     LOG('Trash : backup object', 0, str((container_path, object_id)))
    if save:
      # recreate path of the backup object if necessary
      backup_object_container = trashbin
      for path in container_path:
        if 'portal' in path:
          path += '_items'
        if path not in backup_object_container.objectIds():
69 70 71 72 73 74 75
          if not hasattr(aq_base(backup_object_container), "newContent"):
            backup_object_container.manage_addFolder(id=path,)
            backup_object_container = backup_object_container._getOb(path)
          else:
            backup_object_container = backup_object_container.newContent(portal_type='Trash Folder', id=path,
                                                                         is_indexable=0)
            backup_object_container.edit(isHidden=1)
Aurel's avatar
Aurel committed
76 77 78 79 80
        else:
          backup_object_container = backup_object_container._getOb(path)
      # backup the object
      # here we choose export/import to copy because cut/paste
      # do too many things and check for what we want to do
81
      obj = None
Aurel's avatar
Aurel committed
82 83 84
      if object_id not in backup_object_container.objectIds():
        # export object
        object_path = container_path + [object_id]
85
        obj = self.unrestrictedTraverse(object_path)
86
        if obj is not None:
87 88 89 90 91
          connection = obj._p_jar
          o = obj
          while connection is None:
            o = o.aq_parent
            connection=o._p_jar
92 93 94 95
          if obj._p_oid is None:
            LOG("Trash Tool backupObject", 100,
                "Trying to backup uncommitted object %s" % object_path)
            return {}
96
          copy = connection.exportFile(obj._p_oid)
97 98 99 100 101 102 103 104
          # import object in trash
          connection = backup_object_container._p_jar
          o = backup_object_container
          while connection is None:
            o = o.aq_parent
            connection=o._p_jar
          copy.seek(0)
          try:
105
            backup = connection.importFile(copy)
106
            backup.isIndexable = ConstantGetter('isIndexable', value=False)
107 108 109 110 111 112 113 114 115 116 117 118
            try:
              # the isIndexable setting above avoids the recursion of
              # manage_afterAdd on
              # Products.ERP5Type.CopySupport.CopySupport.manage_afterAdd()
              # but not on event subscribers, so we need to suppress_events,
              # otherwise subobjects will be reindexed
              backup_object_container._setObject(object_id, backup,
                                                 suppress_events=True)
            except TypeError:
              # BACK: On Zope 2.8. _setObject does not accept "suppress_events"
              # remove when we drop support
              backup_object_container._setObject(object_id, backup)
119 120 121 122 123
          except (AttributeError, ImportError):
            # XXX we can go here due to formulator because attribute
            # field_added doesn't not exists on parent if it is a Trash
            # Folder and not a Form, or a module for the old object is
            # already removed, and we cannot backup the object
124
            LOG("Trash Tool backupObject", 100, "Can't backup object %s" %(object_id))
125
            return {}
126 127 128 129 130 131 132 133

    keep_sub = kw.get('keep_subobjects', 0)
    subobjects_dict = {}

    if not keep_sub:
      # export subobjects
      if save:
        obj = backup_object_container._getOb(object_id)
134
        object_path = list(obj.getPhysicalPath())
135
      else:
Alexandre Boeglin's avatar
Alexandre Boeglin committed
136
        object_path = container_path + [object_id]
137
        obj = self.unrestrictedTraverse(object_path)
Julien Muchembled's avatar
Julien Muchembled committed
138
      if obj is not None:
139 140 141 142 143 144
        for subobject_id in list(obj.objectIds()):
          subobject_path = object_path + [subobject_id]
          subobject = self.unrestrictedTraverse(subobject_path)
          subobject_copy = subobject._p_jar.exportFile(subobject._p_oid)
          subobjects_dict[subobject_id] = subobject_copy
          if save: # remove subobjecs from backup object
Julien Muchembled's avatar
Julien Muchembled committed
145
            obj._delObject(subobject_id)
146 147 148 149 150
            if subobject_id in obj.objectIds():
              LOG('Products.ERP5.Tool.TrashTool', WARNING,
                  'Cleaning corrupted BTreeFolder2 object at %r.' % \
                                                       (subobject_path,))
              obj._cleanup()
Aurel's avatar
Aurel committed
151 152 153 154 155 156 157 158 159
    return subobjects_dict

  def newTrashBin(self, bt_title='trash', bt=None):
    """
      Create a new trash bin at upgrade of bt
    """
    # construct date
    date = DateTime()
    start_date = date.strftime('%Y-%m-%d')
160 161 162 163 164 165 166 167 168

    def getBaseTrashId():
      ''' A little function to get an id without leading underscore
      '''
      base_id = '%s' % start_date
      if bt_title not in ('', None):
        base_id = '%s_%s' % (bt_title, base_id)
      return base_id

Aurel's avatar
Aurel committed
169 170 171
    # generate id
    trash_ids = self.objectIds()
    n = 0
172 173 174 175
    new_trash_id = getBaseTrashId()
    while new_trash_id in trash_ids:
      n += 1
      new_trash_id = '%s_%s' % (getBaseTrashId(), n)
Aurel's avatar
Aurel committed
176
    # create trash bin
177 178 179 180 181 182
    trashbin = self.newContent( portal_type     = 'Trash Bin'
                              , id              = new_trash_id
                              , title           = bt_title
                              , start_date      = start_date
                              , causality_value = bt
                              )
Aurel's avatar
Aurel committed
183 184 185 186 187 188
    return trashbin

  def getTrashBinObjectsList(self, trashbin):
    """
      Return a list of trash objects for a given trash bin
    """
189
    def getChildObjects(obj):
Aurel's avatar
Aurel committed
190
      object_list = []
191 192 193 194 195
      if hasattr(aq_base(obj), 'objectValues'):
        childObjects = obj.objectValues()
      if hasattr(aq_base(obj), 'isHidden'):
        if not obj.isHidden:
          object_list.append(obj)
Aurel's avatar
Aurel committed
196
      if len(childObjects) > 0:
197
        for o in childObjects:
198
          object_list.extend(getChildObjects(o))
Aurel's avatar
Aurel committed
199
      else:
200
        object_list.append(obj)
Aurel's avatar
Aurel committed
201
      return object_list
202

Aurel's avatar
Aurel committed
203 204 205 206 207
    list = getChildObjects(trashbin)
    list.sort()
    return list

InitializeClass(TrashTool)