[Verify] Fixing issue with system-wide Tempest installation
The issue was noticed in a Rally docker container. In the docker container, as we know, Rally is installed in the global system environment. When we try to execute $ rally verify install --system-wide Tempest installation will be broken on $ pip install --no-deps -e ./ because of permission issues. The command requires the 'sudo' prefix. So it looks like the best solution for resolving the issue is that user will take care of installing Tempest package like he does for Tempest requirements. Change-Id: I3c606d1f659600f027ecaf6fe99eae26c329c9a3
This commit is contained in:
parent
d7629a9e38
commit
59aa4e8b91
@ -455,7 +455,8 @@ class Verification(object):
|
|||||||
:param source: Path/URL to repo to clone Tempest from
|
:param source: Path/URL to repo to clone Tempest from
|
||||||
:param version: Commit ID or tag to checkout before Tempest
|
:param version: Commit ID or tag to checkout before Tempest
|
||||||
installation
|
installation
|
||||||
:param system_wide: Whether or not to create a Tempest virtual env
|
:param system_wide: Whether or not to install Tempest package and
|
||||||
|
create a Tempest virtual env
|
||||||
"""
|
"""
|
||||||
deployment_uuid = objects.Deployment.get(deployment)["uuid"]
|
deployment_uuid = objects.Deployment.get(deployment)["uuid"]
|
||||||
verifier = tempest.Tempest(deployment_uuid, source=source,
|
verifier = tempest.Tempest(deployment_uuid, source=source,
|
||||||
@ -481,7 +482,8 @@ class Verification(object):
|
|||||||
:param source: Path/URL to repo to clone Tempest from
|
:param source: Path/URL to repo to clone Tempest from
|
||||||
:param version: Commit ID or tag to checkout before Tempest
|
:param version: Commit ID or tag to checkout before Tempest
|
||||||
installation
|
installation
|
||||||
:param system_wide: Whether or not to create a Tempest virtual env
|
:param system_wide: Whether or not to install Tempest package and
|
||||||
|
create a Tempest virtual env
|
||||||
"""
|
"""
|
||||||
deployment_uuid = objects.Deployment.get(deployment)["uuid"]
|
deployment_uuid = objects.Deployment.get(deployment)["uuid"]
|
||||||
verifier = tempest.Tempest(deployment_uuid, source=source,
|
verifier = tempest.Tempest(deployment_uuid, source=source,
|
||||||
|
@ -420,8 +420,9 @@ class VerifyCommands(object):
|
|||||||
help="Commit ID or tag to checkout before Tempest "
|
help="Commit ID or tag to checkout before Tempest "
|
||||||
"installation")
|
"installation")
|
||||||
@cliutils.args("--system-wide", dest="system_wide",
|
@cliutils.args("--system-wide", dest="system_wide",
|
||||||
help="Don't create a virtual env for Tempest. Note "
|
help="Not to install Tempest package and not to create "
|
||||||
"that all Tempest requirements have to be already "
|
"a virtual env for Tempest. Note that Tempest package "
|
||||||
|
"and all Tempest requirements have to be already "
|
||||||
"installed in the local env!",
|
"installed in the local env!",
|
||||||
required=False, action="store_true")
|
required=False, action="store_true")
|
||||||
@envutils.with_default_deployment(cli_arg_name="deployment")
|
@envutils.with_default_deployment(cli_arg_name="deployment")
|
||||||
@ -433,7 +434,8 @@ class VerifyCommands(object):
|
|||||||
:param source: Path/URL to repo to clone Tempest from
|
:param source: Path/URL to repo to clone Tempest from
|
||||||
:param version: Commit ID or tag to checkout before Tempest
|
:param version: Commit ID or tag to checkout before Tempest
|
||||||
installation
|
installation
|
||||||
:param system_wide: Whether or not to create a Tempest virtual env
|
:param system_wide: Whether or not to install Tempest package and
|
||||||
|
create a Tempest virtual env
|
||||||
"""
|
"""
|
||||||
api.Verification.install_tempest(deployment, source,
|
api.Verification.install_tempest(deployment, source,
|
||||||
version, system_wide)
|
version, system_wide)
|
||||||
@ -458,8 +460,9 @@ class VerifyCommands(object):
|
|||||||
help="Commit ID or tag to checkout before Tempest "
|
help="Commit ID or tag to checkout before Tempest "
|
||||||
"installation")
|
"installation")
|
||||||
@cliutils.args("--system-wide", dest="system_wide",
|
@cliutils.args("--system-wide", dest="system_wide",
|
||||||
help="Don't create a virtual env for Tempest. Note "
|
help="Not to install Tempest package and not to create "
|
||||||
"that all Tempest requirements have to be already "
|
"a virtual env for Tempest. Note that Tempest package "
|
||||||
|
"and all Tempest requirements have to be already "
|
||||||
"installed in the local env!",
|
"installed in the local env!",
|
||||||
required=False, action="store_true")
|
required=False, action="store_true")
|
||||||
@envutils.with_default_deployment(cli_arg_name="deployment")
|
@envutils.with_default_deployment(cli_arg_name="deployment")
|
||||||
@ -471,7 +474,8 @@ class VerifyCommands(object):
|
|||||||
:param source: Path/URL to repo to clone Tempest from
|
:param source: Path/URL to repo to clone Tempest from
|
||||||
:param version: Commit ID or tag to checkout before Tempest
|
:param version: Commit ID or tag to checkout before Tempest
|
||||||
installation
|
installation
|
||||||
:param system_wide: Whether or not to create a Tempest virtual env
|
:param system_wide: Whether or not to install Tempest package and
|
||||||
|
create a Tempest virtual env
|
||||||
"""
|
"""
|
||||||
api.Verification.reinstall_tempest(deployment, source,
|
api.Verification.reinstall_tempest(deployment, source,
|
||||||
version, system_wide)
|
version, system_wide)
|
||||||
|
@ -41,7 +41,7 @@ class TempestSetupFailure(exceptions.RallyException):
|
|||||||
|
|
||||||
|
|
||||||
def check_output(*args, **kwargs):
|
def check_output(*args, **kwargs):
|
||||||
print_debug_output = kwargs.pop("print_debug_output", True)
|
debug = kwargs.pop("debug", True)
|
||||||
kwargs["stderr"] = subprocess.STDOUT
|
kwargs["stderr"] = subprocess.STDOUT
|
||||||
try:
|
try:
|
||||||
output = subprocess.check_output(*args, **kwargs)
|
output = subprocess.check_output(*args, **kwargs)
|
||||||
@ -50,8 +50,9 @@ def check_output(*args, **kwargs):
|
|||||||
LOG.error("Error output: '%s'" % encodeutils.safe_decode(e.output))
|
LOG.error("Error output: '%s'" % encodeutils.safe_decode(e.output))
|
||||||
raise
|
raise
|
||||||
|
|
||||||
if print_debug_output:
|
if debug:
|
||||||
LOG.debug("Subprocess output: '%s'" % encodeutils.safe_decode(output))
|
LOG.debug("Subprocess output: '%s'" % encodeutils.safe_decode(output))
|
||||||
|
|
||||||
return output
|
return output
|
||||||
|
|
||||||
|
|
||||||
@ -193,10 +194,8 @@ class Tempest(object):
|
|||||||
cwd=self.path())
|
cwd=self.path())
|
||||||
# NOTE(kun): Using develop mode installation is for running
|
# NOTE(kun): Using develop mode installation is for running
|
||||||
# multiple Tempest instances.
|
# multiple Tempest instances.
|
||||||
check_output([self.venv_wrapper,
|
check_output([self.venv_wrapper, "pip", "install", "-e", "./"],
|
||||||
"pip", "install", "-r",
|
cwd=self.path())
|
||||||
"requirements.txt", "-r",
|
|
||||||
"test-requirements.txt"], cwd=self.path())
|
|
||||||
except subprocess.CalledProcessError:
|
except subprocess.CalledProcessError:
|
||||||
if os.path.exists(self.path(".venv")):
|
if os.path.exists(self.path(".venv")):
|
||||||
shutil.rmtree(self.path(".venv"))
|
shutil.rmtree(self.path(".venv"))
|
||||||
@ -268,14 +267,14 @@ class Tempest(object):
|
|||||||
if not self._is_git_repo(self.base_repo):
|
if not self._is_git_repo(self.base_repo):
|
||||||
self._clone()
|
self._clone()
|
||||||
shutil.copytree(self.base_repo, self.path())
|
shutil.copytree(self.base_repo, self.path())
|
||||||
if self.version:
|
|
||||||
cmd = ["git", "checkout", self.version]
|
if self.version:
|
||||||
subprocess.check_call(cmd, cwd=self.path("tempest"))
|
check_output(["git", "checkout", self.version],
|
||||||
cmd = ["pip", "install", "--no-deps", "-e", "./"]
|
cwd=self.path())
|
||||||
|
|
||||||
if not self._system_wide:
|
if not self._system_wide:
|
||||||
self._install_venv()
|
self._install_venv()
|
||||||
cmd.insert(0, self.path("tools/with_venv.sh"))
|
|
||||||
check_output(cmd, cwd=self.path())
|
|
||||||
self._initialize_testr()
|
self._initialize_testr()
|
||||||
except subprocess.CalledProcessError as e:
|
except subprocess.CalledProcessError as e:
|
||||||
self.uninstall()
|
self.uninstall()
|
||||||
@ -313,10 +312,18 @@ class Tempest(object):
|
|||||||
|
|
||||||
def list_plugins(self):
|
def list_plugins(self):
|
||||||
"""List all installed Tempest plugins for local Tempest repo."""
|
"""List all installed Tempest plugins for local Tempest repo."""
|
||||||
cmd = ["tempest", "list-plugins"]
|
cmd_list_plugins = ["tempest", "list-plugins"]
|
||||||
if not self._system_wide:
|
if not self._system_wide:
|
||||||
cmd.insert(0, self.path("tools/with_venv.sh"))
|
cmd_list_plugins.insert(0, self.path("tools/with_venv.sh"))
|
||||||
return check_output(cmd, cwd=self.path(), print_debug_output=False)
|
else:
|
||||||
|
cmd_pip_list = ["pip", "list"]
|
||||||
|
if "tempest" not in check_output(cmd_pip_list,
|
||||||
|
cwd=self.path(), debug=False):
|
||||||
|
return _("Cannot list Tempest plugins because Tempest "
|
||||||
|
"package is not installed in your environment. "
|
||||||
|
"Please, install Tempest package and try again.")
|
||||||
|
|
||||||
|
return check_output(cmd_list_plugins, cwd=self.path(), debug=False)
|
||||||
|
|
||||||
def uninstall_plugin(self, repo_name):
|
def uninstall_plugin(self, repo_name):
|
||||||
"""Uninstall Tempest plugin for local Tempest repo."""
|
"""Uninstall Tempest plugin for local Tempest repo."""
|
||||||
|
@ -98,8 +98,8 @@ class TempestUtilsTestCase(BaseTestCase):
|
|||||||
@mock.patch("os.path.isdir", return_value=False)
|
@mock.patch("os.path.isdir", return_value=False)
|
||||||
@mock.patch("%s.tempest.check_output" % TEMPEST_PATH,
|
@mock.patch("%s.tempest.check_output" % TEMPEST_PATH,
|
||||||
return_value="some_output")
|
return_value="some_output")
|
||||||
def test__venv_install_when_venv_not_exist(self, mock_check_output,
|
def test__venv_install_when_venv_does_not_exist(self, mock_check_output,
|
||||||
mock_isdir, mock_sys):
|
mock_isdir, mock_sys):
|
||||||
mock_sys.version_info = "not_py27_env"
|
mock_sys.version_info = "not_py27_env"
|
||||||
self.verifier._install_venv()
|
self.verifier._install_venv()
|
||||||
|
|
||||||
@ -107,9 +107,8 @@ class TempestUtilsTestCase(BaseTestCase):
|
|||||||
mock_check_output.assert_has_calls([
|
mock_check_output.assert_has_calls([
|
||||||
mock.call(["virtualenv", "-p", mock_sys.executable, ".venv"],
|
mock.call(["virtualenv", "-p", mock_sys.executable, ".venv"],
|
||||||
cwd="/tmp"),
|
cwd="/tmp"),
|
||||||
mock.call(["/tmp/tools/with_venv.sh", "pip", "install", "-r",
|
mock.call(["/tmp/tools/with_venv.sh", "pip",
|
||||||
"requirements.txt", "-r", "test-requirements.txt"],
|
"install", "-e", "./"], cwd="/tmp")
|
||||||
cwd="/tmp")
|
|
||||||
])
|
])
|
||||||
|
|
||||||
@mock.patch("%s.tempest.sys" % TEMPEST_PATH)
|
@mock.patch("%s.tempest.sys" % TEMPEST_PATH)
|
||||||
@ -234,7 +233,6 @@ class TempestInstallAndUninstallTestCase(BaseTestCase):
|
|||||||
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._initialize_testr")
|
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._initialize_testr")
|
||||||
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._install_venv")
|
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._install_venv")
|
||||||
@mock.patch(TEMPEST_PATH + ".tempest.check_output")
|
@mock.patch(TEMPEST_PATH + ".tempest.check_output")
|
||||||
@mock.patch(TEMPEST_PATH + ".tempest.subprocess.check_call")
|
|
||||||
@mock.patch("shutil.copytree")
|
@mock.patch("shutil.copytree")
|
||||||
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._clone")
|
@mock.patch(TEMPEST_PATH + ".tempest.Tempest._clone")
|
||||||
@mock.patch("os.path.exists", return_value=False)
|
@mock.patch("os.path.exists", return_value=False)
|
||||||
@ -242,8 +240,7 @@ class TempestInstallAndUninstallTestCase(BaseTestCase):
|
|||||||
return_value=False)
|
return_value=False)
|
||||||
def test_install_successful(self, mock_tempest__is_git_repo, mock_exists,
|
def test_install_successful(self, mock_tempest__is_git_repo, mock_exists,
|
||||||
mock_tempest__clone, mock_copytree,
|
mock_tempest__clone, mock_copytree,
|
||||||
mock_check_call, mock_check_output,
|
mock_check_output, mock_tempest__install_venv,
|
||||||
mock_tempest__install_venv,
|
|
||||||
mock_tempest__initialize_testr,
|
mock_tempest__initialize_testr,
|
||||||
mock_tempest_base_repo):
|
mock_tempest_base_repo):
|
||||||
mock_tempest_base_repo.__get__ = mock.Mock(return_value="fake_dir")
|
mock_tempest_base_repo.__get__ = mock.Mock(return_value="fake_dir")
|
||||||
@ -258,9 +255,9 @@ class TempestInstallAndUninstallTestCase(BaseTestCase):
|
|||||||
mock_copytree.assert_called_once_with(
|
mock_copytree.assert_called_once_with(
|
||||||
self.verifier.base_repo,
|
self.verifier.base_repo,
|
||||||
self.verifier.path())
|
self.verifier.path())
|
||||||
cwd = self.verifier.path("tempest")
|
cwd = self.verifier.path()
|
||||||
expected = [mock.call(["git", "checkout", "3f4c8d44"], cwd=cwd)]
|
expected = [mock.call(["git", "checkout", "3f4c8d44"], cwd=cwd)]
|
||||||
self.assertEqual(expected, mock_check_call.mock_calls)
|
self.assertEqual(expected, mock_check_output.mock_calls)
|
||||||
mock_tempest__install_venv.assert_called_once_with()
|
mock_tempest__install_venv.assert_called_once_with()
|
||||||
mock_tempest__initialize_testr.assert_called_once_with()
|
mock_tempest__initialize_testr.assert_called_once_with()
|
||||||
|
|
||||||
@ -347,7 +344,7 @@ class TempestInstallPluginsTestCase(BaseTestCase):
|
|||||||
|
|
||||||
cmd = [self.verifier.venv_wrapper, "tempest", "list-plugins"]
|
cmd = [self.verifier.venv_wrapper, "tempest", "list-plugins"]
|
||||||
mock_tempest_check_output.assert_called_with(
|
mock_tempest_check_output.assert_called_with(
|
||||||
cmd, cwd=self.verifier.path(), print_debug_output=False)
|
cmd, cwd=self.verifier.path(), debug=False)
|
||||||
|
|
||||||
@mock.patch("shutil.rmtree")
|
@mock.patch("shutil.rmtree")
|
||||||
@mock.patch("os.path.exists")
|
@mock.patch("os.path.exists")
|
||||||
|
Loading…
x
Reference in New Issue
Block a user