2022-06-30 20:43:04 +00:00
|
|
|
import os
|
|
|
|
from unittest import mock
|
2022-08-02 08:31:09 +00:00
|
|
|
from unittest.mock import MagicMock
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
from click.testing import CliRunner
|
|
|
|
from lightning_cloud.openapi import Externalv1LightningappInstance
|
|
|
|
|
2022-09-01 07:58:09 +00:00
|
|
|
from lightning_app import __version__
|
2022-07-25 17:13:46 +00:00
|
|
|
from lightning_app.cli.lightning_cli import _main, get_app_url, login, logout, run
|
2022-08-02 08:31:09 +00:00
|
|
|
from lightning_app.cli.lightning_cli_create import create, create_cluster
|
|
|
|
from lightning_app.cli.lightning_cli_delete import delete, delete_cluster
|
2022-08-05 20:42:00 +00:00
|
|
|
from lightning_app.cli.lightning_cli_list import get_list, list_apps, list_clusters
|
2022-06-30 20:43:04 +00:00
|
|
|
from lightning_app.runners.runtime_type import RuntimeType
|
2022-10-27 13:44:22 +00:00
|
|
|
from lightning_app.utilities.exceptions import _ApiExceptionHandler
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"runtime_type, extra_args, lightning_cloud_url, expected_url",
|
|
|
|
[
|
|
|
|
(
|
|
|
|
RuntimeType.CLOUD,
|
|
|
|
(Externalv1LightningappInstance(id="test-app-id"),),
|
|
|
|
"https://b975913c4b22eca5f0f9e8eff4c4b1c315340a0d.staging.lightning.ai",
|
|
|
|
"https://b975913c4b22eca5f0f9e8eff4c4b1c315340a0d.staging.lightning.ai/me/apps/test-app-id",
|
|
|
|
),
|
|
|
|
(
|
|
|
|
RuntimeType.CLOUD,
|
|
|
|
(Externalv1LightningappInstance(id="test-app-id"),),
|
|
|
|
"http://localhost:9800",
|
|
|
|
"http://localhost:9800/me/apps/test-app-id",
|
|
|
|
),
|
|
|
|
(RuntimeType.SINGLEPROCESS, tuple(), "", "http://127.0.0.1:7501/view"),
|
|
|
|
(RuntimeType.SINGLEPROCESS, tuple(), "http://localhost:9800", "http://127.0.0.1:7501/view"),
|
|
|
|
(RuntimeType.MULTIPROCESS, tuple(), "", "http://127.0.0.1:7501/view"),
|
|
|
|
(RuntimeType.MULTIPROCESS, tuple(), "http://localhost:9800", "http://127.0.0.1:7501/view"),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_start_target_url(runtime_type, extra_args, lightning_cloud_url, expected_url):
|
|
|
|
with mock.patch(
|
|
|
|
"lightning_app.cli.lightning_cli.get_lightning_cloud_url", mock.MagicMock(return_value=lightning_cloud_url)
|
|
|
|
):
|
|
|
|
assert get_app_url(runtime_type, *extra_args) == expected_url
|
|
|
|
|
|
|
|
|
2022-08-02 08:31:09 +00:00
|
|
|
@pytest.mark.parametrize("command", [_main, run, get_list, create, delete])
|
2022-06-30 20:43:04 +00:00
|
|
|
def test_commands(command):
|
|
|
|
runner = CliRunner()
|
|
|
|
result = runner.invoke(command)
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
|
2022-09-27 03:43:36 +00:00
|
|
|
def test_main_lightning_cli_no_arguments():
|
|
|
|
"""Validate the Lightning CLI without args."""
|
|
|
|
res = os.popen("python -m lightning").read()
|
|
|
|
assert "login " in res
|
|
|
|
assert "logout " in res
|
|
|
|
assert "run " in res
|
|
|
|
assert "list " in res
|
|
|
|
assert "delete " in res
|
|
|
|
assert "create " in res
|
|
|
|
assert "show " in res
|
2022-11-03 15:32:08 +00:00
|
|
|
assert "ssh " in res
|
2022-10-28 18:42:26 +00:00
|
|
|
assert "add " in res
|
|
|
|
assert "remove " in res
|
2022-09-27 03:43:36 +00:00
|
|
|
|
|
|
|
|
2022-06-30 20:43:04 +00:00
|
|
|
def test_main_lightning_cli_help():
|
|
|
|
"""Validate the Lightning CLI."""
|
2022-07-25 17:13:46 +00:00
|
|
|
res = os.popen("python -m lightning --help").read()
|
2022-06-30 20:43:04 +00:00
|
|
|
assert "login " in res
|
|
|
|
assert "logout " in res
|
|
|
|
assert "run " in res
|
2022-08-02 08:31:09 +00:00
|
|
|
assert "list " in res
|
|
|
|
assert "delete " in res
|
|
|
|
assert "create " in res
|
2022-08-23 19:48:22 +00:00
|
|
|
assert "show " in res
|
2022-11-03 15:32:08 +00:00
|
|
|
assert "ssh " in res
|
2022-10-28 18:42:26 +00:00
|
|
|
assert "add " in res
|
|
|
|
assert "remove " in res
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2022-07-25 17:13:46 +00:00
|
|
|
res = os.popen("python -m lightning run --help").read()
|
2022-06-30 20:43:04 +00:00
|
|
|
assert "app " in res
|
|
|
|
|
|
|
|
# hidden run commands should not appear in the help text
|
|
|
|
assert "server" not in res
|
|
|
|
assert "flow" not in res
|
|
|
|
assert "work" not in res
|
|
|
|
assert "frontend" not in res
|
|
|
|
|
2022-08-23 19:48:22 +00:00
|
|
|
# inspect show group
|
|
|
|
res = os.popen("python -m lightning show --help").read()
|
|
|
|
assert "logs " in res
|
|
|
|
assert "cluster " in res
|
|
|
|
|
|
|
|
# inspect show cluster group
|
|
|
|
res = os.popen("python -m lightning show cluster --help").read()
|
|
|
|
assert "logs " in res
|
|
|
|
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2022-08-02 08:31:09 +00:00
|
|
|
@mock.patch("lightning_cloud.login.Auth.authenticate", MagicMock())
|
|
|
|
@mock.patch("lightning_app.cli.cmd_clusters.AWSClusterManager.create")
|
2022-08-09 19:17:57 +00:00
|
|
|
@pytest.mark.parametrize(
|
2022-10-31 18:19:26 +00:00
|
|
|
"extra_arguments,expected_cost_savings_mode",
|
2022-08-09 19:17:57 +00:00
|
|
|
[
|
2022-10-31 18:19:26 +00:00
|
|
|
([], True),
|
|
|
|
(["--enable-performance"], False),
|
2022-08-09 19:17:57 +00:00
|
|
|
],
|
|
|
|
)
|
2022-10-31 18:19:26 +00:00
|
|
|
def test_create_cluster(create_command: mock.MagicMock, extra_arguments, expected_cost_savings_mode):
|
2022-08-02 08:31:09 +00:00
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(
|
|
|
|
create_cluster,
|
|
|
|
[
|
|
|
|
"test-7",
|
|
|
|
"--provider",
|
|
|
|
"aws",
|
|
|
|
"--external-id",
|
|
|
|
"dummy",
|
|
|
|
"--role-arn",
|
|
|
|
"arn:aws:iam::1234567890:role/lai-byoc",
|
2022-08-09 19:17:57 +00:00
|
|
|
]
|
2022-08-11 15:19:21 +00:00
|
|
|
+ extra_arguments,
|
2022-08-02 08:31:09 +00:00
|
|
|
)
|
|
|
|
|
2022-08-09 19:17:57 +00:00
|
|
|
create_command.assert_called_once_with(
|
2022-08-02 08:31:09 +00:00
|
|
|
cluster_name="test-7",
|
|
|
|
region="us-east-1",
|
|
|
|
role_arn="arn:aws:iam::1234567890:role/lai-byoc",
|
|
|
|
external_id="dummy",
|
|
|
|
edit_before_creation=False,
|
2022-08-11 15:19:21 +00:00
|
|
|
cost_savings=expected_cost_savings_mode,
|
2022-08-02 08:31:09 +00:00
|
|
|
wait=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-08-05 20:42:00 +00:00
|
|
|
@mock.patch("lightning_cloud.login.Auth.authenticate", MagicMock())
|
|
|
|
@mock.patch("lightning_app.cli.cmd_apps._AppManager.list")
|
|
|
|
def test_list_apps(list_command: mock.MagicMock):
|
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(list_apps)
|
|
|
|
|
|
|
|
list_command.assert_called_once_with(cluster_id=None)
|
|
|
|
|
|
|
|
|
2022-08-02 08:31:09 +00:00
|
|
|
@mock.patch("lightning_cloud.login.Auth.authenticate", MagicMock())
|
|
|
|
@mock.patch("lightning_app.cli.cmd_clusters.AWSClusterManager.list")
|
2022-08-05 20:42:00 +00:00
|
|
|
def test_list_clusters(list_command: mock.MagicMock):
|
2022-08-02 08:31:09 +00:00
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(list_clusters)
|
|
|
|
|
2022-08-05 20:42:00 +00:00
|
|
|
list_command.assert_called_once_with()
|
2022-08-02 08:31:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock.patch("lightning_cloud.login.Auth.authenticate", MagicMock())
|
|
|
|
@mock.patch("lightning_app.cli.cmd_clusters.AWSClusterManager.delete")
|
|
|
|
def test_delete_cluster(delete: mock.MagicMock):
|
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(delete_cluster, ["test-7"])
|
|
|
|
|
|
|
|
delete.assert_called_once_with(cluster_id="test-7", force=False, wait=False)
|
|
|
|
|
|
|
|
|
2022-06-30 20:43:04 +00:00
|
|
|
@mock.patch("lightning_app.utilities.login.Auth._run_server")
|
|
|
|
@mock.patch("lightning_app.utilities.login.Auth.clear")
|
|
|
|
def test_cli_login(clear: mock.MagicMock, run_server: mock.MagicMock):
|
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(login)
|
|
|
|
|
|
|
|
clear.assert_called_once_with()
|
|
|
|
run_server.assert_called_once()
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch("pathlib.Path.unlink")
|
|
|
|
@mock.patch("pathlib.Path.exists")
|
|
|
|
@pytest.mark.parametrize("creds", [True, False])
|
|
|
|
def test_cli_logout(exists: mock.MagicMock, unlink: mock.MagicMock, creds: bool):
|
|
|
|
exists.return_value = creds
|
|
|
|
runner = CliRunner()
|
|
|
|
runner.invoke(logout)
|
|
|
|
|
|
|
|
exists.assert_called_once_with()
|
|
|
|
if creds:
|
|
|
|
unlink.assert_called_once_with()
|
|
|
|
else:
|
|
|
|
unlink.assert_not_called()
|
2022-08-26 23:55:22 +00:00
|
|
|
|
|
|
|
|
2022-09-01 07:58:09 +00:00
|
|
|
def test_lightning_cli_version():
|
|
|
|
res = os.popen("python -m lightning --version").read()
|
|
|
|
assert __version__ in res
|
2022-10-27 13:44:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_main_catches_api_exceptions():
|
|
|
|
assert isinstance(_main, _ApiExceptionHandler)
|