Commit e7f9fa99 authored by Jérome Perrin's avatar Jérome Perrin

erp5/upgrade_test: make sure users can create and search for content

Extend the upgrade tests to create data before the upgrade, to
confirm that this data can still be found after the upgrade.

This also extends the test to confirm that after upgrade, inituser
can still login, create content and find content created before and
after upgrade.

After 4fa33dfc (erp5: py3: `func_{code,defaults}` was replaced in
Python3 by `__{code,defaults}__`., 2022-04-25), this was no longer
the case, because during indexation, using existing scripts failed
with AttributeError __code__
parent eeeab727
...@@ -25,35 +25,43 @@ ...@@ -25,35 +25,43 @@
# #
############################################################################## ##############################################################################
import contextlib
import glob import glob
import json import json
import os import os
import ssl
import sys
import tempfile import tempfile
import time import time
import requests import requests
import urlparse import six.moves.urllib as urllib
import six.moves.xmlrpc_client
import urllib3
from slapos.testing.testcase import makeModuleSetUpAndTestCaseClass
from slapos.testing.testcase import installSoftwareUrlList
from slapos.testing.testcase import SlapOSNodeCommandError
from slapos.grid.utils import md5digest from slapos.grid.utils import md5digest
from slapos.testing.testcase import (
SlapOSNodeCommandError,
installSoftwareUrlList,
makeModuleSetUpAndTestCaseClass,
)
old_software_release_url = 'https://lab.nexedi.com/nexedi/slapos/raw/1.0.167.5/software/erp5/software.cfg' old_software_release_url = 'https://lab.nexedi.com/nexedi/slapos/raw/1.0.167.5/software/erp5/software.cfg'
new_software_release_url = os.path.abspath( new_software_release_url = os.path.abspath(
os.path.join(os.path.dirname(__file__), '..', 'software.cfg')) os.path.join(os.path.dirname(__file__), '..', 'software.cfg'))
_, SlapOSInstanceTestCase = makeModuleSetUpAndTestCaseClass( _, SlapOSInstanceTestCase = makeModuleSetUpAndTestCaseClass(
old_software_release_url, old_software_release_url,
software_id="upgrade_erp5", software_id="upgrade_erp5",
skip_software_check=True, skip_software_check=True,
) )
def setUpModule(): def setUpModule():
installSoftwareUrlList( installSoftwareUrlList(
SlapOSInstanceTestCase, SlapOSInstanceTestCase,
[old_software_release_url, new_software_release_url], [old_software_release_url, new_software_release_url],
debug=SlapOSInstanceTestCase._debug, debug=SlapOSInstanceTestCase._debug,
) )
...@@ -87,7 +95,7 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase): ...@@ -87,7 +95,7 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase):
# wait for slapos node instance # wait for slapos node instance
snapshot_name = "{}.{}.setUpClass new instance".format( snapshot_name = "{}.{}.setUpClass new instance".format(
cls.__module__, cls.__name__) cls.__module__, cls.__name__)
try: try:
if cls._debug and cls.instance_max_retry: if cls._debug and cls.instance_max_retry:
try: try:
...@@ -95,8 +103,8 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase): ...@@ -95,8 +103,8 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase):
except SlapOSNodeCommandError: except SlapOSNodeCommandError:
cls.slap.waitForInstance(debug=True) cls.slap.waitForInstance(debug=True)
else: else:
cls.slap.waitForInstance(max_retry=cls.instance_max_retry, cls.slap.waitForInstance(
debug=cls._debug) max_retry=cls.instance_max_retry, debug=cls._debug)
cls.logger.debug("instance on new software done") cls.logger.debug("instance on new software done")
except BaseException: except BaseException:
cls.logger.exception("Error during instance on new software") cls.logger.exception("Error during instance on new software")
...@@ -113,20 +121,149 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase): ...@@ -113,20 +121,149 @@ class ERP5UpgradeTestCase(SlapOSInstanceTestCase):
class TestERP5Upgrade(ERP5UpgradeTestCase): class TestERP5Upgrade(ERP5UpgradeTestCase):
@classmethod @classmethod
def setUpOldInstance(cls): def setUpOldInstance(cls):
cls._default_instance_old_parameter_dict = json.loads( cls._default_instance_old_parameter_dict = param_dict = json.loads(
cls.computer_partition.getConnectionParameterDict()['_']) cls.computer_partition.getConnectionParameterDict()['_'])
# use a session to retry on failures, when ERP5 is not ready.
# (see also TestPublishedURLIsReachableMixin)
cls.session = requests.Session()
cls.session.mount(
param_dict['family-default-v6'],
requests.adapters.HTTPAdapter(
max_retries=urllib3.util.retry.Retry(
total=20,
backoff_factor=.1,
status_forcelist=(404, 500, 503),
)))
# rebuild an url with user and password
parsed = urllib.parse.urlparse(param_dict['family-default'])
cls.authenticated_zope_base_url = parsed._replace(
netloc='{}:{}@{}:{}'.format(
param_dict['inituser-login'],
param_dict['inituser-password'],
parsed.hostname,
parsed.port,
),
path=param_dict['site-id'] + '/',
).geturl()
cls.zope_base_url = '{family_default_v6}/{site_id}'.format(
family_default_v6=param_dict['family-default-v6'],
site_id=param_dict['site-id'],
)
# wait for old site creation
cls.session.get(
'{zope_base_url}/person_module'.format(zope_base_url=cls.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
verify=False,
allow_redirects=False,
).raise_for_status()
# Create scripts to create test data and search catalog for test data.
@contextlib.contextmanager
def getXMLRPCClient():
# don't verify certificate
ssl_context = ssl.create_default_context()
ssl_context.check_hostname = False
ssl_context.verify_mode = ssl.CERT_NONE
erp5_xmlrpc_client = six.moves.xmlrpc_client.ServerProxy(
cls.authenticated_zope_base_url,
context=ssl_context,
)
# BBB use as a context manager only on python3
if sys.version_info < (3, ):
yield erp5_xmlrpc_client
else:
with erp5_xmlrpc_client:
yield erp5_xmlrpc_client
def addPythonScript(script_id, params, body):
with getXMLRPCClient() as erp5_xmlrpc_client:
custom = erp5_xmlrpc_client.portal_skins.custom
try:
custom.manage_addProduct.PythonScripts.manage_addPythonScript(
script_id)
except six.moves.xmlrpc_client.ProtocolError as e:
if e.errcode != 302:
raise
getattr(custom, script_id).ZPythonScriptHTML_editAction(
'',
'',
params,
body,
)
# a python script to create a person with a name
addPythonScript(
script_id='ERP5Site_createTestPerson',
params='name',
body='''if 1:
portal = context.getPortalObject()
portal.person_module.newContent(
first_name=name,
)
return 'Done.'
''',
)
# a python script to search for persons by name
addPythonScript(
script_id='ERP5Site_searchTestPerson',
params='name',
body='''if 1:
import json
portal = context.getPortalObject()
result = [brain.getObject().getTitle() for brain in portal.portal_catalog(
portal_type='Person',
title=name,)]
assert result # raise so that we retry until indexed
return json.dumps(result)
''',
)
cls.session.post(
'{zope_base_url}/ERP5Site_createTestPerson'.format(
zope_base_url=cls.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
data={
'name': 'before upgrade'
},
verify=False,
allow_redirects=False,
).raise_for_status()
assert cls.session.get(
'{zope_base_url}/ERP5Site_searchTestPerson'.format(
zope_base_url=cls.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
params={
'name': 'before upgrade'
},
verify=False,
allow_redirects=False,
).json() == ['before upgrade']
def test_published_url_is_same(self): def test_published_url_is_same(self):
default_instance_new_parameter_dict = json.loads( default_instance_new_parameter_dict = json.loads(
self.computer_partition.getConnectionParameterDict()['_']) self.computer_partition.getConnectionParameterDict()['_'])
self.assertEqual( self.assertEqual(
default_instance_new_parameter_dict['family-default-v6'], default_instance_new_parameter_dict['family-default-v6'],
self._default_instance_old_parameter_dict['family-default-v6'], self._default_instance_old_parameter_dict['family-default-v6'],
) )
def test_published_url_is_reachable(self): def test_published_url_is_reachable(self):
default_instance_new_parameter_dict = json.loads( default_instance_new_parameter_dict = json.loads(
self.computer_partition.getConnectionParameterDict()['_']) self.computer_partition.getConnectionParameterDict()['_'])
# get certificate from caucase # get certificate from caucase
with tempfile.NamedTemporaryFile( with tempfile.NamedTemporaryFile(
...@@ -135,44 +272,84 @@ class TestERP5Upgrade(ERP5UpgradeTestCase): ...@@ -135,44 +272,84 @@ class TestERP5Upgrade(ERP5UpgradeTestCase):
delete=False, delete=False,
) as ca_cert: ) as ca_cert:
ca_cert.write( ca_cert.write(
requests.get( requests.get(
urlparse.urljoin( urllib.parse.urljoin(
default_instance_new_parameter_dict['caucase-http-url'], default_instance_new_parameter_dict['caucase-http-url'],
'/cas/crt/ca.crt.pem', '/cas/crt/ca.crt.pem',
)).text) )).text)
ca_cert.flush() ca_cert.flush()
# use a session to retry on failures, when ERP5 is not ready. self.session.get(
# (see also TestPublishedURLIsReachableMixin) '{}/{}/login_form'.format(
session = requests.Session() default_instance_new_parameter_dict['family-default-v6'],
session.mount( default_instance_new_parameter_dict['site-id'],
default_instance_new_parameter_dict['family-default-v6'], ),
requests.adapters.HTTPAdapter( verify=False,
max_retries=requests.packages.urllib3.util.retry.Retry( # TODO: we don't use caucase yet here.
total=60, # verify=ca_cert.name,
backoff_factor=.5, ).raise_for_status()
status_forcelist=(404, 500, 503))))
session.get(
'{}/{}/login_form'.format(
default_instance_new_parameter_dict['family-default-v6'],
default_instance_new_parameter_dict['site-id'],
),
verify=False,
# TODO: we don't use caucase yet here.
# verify=ca_cert.name,
).raise_for_status()
def test_all_instances_use_new_software_release(self): def test_all_instances_use_new_software_release(self):
self.assertEqual( self.assertEqual(
{ {
os.path.basename(os.readlink(sr)) os.path.basename(os.readlink(sr))
for sr in glob.glob( for sr in glob.glob(
os.path.join( os.path.join(
self.slap.instance_directory, self.slap.instance_directory,
'*', '*',
'software_release', 'software_release',
)) ))
},
{md5digest(self.getSoftwareURL())},
)
def test_catalog_available(self):
param_dict = json.loads(
self.computer_partition.getConnectionParameterDict()['_'])
# data created before upgrade is available
self.assertEqual(
self.session.get(
'{zope_base_url}/ERP5Site_searchTestPerson'.format(
zope_base_url=self.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
params={
'name': 'before upgrade'
},
verify=False,
allow_redirects=False,
).json(), ['before upgrade'])
# create data after upgrade
self.session.post(
'{zope_base_url}/ERP5Site_createTestPerson'.format(
zope_base_url=self.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
data={
'name': 'after upgrade'
},
verify=False,
allow_redirects=False,
).raise_for_status()
# new data can also be found
self.assertEqual(
self.session.get(
'{zope_base_url}/ERP5Site_searchTestPerson'.format(
zope_base_url=self.zope_base_url),
auth=requests.auth.HTTPBasicAuth(
username=param_dict['inituser-login'],
password=param_dict['inituser-password'],
),
params={
'name': 'after upgrade'
}, },
{md5digest(self.getSoftwareURL())},) verify=False,
allow_redirects=False,
).json(), ['after upgrade'])
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