|
7 | 7 | import pytest
|
8 | 8 | import requests
|
9 | 9 | from pytest import CaptureFixture
|
| 10 | +from pytest import param as case |
10 | 11 | from pytest_mock import MockerFixture
|
11 | 12 |
|
12 | 13 | from zulipterminal.api_types import ServerSettings
|
13 | 14 | from zulipterminal.cli.run import (
|
14 | 15 | NotAZulipOrganizationError,
|
15 | 16 | _write_zuliprc,
|
| 17 | + check_for_default_zuliprc, |
16 | 18 | exit_with_error,
|
17 | 19 | get_login_label,
|
18 | 20 | get_server_settings,
|
19 | 21 | in_color,
|
20 | 22 | main,
|
21 | 23 | parse_args,
|
| 24 | + path_to_realm_zuliprc, |
22 | 25 | )
|
23 | 26 | from zulipterminal.model import ServerConnectionFailure
|
24 | 27 | from zulipterminal.version import ZT_VERSION
|
@@ -410,6 +413,146 @@ def test_main_cannot_write_zuliprc_given_good_credentials(
|
410 | 413 | assert lines[-1] == expected_line
|
411 | 414 |
|
412 | 415 |
|
| 416 | +# NOTE: Fixture is necessary to ensure unreadable dir is garbage-collected |
| 417 | +# See pytest issue #7821 |
| 418 | +@pytest.fixture |
| 419 | +def mock_config_dir_with_no_zuliprc(tmp_path: Path) -> Generator[Path, None, None]: |
| 420 | + config_dir_path = tmp_path / "config" |
| 421 | + config_dir_path.parent.mkdir(parents=True, exist_ok=True) |
| 422 | + yield (tmp_path / "config") |
| 423 | + |
| 424 | + |
| 425 | +def test_no_zuliprc_files_found(mocker: MockerFixture) -> None: |
| 426 | + mocker.patch("pathlib.Path.exists", return_value=False) |
| 427 | + result = check_for_default_zuliprc() |
| 428 | + assert result == "" |
| 429 | + |
| 430 | + |
| 431 | +def test_one_zuliprc_file_found( |
| 432 | + mocker: MockerFixture, mock_config_dir_with_no_zuliprc: Path |
| 433 | +) -> None: |
| 434 | + mocker.patch("pathlib.Path.exists", side_effect=[False, True]) |
| 435 | + mocker.patch( |
| 436 | + "zulipterminal.cli.run.ZULIP_CONFIG_PATH", str(mock_config_dir_with_no_zuliprc) |
| 437 | + ) |
| 438 | + mocker.patch("zulipterminal.cli.run.DOWNLOADED_PATH_ZULIPRC") |
| 439 | + mock_HOME_PATH_ZULIPRC = mocker.patch("zulipterminal.cli.run.HOME_PATH_ZULIPRC") |
| 440 | + |
| 441 | + result = check_for_default_zuliprc() |
| 442 | + |
| 443 | + assert result == mock_HOME_PATH_ZULIPRC |
| 444 | + |
| 445 | + |
| 446 | +def test_multiple_zuliprc_files_found( |
| 447 | + mocker: MockerFixture, mock_config_dir_with_no_zuliprc: Path |
| 448 | +) -> None: |
| 449 | + mocker.patch("pathlib.Path.exists", side_effect=[True, True]) |
| 450 | + mocker.patch( |
| 451 | + "zulipterminal.cli.run.ZULIP_CONFIG_PATH", str(mock_config_dir_with_no_zuliprc) |
| 452 | + ) |
| 453 | + mock_DOWNLOADED_PATH_ZULIPRC = mocker.patch( |
| 454 | + "zulipterminal.cli.run.DOWNLOADED_PATH_ZULIPRC", "downloads" |
| 455 | + ) |
| 456 | + mock_HOME_PATH_ZULIPRC = mocker.patch( |
| 457 | + "zulipterminal.cli.run.HOME_PATH_ZULIPRC", "home" |
| 458 | + ) |
| 459 | + mock_exit_with_error = mocker.patch("zulipterminal.cli.run.exit_with_error") |
| 460 | + |
| 461 | + check_for_default_zuliprc() |
| 462 | + |
| 463 | + mock_exit_with_error.assert_called_once_with( |
| 464 | + "Found multiple zuliprc files at:\n" |
| 465 | + f"{mock_DOWNLOADED_PATH_ZULIPRC}\n{mock_HOME_PATH_ZULIPRC}\n" |
| 466 | + "Please retry by specifying the path to your target zuliprc file." |
| 467 | + ) |
| 468 | + |
| 469 | + |
| 470 | +# NOTE: Fixture is necessary to ensure unreadable dir is garbage-collected |
| 471 | +# See pytest issue #7821 |
| 472 | +@pytest.fixture |
| 473 | +def mock_config_dir_with_multiple_zuliprc( |
| 474 | + tmp_path: Path, |
| 475 | +) -> Generator[Path, None, None]: |
| 476 | + config_dir_zuliprc_path = tmp_path / "config" / "zuliprc" |
| 477 | + config_dir_zuliprc_path.parent.mkdir(parents=True, exist_ok=True) |
| 478 | + config_dir_zuliprc_path.touch() |
| 479 | + |
| 480 | + org1_zuliprc_path = tmp_path / "config" / "org1" / "zuliprc" |
| 481 | + org1_zuliprc_path.parent.mkdir(parents=True, exist_ok=True) |
| 482 | + org1_zuliprc_path.touch() |
| 483 | + |
| 484 | + org2_subdir_zuliprc_path = tmp_path / "config" / "org2" / "subdir" / "zuliprc" |
| 485 | + org2_subdir_zuliprc_path.parent.mkdir(parents=True, exist_ok=True) |
| 486 | + org2_subdir_zuliprc_path.touch() |
| 487 | + |
| 488 | + yield (tmp_path / "config") |
| 489 | + |
| 490 | + |
| 491 | +def test_multiple_zuliprc_files_found_in_config_dir( |
| 492 | + mocker: MockerFixture, |
| 493 | + mock_config_dir_with_multiple_zuliprc: Path, |
| 494 | +) -> None: |
| 495 | + config_dir = mock_config_dir_with_multiple_zuliprc |
| 496 | + mocker.patch("zulipterminal.cli.run.DOWNLOADED_PATH_ZULIPRC") |
| 497 | + mocker.patch("zulipterminal.cli.run.HOME_PATH_ZULIPRC") |
| 498 | + mocker.patch("zulipterminal.cli.run.ZULIP_CONFIG_PATH", str(config_dir)) |
| 499 | + mock_exit_with_error = mocker.patch("zulipterminal.cli.run.exit_with_error") |
| 500 | + |
| 501 | + check_for_default_zuliprc() |
| 502 | + |
| 503 | + mock_exit_with_error.assert_called_once_with( |
| 504 | + "Found multiple zuliprc files at:\n" |
| 505 | + f"{str(config_dir / 'zuliprc')}\n" |
| 506 | + f"{config_dir / 'org1' / 'zuliprc'}\n" |
| 507 | + "Please retry by specifying the path to your target zuliprc file." |
| 508 | + ) |
| 509 | + |
| 510 | + |
| 511 | +@pytest.mark.parametrize( |
| 512 | + "prefix, expected_str", |
| 513 | + [ |
| 514 | + case( |
| 515 | + "or", |
| 516 | + "Organization name prefix must be at least 3 characters long", |
| 517 | + id="too_short", |
| 518 | + ), |
| 519 | + case( |
| 520 | + "org", |
| 521 | + "Found multiple organizations starting with 'org':\n" |
| 522 | + "config/org1\nconfig/org2\n", |
| 523 | + id="multiple_dirs", |
| 524 | + ), |
| 525 | + case( |
| 526 | + "org3", |
| 527 | + "Could not find any organizations starting with 'org3'", |
| 528 | + id="no_dirs", |
| 529 | + ), |
| 530 | + case( |
| 531 | + "org2", |
| 532 | + "Could not find any zuliprc files at:\nconfig/org2\n", |
| 533 | + id="no_zuliprc", |
| 534 | + ), |
| 535 | + ], |
| 536 | +) |
| 537 | +def test_path_to_realm_zuliprc( |
| 538 | + mocker: MockerFixture, |
| 539 | + mock_config_dir_with_multiple_zuliprc: Path, |
| 540 | + prefix: str, |
| 541 | + expected_str: str, |
| 542 | +) -> None: |
| 543 | + config_dir = mock_config_dir_with_multiple_zuliprc |
| 544 | + mocker.patch("zulipterminal.cli.run.ZULIP_CONFIG_PATH", str(config_dir)) |
| 545 | + mock_exit_with_error = mocker.patch( |
| 546 | + "zulipterminal.cli.run.exit_with_error", side_effect=SystemExit(1) |
| 547 | + ) |
| 548 | + expected_str = expected_str.replace("config", str(config_dir)) |
| 549 | + |
| 550 | + with pytest.raises(SystemExit): |
| 551 | + path_to_realm_zuliprc(prefix) |
| 552 | + |
| 553 | + mock_exit_with_error.assert_called_once_with(expected_str) |
| 554 | + |
| 555 | + |
413 | 556 | @pytest.fixture
|
414 | 557 | def parameterized_zuliprc(tmp_path: Path) -> Callable[[Dict[str, str]], str]:
|
415 | 558 | def func(config: Dict[str, str]) -> str:
|
|
0 commit comments