2021-11-23 16:25:09 +08:00
|
|
|
from unittest.mock import Mock, patch
|
|
|
|
|
2022-02-07 18:11:43 +01:00
|
|
|
from tests.helpers import PluginsTestCase, temporary_root
|
2021-11-23 16:25:09 +08:00
|
|
|
from tutor import images, plugins
|
2022-02-07 18:11:43 +01:00
|
|
|
from tutor.__about__ import __version__
|
|
|
|
from tutor.commands.images import ImageNotFoundError
|
|
|
|
|
|
|
|
from .base import TestCommandMixin
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
|
2022-02-07 18:11:43 +01:00
|
|
|
class ImagesTests(PluginsTestCase, TestCommandMixin):
|
2021-11-23 16:25:09 +08:00
|
|
|
def test_images_help(self) -> None:
|
2022-02-07 18:11:43 +01:00
|
|
|
result = self.invoke(["images", "--help"])
|
2021-11-23 16:25:09 +08:00
|
|
|
self.assertIsNone(result.exception)
|
2022-02-07 18:11:43 +01:00
|
|
|
self.assertEqual(0, result.exit_code)
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
def test_images_pull_image(self) -> None:
|
2022-02-07 18:11:43 +01:00
|
|
|
result = self.invoke(["images", "pull"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
def test_images_pull_plugin_invalid_plugin_should_throw_error(self) -> None:
|
2022-02-07 18:11:43 +01:00
|
|
|
result = self.invoke(["images", "pull", "plugin"])
|
|
|
|
self.assertEqual(1, result.exit_code)
|
|
|
|
self.assertEqual(ImageNotFoundError, type(result.exception))
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
@patch.object(images, "pull", return_value=None)
|
2022-02-07 18:11:43 +01:00
|
|
|
def test_images_pull_plugin(self, image_pull: Mock) -> None:
|
|
|
|
plugins.v0.DictPlugin(
|
|
|
|
{
|
|
|
|
"name": "plugin1",
|
|
|
|
"hooks": {
|
|
|
|
"remote-image": {
|
|
|
|
"service1": "service1:1.0.0",
|
|
|
|
"service2": "service2:2.0.0",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
plugins.load("plugin1")
|
|
|
|
result = self.invoke(["images", "pull", "service1"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
|
|
|
image_pull.assert_called_once_with("service1:1.0.0")
|
|
|
|
|
|
|
|
@patch.object(images, "pull", return_value=None)
|
|
|
|
def test_images_pull_all_vendor_images(self, image_pull: Mock) -> None:
|
|
|
|
result = self.invoke(["images", "pull", "mysql"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
|
|
|
# Note: we should update this tag whenever the mysql image is updated
|
|
|
|
image_pull.assert_called_once_with("docker.io/mysql:5.7.35")
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
def test_images_printtag_image(self) -> None:
|
2022-02-07 18:11:43 +01:00
|
|
|
result = self.invoke(["images", "printtag", "openedx"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
|
|
|
self.assertRegex(
|
|
|
|
result.output, rf"docker.io/overhangio/openedx:{__version__}\n"
|
|
|
|
)
|
2021-11-23 16:25:09 +08:00
|
|
|
|
2022-02-07 18:11:43 +01:00
|
|
|
def test_images_printtag_plugin(self) -> None:
|
|
|
|
plugins.v0.DictPlugin(
|
|
|
|
{
|
|
|
|
"name": "plugin1",
|
|
|
|
"hooks": {
|
|
|
|
"build-image": {
|
|
|
|
"service1": "service1:1.0.0",
|
|
|
|
"service2": "service2:2.0.0",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
plugins.load("plugin1")
|
|
|
|
result = self.invoke(["images", "printtag", "service1"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code, result)
|
|
|
|
self.assertEqual(result.output, "service1:1.0.0\n")
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
@patch.object(images, "build", return_value=None)
|
2022-02-07 18:11:43 +01:00
|
|
|
def test_images_build_plugin(self, mock_image_build: Mock) -> None:
|
|
|
|
plugins.v0.DictPlugin(
|
|
|
|
{
|
|
|
|
"name": "plugin1",
|
|
|
|
"hooks": {
|
|
|
|
"build-image": {
|
|
|
|
"service1": "service1:1.0.0",
|
|
|
|
"service2": "service2:2.0.0",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
plugins.load("plugin1")
|
2021-11-23 16:25:09 +08:00
|
|
|
with temporary_root() as root:
|
2022-02-07 18:11:43 +01:00
|
|
|
self.invoke_in_root(root, ["config", "save"])
|
|
|
|
result = self.invoke_in_root(root, ["images", "build", "service1"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
|
|
|
mock_image_build.assert_called()
|
|
|
|
self.assertIn("service1:1.0.0", mock_image_build.call_args[0])
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
@patch.object(images, "build", return_value=None)
|
2022-02-07 18:11:43 +01:00
|
|
|
def test_images_build_plugin_with_args(self, image_build: Mock) -> None:
|
|
|
|
plugins.v0.DictPlugin(
|
|
|
|
{
|
|
|
|
"name": "plugin1",
|
|
|
|
"hooks": {
|
|
|
|
"build-image": {
|
|
|
|
"service1": "service1:1.0.0",
|
|
|
|
"service2": "service2:2.0.0",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
plugins.load("plugin1")
|
|
|
|
build_args = [
|
|
|
|
"images",
|
|
|
|
"build",
|
|
|
|
"--no-cache",
|
|
|
|
"-a",
|
|
|
|
"myarg=value",
|
|
|
|
"--add-host",
|
|
|
|
"host",
|
|
|
|
"--target",
|
|
|
|
"target",
|
|
|
|
"-d",
|
|
|
|
"docker_args",
|
|
|
|
"service1",
|
|
|
|
]
|
2021-11-23 16:25:09 +08:00
|
|
|
with temporary_root() as root:
|
2022-02-07 18:11:43 +01:00
|
|
|
self.invoke_in_root(root, ["config", "save"])
|
|
|
|
result = self.invoke_in_root(root, build_args)
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|
|
|
|
image_build.assert_called()
|
|
|
|
self.assertIn("service1:1.0.0", image_build.call_args[0])
|
|
|
|
for arg in image_build.call_args[0][2:]:
|
|
|
|
# The only extra args are `--build-arg`
|
|
|
|
if arg != "--build-arg":
|
|
|
|
self.assertIn(arg, build_args)
|
2021-11-23 16:25:09 +08:00
|
|
|
|
|
|
|
def test_images_push(self) -> None:
|
2022-02-07 18:11:43 +01:00
|
|
|
result = self.invoke(["images", "push"])
|
|
|
|
self.assertIsNone(result.exception)
|
|
|
|
self.assertEqual(0, result.exit_code)
|