Commit 8bc6d650 authored by Ayush Tiwari's avatar Ayush Tiwari

bt5_config: Add migration function

parent 423608ef
...@@ -335,7 +335,7 @@ class BusinessManager(Folder): ...@@ -335,7 +335,7 @@ class BusinessManager(Folder):
__rsub__ = __sub__ __rsub__ = __sub__
security.declareProtected(Permissions.ManagePortal, 'storeTemplateData') security.declareProtected(Permissions.ManagePortal, 'storeTemplateData')
def storeTemplateData(self, isBuild=False): def storeTemplateData(self, isBuild=False, **kw):
""" """
Store data for objects in the ERP5 Store data for objects in the ERP5
""" """
...@@ -451,10 +451,16 @@ class BusinessManager(Folder): ...@@ -451,10 +451,16 @@ class BusinessManager(Folder):
"""Creates new values for business item from the values from """Creates new values for business item from the values from
OFS Database""" OFS Database"""
LOG('Business Manager', INFO, 'Building Business Manager') LOG('Business Manager', INFO, 'Building Business Manager')
removable_sub_object_path_list = kw.get('removable_sub_object_path', [])
removable_property_dict = kw.get('removable_property', {})
if not no_action: if not no_action:
self.storeTemplateData(isBuild=True) self.storeTemplateData(isBuild=True, **kw)
for path_item in self.objectValues(): for path_item in self.objectValues():
path_item.build(self, **kw) kwargs = {}
item_path = path_item.getProperty('item_path')
kwargs['removable_property_list'] = removable_property_dict.get(item_path, [])
kwargs['remove_sub_objects'] = item_path in removable_sub_object_path_list
path_item.build(self, **kwargs)
self.status = 'built' self.status = 'built'
return self return self
...@@ -652,6 +658,19 @@ class BusinessItem(XMLObject): ...@@ -652,6 +658,19 @@ class BusinessItem(XMLObject):
for relative_url in self._resolvePath(p, [], path.split('/')): for relative_url in self._resolvePath(p, [], path.split('/')):
obj = p.unrestrictedTraverse(relative_url) obj = p.unrestrictedTraverse(relative_url)
obj = obj._getCopy(context) obj = obj._getCopy(context)
# We should remove the extra properties of object so that there
# shouldn't be redundancy of the proeprties
removable_property_list = kw.get('removable_property_list')
# We should also add extra parameter to remove sub-objects by removing
# `_tree` for any erp5 object. This way we can have control over adding
# sub-objects as new Business Item objects
remove_sub_objects = kw.get('remove_sub_objects')
if remove_sub_objects:
removable_property_list.append('_tree')
obj = self.removeProperties(obj, 1, properties=removable_property_list)
obj = obj.__of__(context) obj = obj.__of__(context)
# XXX: '_recursiveRemoveUid' is not working as expected # XXX: '_recursiveRemoveUid' is not working as expected
_recursiveRemoveUid(obj) _recursiveRemoveUid(obj)
...@@ -888,7 +907,13 @@ class BusinessItem(XMLObject): ...@@ -888,7 +907,13 @@ class BusinessItem(XMLObject):
for attr in obj.__dict__.keys(): for attr in obj.__dict__.keys():
if attr in attr_set or attr.startswith('_cache_cookie_'): if attr in attr_set or attr.startswith('_cache_cookie_'):
try:
delattr(obj, attr) delattr(obj, attr)
except AttributeError:
# XXX: Continue in cases where we want to delete some properties which
# are not in attribute list
# Raise an error
continue
if classname == 'PDFForm': if classname == 'PDFForm':
if not obj.getProperty('business_template_include_content', 1): if not obj.getProperty('business_template_include_content', 1):
......
...@@ -2412,6 +2412,7 @@ class ERP5Generator(PortalGenerator): ...@@ -2412,6 +2412,7 @@ class ERP5Generator(PortalGenerator):
continue continue
url = getBootstrapBusinessTemplateUrl(bt) url = getBootstrapBusinessTemplateUrl(bt)
bt = template_tool.download(url) bt = template_tool.download(url)
if bt.getPortalType() == 'Business Manager': if bt.getPortalType() == 'Business Manager':
template_tool.updateInstallationState([bt]) template_tool.updateInstallationState([bt])
else: else:
......
...@@ -40,11 +40,12 @@ from Acquisition import Implicit, Explicit ...@@ -40,11 +40,12 @@ from Acquisition import Implicit, Explicit
from AccessControl import ClassSecurityInfo from AccessControl import ClassSecurityInfo
from AccessControl.SecurityInfo import ModuleSecurityInfo from AccessControl.SecurityInfo import ModuleSecurityInfo
from Products.CMFActivity.ActiveResult import ActiveResult from Products.CMFActivity.ActiveResult import ActiveResult
from Products.PythonScripts.PythonScript import PythonScript
from Products.ERP5Type.Globals import InitializeClass, DTMLFile, PersistentMapping from Products.ERP5Type.Globals import InitializeClass, DTMLFile, PersistentMapping
from Products.ERP5Type.DiffUtils import DiffFile from Products.ERP5Type.DiffUtils import DiffFile
from Products.ERP5Type.Tool.BaseTool import BaseTool from Products.ERP5Type.Tool.BaseTool import BaseTool
from Products.ERP5Type.Cache import transactional_cached from Products.ERP5Type.Cache import transactional_cached
from Products.ERP5Type import Permissions from Products.ERP5Type import Permissions, interfaces
from Products.ERP5.Document.BusinessTemplate import BusinessTemplateMissingDependency from Products.ERP5.Document.BusinessTemplate import BusinessTemplateMissingDependency
from Products.ERP5Type.Accessor.Constant import PropertyGetter as ConstantGetter from Products.ERP5Type.Accessor.Constant import PropertyGetter as ConstantGetter
from Products.ERP5.genbt5list import generateInformation from Products.ERP5.genbt5list import generateInformation
...@@ -580,25 +581,25 @@ class TemplateTool (BaseTool): ...@@ -580,25 +581,25 @@ class TemplateTool (BaseTool):
self.activate(activity='SQLQueue').\ self.activate(activity='SQLQueue').\
importAndReExportBusinessTemplateFromPath(template_path) importAndReExportBusinessTemplateFromPath(template_path)
security.declareProtected( 'Import/Export objects', 'migrateBTToBP') security.declareProtected( 'Import/Export objects', 'migrateBTToBM')
def migrateBTToBP(self, template_path, REQUEST=None, **kw): def migrateBTToBM(self, template_path, REQUEST=None, **kw):
""" """
Migrate business template repository to Business Package repo. Migrate business template repository to Business Manager repo.
Business Package completely rely only on PathTemplateItem and to show Business Manager completely rely only on BusinessItem and to show
the difference between both of them the difference between both of them
So, the steps should be: So, the steps should be:
1. Install the business template which is going to be migrated 1. Install the business template which is going to be migrated
2. Create a new Business Package with random id and title 2. Create a new Business Manager with random id and title
3. Add the path, build and export the template 3. Add the path, build and export the template
4. Remove the business template from the directory and add the business 4. Remove the business template from the directory and add the business
package there instead manager there instead
5. Change the ID and title of the business package 5. Change the ID and title of the business manager
6. Export the business package to the directory, leaving anything in 6. Export the business manager to the directory, leaving anything in
the installed erp5 unchanged the installed erp5 unchanged
""" """
import_template = self.download(url=template_path) import_template = self.download(url=template_path)
if import_template.getPortalType == 'Business Package': if import_template.getPortalType() == 'Business Manager':
LOG(import_template.getTitle(),0,'Already migrated') LOG(import_template.getTitle(),0,'Already migrated')
return return
...@@ -613,7 +614,7 @@ class TemplateTool (BaseTool): ...@@ -613,7 +614,7 @@ class TemplateTool (BaseTool):
import_template.install(**kw) import_template.install(**kw)
is_installed = True is_installed = True
# Make list of object paths which needs to be added in the bp5 # Make list of object paths which needs to be added in the bm5
# This can be decided by looping over every type of items we do have in # This can be decided by looping over every type of items we do have in
# bt5 and checking if there have been any changes being made to it via this # bt5 and checking if there have been any changes being made to it via this
# bt5 installation or not. # bt5 installation or not.
...@@ -629,30 +630,31 @@ class TemplateTool (BaseTool): ...@@ -629,30 +630,31 @@ class TemplateTool (BaseTool):
# For modules, we don't need to create path for the module # For modules, we don't need to create path for the module
module_list = import_template.getTemplateModuleIdList() module_list = import_template.getTemplateModuleIdList()
template_path_list.extend(module_list) for path in module_list:
template_path_list.append(path + ' | 1 | 1')
# For portal_types, we have to add path and subobjetcs # For portal_types, we have to add path and subobjects
portal_type_id_list = import_template.getTemplatePortalTypeIdList() portal_type_id_list = import_template.getTemplatePortalTypeIdList()
portal_type_path_list = [] portal_type_path_list = []
for id in portal_type_id_list: for id in portal_type_id_list:
portal_type_path_list.append('portal_types/'+id) portal_type_path_list.append('portal_types/'+id + ' | 1 | 1')
portal_type_path_list.append('portal_types/'+id+'/**') #portal_type_path_list.append('portal_types/'+id+'/**')
template_path_list.extend(portal_type_path_list) template_path_list.extend(portal_type_path_list)
# For categories, we create path for category objects as well as the subcategories # For categories, we create path for category objects as well as the subcategories
category_list = import_template.getTemplateBaseCategoryList() category_list = import_template.getTemplateBaseCategoryList()
category_path_list = [] category_path_list = []
for base_category in category_list: for base_category in category_list:
category_path_list.append('portal_categories/'+base_category) category_path_list.append('portal_categories/'+base_category + ' | 1 | 1')
category_path_list.append('portal_categories/'+base_category+'/**') #category_path_list.append('portal_categories/'+base_category+'/**')
template_path_list.extend(category_path_list) template_path_list.extend(category_path_list)
# For portal_skins, we export the folder # For portal_skins, we export the folder
portal_skin_list = import_template.getTemplateSkinIdList() portal_skin_list = import_template.getTemplateSkinIdList()
portal_skin_path_list = [] portal_skin_path_list = []
for skin in portal_skin_list: for skin in portal_skin_list:
portal_skin_path_list.append('portal_skins/'+skin) portal_skin_path_list.append('portal_skins/'+skin + ' | 1 | 1')
portal_skin_path_list.append('portal_skins/'+skin+'/**') #portal_skin_path_list.append('portal_skins/'+skin+'/**')
template_path_list.extend(portal_skin_path_list) template_path_list.extend(portal_skin_path_list)
# For workflow chains, # For workflow chains,
...@@ -665,18 +667,20 @@ class TemplateTool (BaseTool): ...@@ -665,18 +667,20 @@ class TemplateTool (BaseTool):
workflow_id_list = import_template.getTemplateWorkflowIdList() workflow_id_list = import_template.getTemplateWorkflowIdList()
workflow_path_list = [] workflow_path_list = []
for workflow in workflow_id_list: for workflow in workflow_id_list:
workflow_path_list.append('portal_workflow/' + workflow) workflow_path_list.append('portal_workflow/' + workflow + ' | 1 | 1')
workflow_path_list.append('portal_workflow/' + workflow + '/**') #workflow_path_list.append('portal_workflow/' + workflow + '/**')
template_path_list.extend(workflow_path_list) template_path_list.extend(workflow_path_list)
# For paths, we add them directly to the path list # For paths, we add them directly to the path list
template_path_list.extend(import_template.getTemplatePathList()) path_list = import_template.getTemplatePathList()
for path in path_list:
template_path_list.append(path + ' | 1 | 1')
# Catalog methods would be added as sub objects # Catalog methods would be added as sub objects
catalog_method_item_list = import_template.getTemplateCatalogMethodIdList() catalog_method_item_list = import_template.getTemplateCatalogMethodIdList()
catalog_method_path_list = [] catalog_method_path_list = []
for method in catalog_method_item_list: for method in catalog_method_item_list:
catalog_method_path_list.append('portal_catalog/' + method) catalog_method_path_list.append('portal_catalog/' + method + ' | 1 | 1')
template_path_list.extend(catalog_method_path_list) template_path_list.extend(catalog_method_path_list)
# For catalog objects, we check if there is any catalog object, and then # For catalog objects, we check if there is any catalog object, and then
...@@ -685,6 +689,7 @@ class TemplateTool (BaseTool): ...@@ -685,6 +689,7 @@ class TemplateTool (BaseTool):
template_catalog_full_text_key = import_template.getTemplateCatalogFullTextKeyList() template_catalog_full_text_key = import_template.getTemplateCatalogFullTextKeyList()
template_catalog_keyword_key = import_template.getTemplateCatalogKeywordKeyList() template_catalog_keyword_key = import_template.getTemplateCatalogKeywordKeyList()
template_catalog_local_role_key = import_template.getTemplateCatalogLocalRoleKeyList() template_catalog_local_role_key = import_template.getTemplateCatalogLocalRoleKeyList()
template_catalog_method_id = import_template.getTemplateCatalogMethodIdList()
template_catalog_multivalue_key = import_template.getTemplateCatalogMultivalueKeyList() template_catalog_multivalue_key = import_template.getTemplateCatalogMultivalueKeyList()
template_catalog_related_key = import_template.getTemplateCatalogRelatedKeyList() template_catalog_related_key = import_template.getTemplateCatalogRelatedKeyList()
template_catalog_request_key = import_template.getTemplateCatalogRequestKeyList() template_catalog_request_key = import_template.getTemplateCatalogRequestKeyList()
...@@ -701,6 +706,7 @@ class TemplateTool (BaseTool): ...@@ -701,6 +706,7 @@ class TemplateTool (BaseTool):
template_catalog_full_text_key, template_catalog_full_text_key,
template_catalog_keyword_key, template_catalog_keyword_key,
template_catalog_local_role_key, template_catalog_local_role_key,
template_catalog_method_id,
template_catalog_multivalue_key, template_catalog_multivalue_key,
template_catalog_related_key, template_catalog_related_key,
template_catalog_request_key, template_catalog_request_key,
...@@ -719,6 +725,7 @@ class TemplateTool (BaseTool): ...@@ -719,6 +725,7 @@ class TemplateTool (BaseTool):
'sql_catalog_full_text_search_keys_list', 'sql_catalog_full_text_search_keys_list',
'sql_catalog_keyword_search_keys_list', 'sql_catalog_keyword_search_keys_list',
'sql_catalog_local_role_keys_list', 'sql_catalog_local_role_keys_list',
'sql_catalog_method_id_list',
'sql_catalog_multivalue_keys_list', 'sql_catalog_multivalue_keys_list',
'sql_catalog_related_keys_list', 'sql_catalog_related_keys_list',
'sql_catalog_request_keys_list', 'sql_catalog_request_keys_list',
...@@ -728,54 +735,88 @@ class TemplateTool (BaseTool): ...@@ -728,54 +735,88 @@ class TemplateTool (BaseTool):
'sql_catalog_scriptable_keys_list', 'sql_catalog_scriptable_keys_list',
'sql_catalog_search_keys_list', 'sql_catalog_search_keys_list',
'sql_catalog_security_uid_columns_list', 'sql_catalog_security_uid_columns_list',
'sql_catalog_topic_search_keys_list' 'sql_catalog_topic_search_keys_list',
] ]
removable_property = {} removable_property = {}
removable_sub_object_path = []
if is_property_added: if is_property_added:
if catalog_method_path_list: if catalog_method_path_list:
catalog_path = catalog_method_path_list[0].rsplit('/', 1)[0] catalog_path = catalog_method_path_list[0].rsplit('/', 1)[0]
else: else:
catalog_path = 'portal_catalog/erp5_mysql_innodb' catalog_path = 'portal_catalog/erp5_mysql_innodb'
template_path_list.append(catalog_path) removable_sub_object_path.append(catalog_path)
template_path_list.append(catalog_path + ' | 1 | 1')
removable_property[catalog_path] = properties_removed removable_property[catalog_path] = properties_removed
for prop in properties_removed: for prop in properties_removed:
property_path_list.append(catalog_path + ' | ' + prop) property_path_list.append('%s#%s | 1 | 1' % (catalog_path, prop))
# Add these catalog items in the object_property instead of adding # Add these catalog items in the object_property instead of adding
# dummy path item for them # dummy path item for them
if import_template.getTitle() == 'erp5_mysql_innodb_catalog': if import_template.getTitle() == 'erp5_mysql_innodb_catalog':
template_path_list.extend('portal_catalog/erp5_mysql_innodb') template_path_list.extend('portal_catalog/erp5_mysql_innodb | 1 | 1')
# Add portal_property_sheets # Add portal_property_sheets
property_sheet_id_list = import_template.getTemplatePropertySheetIdList() property_sheet_id_list = import_template.getTemplatePropertySheetIdList()
property_sheet_path_list = [] property_sheet_path_list = []
for property_sheet in property_sheet_id_list: for property_sheet in property_sheet_id_list:
property_sheet_path_list.append('portal_property_sheets/' + property_sheet) property_sheet_path_list.append('portal_property_sheets/' + property_sheet + ' | 1 | 1')
property_sheet_path_list.append('portal_property_sheets/' + property_sheet + '/**') #property_sheet_path_list.append('portal_property_sheets/' + property_sheet + '/**')
template_path_list.extend(property_sheet_path_list) template_path_list.extend(property_sheet_path_list)
# Create new objects for business package # Create new objects for business package
bp5_package = self.newContent( migrated_bm = self.newContent(
portal_type='Business Package', portal_type='Business Manager',
title=import_template.getTitle() title=import_template.getTitle()
) )
bp5_package.edit( template_path_list.extend(property_path_list)
template_path_list=template_path_list, template_path_list = self.cleanTemplatePathList(template_path_list)
template_object_property_list=property_path_list migrated_bm.setProperty('template_path_list', template_path_list)
)
kw['removable_property'] = removable_property kw['removable_property'] = removable_property
bp5_package.build(**kw) kw['removable_sub_object_path'] = removable_sub_object_path
migrated_bm.build(**kw)
# Export the newly built business package to the export directory # Export the newly built business package to the export directory
bp5_package.export(path=export_dir, local=True) migrated_bm.export(path=export_dir, local=True)
if is_installed: if is_installed:
import_template.uninstall() import_template.uninstall()
security.declareProtected( 'Import/Export objects', 'migrateBTListToBP') def cleanTemplatePathList(self, path_list):
def migrateBTListToBP(self, repository_list, REQUEST=None, **kw): """
Remove redundant paths and sub-objects' path if the object path already
exist.
"""
# Remove layer and sign
a1 = [l.split(' | ')[0] for l in path_list]
# Split path into list
a2 = [l.split('/') for l in a1]
# Create new list for paths with **
a3 = [l for l in a2 if l[-1] == '**']
# Create new list for paths without **
a4 = [l for l in a2 if l[-1] != '**']
# Remove ** from paths in a3
reserved_id = ('portal_transforms', 'portal_ids')
a3 = [l[:-1] for l in a3 if l[0] not in reserved_id] + [l for l in a3 if l[0] in reserved_id]
# Create new final path list
a2 = a3+a4
# Join the path list
a2 = [('/').join(l) for l in a2]
# Remove the redundant paths
a2 = list(set(a2))
# Sort the path list
a2.sort()
# Add the layer and signs, for now all 1
a2 = [l+' | 1 | 1' for l in a2]
return a2
security.declareProtected( 'Import/Export objects', 'migrateBTListToBM')
def migrateBTListToBM(self, repository_list, REQUEST=None, **kw):
""" """
Run migration for BT5 one by one in a given repository. This will be done Run migration for BT5 one by one in a given repository. This will be done
via activities. via activities.
...@@ -801,7 +842,7 @@ class TemplateTool (BaseTool): ...@@ -801,7 +842,7 @@ class TemplateTool (BaseTool):
if not os.path.exists((os.path.join(template_path, 'bt'))): if not os.path.exists((os.path.join(template_path, 'bt'))):
LOG(business_template_id,0,'has no bt sub-folder, so it is skipped') LOG(business_template_id,0,'has no bt sub-folder, so it is skipped')
else: else:
self.migrateBTToBP(template_path) self.migrateBTToBM(template_path)
#self.activate(activity='SQLQueue').\ #self.activate(activity='SQLQueue').\
# migrateBTToBP(template_path) # migrateBTToBP(template_path)
...@@ -1704,7 +1745,7 @@ class TemplateTool (BaseTool): ...@@ -1704,7 +1745,7 @@ class TemplateTool (BaseTool):
6. If conflict while comaprison at 3, raise the error 6. If conflict while comaprison at 3, raise the error
7. In all other case, install the BM List 7. In all other case, install the BM List
""" """
# Create old installation state from Instsalled Business Manager # Create old installation state from Installed Business Manager
installed_bm_list = self.getInstalledBusinessManagerList() installed_bm_list = self.getInstalledBusinessManagerList()
combined_installed_path_item = [item for bm combined_installed_path_item = [item for bm
in installed_bm_list in installed_bm_list
...@@ -1741,7 +1782,8 @@ class TemplateTool (BaseTool): ...@@ -1741,7 +1782,8 @@ class TemplateTool (BaseTool):
for item in combined_new_path_item: for item in combined_new_path_item:
item.isIndexable = ConstantGetter('isIndexable', value=False) item.isIndexable = ConstantGetter('isIndexable', value=False)
new_installation_state._setObject(item.getId(), aq_base(item), new_id = new_installation_state.generateNewId()
new_installation_state._setObject(new_id, aq_base(item),
suppress_events=True) suppress_events=True)
# Create installation process, which have the changes to be made in the # Create installation process, which have the changes to be made in the
...@@ -1778,6 +1820,8 @@ class TemplateTool (BaseTool): ...@@ -1778,6 +1820,8 @@ class TemplateTool (BaseTool):
value_list = item.objectValues() value_list = item.objectValues()
if value_list: if value_list:
value = value_list[0] value = value_list[0]
else:
value = ''
if value: if value:
item.setProperty('item_sha', self.calculateComparableHash( item.setProperty('item_sha', self.calculateComparableHash(
value, value,
...@@ -1793,7 +1837,12 @@ class TemplateTool (BaseTool): ...@@ -1793,7 +1837,12 @@ class TemplateTool (BaseTool):
if item.isProperty: if item.isProperty:
value = item.getProperty('item_property_value') value = item.getProperty('item_property_value')
else: else:
value = item.objectValues()[0] value_list = item.objectValues()
if value_list:
value = value_list[0]
else:
value = ''
if value:
item.setProperty('item_sha', self.calculateComparableHash( item.setProperty('item_sha', self.calculateComparableHash(
value, value,
item.isProperty, item.isProperty,
...@@ -1835,20 +1884,36 @@ class TemplateTool (BaseTool): ...@@ -1835,20 +1884,36 @@ class TemplateTool (BaseTool):
if isProperty: if isProperty:
obj_dict = object obj_dict = object
else: else:
klass = object.__class__
classname = klass.__name__
obj_dict = object.__dict__.copy() obj_dict = object.__dict__.copy()
removable_attributes = [attr for attr attr_set = {'_dav_writelocks', '_filepath', '_owner', '_related_index',
in obj_dict.keys() 'last_id', 'uid',
if attr.startswith('_v')] '__ac_local_roles__', '__ac_local_roles_group_id_dict__'}
removable_attributes.append('uid') attr_set.update(('isIndexable',))
removable_attributes.append('_owner')
removable_attributes.append('isIndexable') if classname in ('File', 'Image'):
for attr in removable_attributes: attr_set.update(('_EtagSupport__etag', 'size'))
elif classname == 'Types Tool' and klass.__module__ == 'erp5.portal_type':
attr_set.add('type_provider_list')
for attr in object.__dict__.keys():
if attr in attr_set or attr.startswith('_cache_cookie_') or attr.startswith('_v'):
try: try:
del obj_dict[attr] del obj_dict[attr]
except KeyError: except AttributeError:
# XXX: Continue in cases where we want to delete some properties which
# are not in attribute list
# Raise an error
continue continue
if 'data' in obj_dict:
try:
obj_dict['data'] = obj_dict.get('data').__dict__
except AttributeError:
pass
obj_sha = hash(pprint.pformat(obj_dict)) obj_sha = hash(pprint.pformat(obj_dict))
return obj_sha return obj_sha
...@@ -1861,7 +1926,6 @@ class TemplateTool (BaseTool): ...@@ -1861,7 +1926,6 @@ class TemplateTool (BaseTool):
error_list = [] error_list = []
for path in to_update_path_list: for path in to_update_path_list:
try: try:
if '#' in str(path): if '#' in str(path):
isProperty = True isProperty = True
...@@ -1963,19 +2027,6 @@ class TemplateTool (BaseTool): ...@@ -1963,19 +2027,6 @@ class TemplateTool (BaseTool):
return error_list return error_list
def migrateBTToBM(self, template_path, REQUEST=None, **kw):
"""
Migrate Business Template to Business Manager object.
* Download from path
* Build the template
* Create zexp file from the built object
* Save the zexp in the path
"""
template_downloaded = self.download(template_path)
if template_downloaded.getPortalType() == 'Business Manager':
LOG(template_downloaded.getTitle(), 0, 'Already migrated')
return
security.declareProtected(Permissions.ManagePortal, security.declareProtected(Permissions.ManagePortal,
'createNewInstallationState') 'createNewInstallationState')
def createNewInstallationState(self, bm_list, old_installation_state): def createNewInstallationState(self, bm_list, old_installation_state):
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment