diff --git a/docs/cli.md b/docs/cli.md index 879664918fe..bf323592676 100644 --- a/docs/cli.md +++ b/docs/cli.md @@ -272,6 +272,21 @@ When `--only` is specified, `--with` and `--without` options are ignored. {{% /note %}} +## sync + +The `sync` command makes sure that the project's environment is in sync with the `poetry.lock` file. +It is equivalent to running `poetry install --sync` and provides the same options +(except for `--sync`) as [install]({{< relref "#install" >}}). + +{{% note %}} +Normally, you should prefer `poetry sync` to `poetry install` to avoid untracked outdated packages. +However, if you have set `virtualenvs.create = false` to install dependencies into your system environment, +which is discouraged, or `virtualenvs.options.system-site-packages = true` to make +system site-packages available in your virtual environment, you should use `poetry install` +because `poetry sync` will normally not work well in these cases. +{{% /note %}} + + ## update In order to get the latest versions of the dependencies and to update the `poetry.lock` file, diff --git a/src/poetry/console/application.py b/src/poetry/console/application.py index e2d79c449ae..91633c58d42 100644 --- a/src/poetry/console/application.py +++ b/src/poetry/console/application.py @@ -63,6 +63,7 @@ def _load() -> Command: "run", "search", "show", + "sync", "update", "version", # Cache commands diff --git a/src/poetry/console/commands/sync.py b/src/poetry/console/commands/sync.py new file mode 100644 index 00000000000..29d6c1be873 --- /dev/null +++ b/src/poetry/console/commands/sync.py @@ -0,0 +1,36 @@ +from __future__ import annotations + +from typing import TYPE_CHECKING +from typing import ClassVar + +from poetry.console.commands.install import InstallCommand + + +if TYPE_CHECKING: + from cleo.io.inputs.option import Option + + +class SyncCommand(InstallCommand): + name = "sync" + description = "Update the project's environment according to the lockfile." + + options: ClassVar[list[Option]] = [ + opt for opt in InstallCommand.options if opt.name != "sync" + ] + + help = """\ +The sync command makes sure that the project's environment is in sync with +the poetry.lock file. +It is equivalent to running poetry install --sync. + +poetry sync + +By default, the above command will also install the current project. To install only the +dependencies and not including the current project, run the command with the +--no-root option like below: + + poetry sync --no-root + +If you want to use Poetry only for dependency management but not for packaging, +you can set the "package-mode" to false in your pyproject.toml file. +""" diff --git a/tests/console/commands/test_install.py b/tests/console/commands/test_install.py index ae163df3ec2..15125be4a23 100644 --- a/tests/console/commands/test_install.py +++ b/tests/console/commands/test_install.py @@ -63,6 +63,11 @@ """ +@pytest.fixture +def command() -> str: + return "install" + + @pytest.fixture def poetry(project_factory: ProjectFactory) -> Poetry: return project_factory(name="export", pyproject_content=PYPROJECT_CONTENT) @@ -70,9 +75,9 @@ def poetry(project_factory: ProjectFactory) -> Poetry: @pytest.fixture def tester( - command_tester_factory: CommandTesterFactory, poetry: Poetry + command_tester_factory: CommandTesterFactory, command: str, poetry: Poetry ) -> CommandTester: - return command_tester_factory("install") + return command_tester_factory(command) def _project_factory( @@ -443,6 +448,7 @@ def test_install_logs_output_decorated( @pytest.mark.parametrize("error", ["module", "readme", ""]) def test_install_warning_corrupt_root( command_tester_factory: CommandTesterFactory, + command: str, project_factory: ProjectFactory, with_root: bool, error: str, @@ -461,7 +467,7 @@ def test_install_warning_corrupt_root( if error != "module": (poetry.pyproject_path.parent / f"{name}.py").touch() - tester = command_tester_factory("install", poetry=poetry) + tester = command_tester_factory(command, poetry=poetry) tester.execute("" if with_root else "--no-root") if error and with_root: @@ -481,6 +487,7 @@ def test_install_warning_corrupt_root( ) def test_install_path_dependency_does_not_exist( command_tester_factory: CommandTesterFactory, + command: str, project_factory: ProjectFactory, fixture_dir: FixtureDirGetter, project: str, @@ -489,7 +496,7 @@ def test_install_path_dependency_does_not_exist( poetry = _project_factory(project, project_factory, fixture_dir) assert isinstance(poetry.locker, TestLocker) poetry.locker.locked(True) - tester = command_tester_factory("install", poetry=poetry) + tester = command_tester_factory(command, poetry=poetry) if options: tester.execute(options) else: @@ -500,6 +507,7 @@ def test_install_path_dependency_does_not_exist( @pytest.mark.parametrize("options", ["", "--extras notinstallable"]) def test_install_extra_path_dependency_does_not_exist( command_tester_factory: CommandTesterFactory, + command: str, project_factory: ProjectFactory, fixture_dir: FixtureDirGetter, options: str, @@ -508,7 +516,7 @@ def test_install_extra_path_dependency_does_not_exist( poetry = _project_factory(project, project_factory, fixture_dir) assert isinstance(poetry.locker, TestLocker) poetry.locker.locked(True) - tester = command_tester_factory("install", poetry=poetry) + tester = command_tester_factory(command, poetry=poetry) if not options: tester.execute(options) else: @@ -519,6 +527,7 @@ def test_install_extra_path_dependency_does_not_exist( @pytest.mark.parametrize("options", ["", "--no-directory"]) def test_install_missing_directory_dependency_with_no_directory( command_tester_factory: CommandTesterFactory, + command: str, project_factory: ProjectFactory, fixture_dir: FixtureDirGetter, options: str, @@ -528,7 +537,7 @@ def test_install_missing_directory_dependency_with_no_directory( ) assert isinstance(poetry.locker, TestLocker) poetry.locker.locked(True) - tester = command_tester_factory("install", poetry=poetry) + tester = command_tester_factory(command, poetry=poetry) if options: tester.execute(options) else: @@ -538,6 +547,7 @@ def test_install_missing_directory_dependency_with_no_directory( def test_non_package_mode_does_not_try_to_install_root( command_tester_factory: CommandTesterFactory, + command: str, project_factory: ProjectFactory, ) -> None: content = """\ @@ -546,7 +556,7 @@ def test_non_package_mode_does_not_try_to_install_root( """ poetry = project_factory(name="non-package-mode", pyproject_content=content) - tester = command_tester_factory("install", poetry=poetry) + tester = command_tester_factory(command, poetry=poetry) tester.execute() assert tester.status_code == 0 diff --git a/tests/console/commands/test_sync.py b/tests/console/commands/test_sync.py new file mode 100644 index 00000000000..af75afd86f3 --- /dev/null +++ b/tests/console/commands/test_sync.py @@ -0,0 +1,30 @@ +from __future__ import annotations + +from typing import TYPE_CHECKING + +import pytest + +from cleo.exceptions import CleoNoSuchOptionError + +# import all tests from the install command +# and run them for sync by overriding the command fixture +from tests.console.commands.test_install import * # noqa: F403 + + +if TYPE_CHECKING: + from cleo.testers.command_tester import CommandTester + + +@pytest.fixture # type: ignore[no-redef] +def command() -> str: + return "sync" + + +@pytest.mark.skip("Only relevant for `poetry install`") # type: ignore[no-redef] +def test_sync_option_is_passed_to_the_installer() -> None: + """The only test from the install command that does not work for sync.""" + + +def test_sync_option_not_available(tester: CommandTester) -> None: + with pytest.raises(CleoNoSuchOptionError): + tester.execute("--sync")