Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Config clean command sketch #17514

Merged
merged 23 commits into from
Jan 21, 2025
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 18 additions & 9 deletions conan/api/conan_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
from conan.api.subapi.local import LocalAPI
from conan.api.subapi.lockfile import LockfileAPI
from conan.api.subapi.workspace import WorkspaceAPI
from conan import conan_version
from conan.api.subapi.config import ConfigAPI
from conan.api.subapi.download import DownloadAPI
from conan.api.subapi.export import ExportAPI
Expand All @@ -19,7 +18,6 @@
from conan.api.subapi.remove import RemoveAPI
from conan.api.subapi.search import SearchAPI
from conan.api.subapi.upload import UploadAPI
from conans.client.migrations import ClientMigrator
from conan.errors import ConanException
from conan.internal.paths import get_conan_user_home
from conans.model.version_range import validate_conan_version
Expand All @@ -37,11 +35,10 @@ def __init__(self, cache_folder=None):
self.cache_folder = self.workspace.home_folder() or cache_folder or get_conan_user_home()
self.home_folder = self.cache_folder # Lets call it home, deprecate "cache"

# Migration system
migrator = ClientMigrator(self.cache_folder, conan_version)
migrator.migrate()

# This API is depended upon by the subsequent ones, it should be initialized first
self.config = ConfigAPI(self)
self.config.migrate()

self.remotes = RemotesAPI(self)
self.command = CommandAPI(self)
# Search recipes by wildcard and packages filtering by configuration
Expand All @@ -60,6 +57,18 @@ def __init__(self, cache_folder=None):
self.lockfile = LockfileAPI(self)
self.local = LocalAPI(self)

required_range_new = self.config.global_conf.get("core:required_conan_version")
if required_range_new:
validate_conan_version(required_range_new)
_check_conan_version(self)

def reinit(self):
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
self.config.reinit()
self.config.migrate()
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
self.remotes.reinit()
self.local.reinit()

_check_conan_version(self)


def _check_conan_version(conan_api):
required_range_new = conan_api.config.global_conf.get("core:required_conan_version")
if required_range_new:
validate_conan_version(required_range_new)
26 changes: 25 additions & 1 deletion conan/api/subapi/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
from conans.model.pkg_type import PackageType
from conans.model.recipe_ref import RecipeReference
from conans.model.settings import Settings
from conans.util.files import load, save
from conans.util.files import load, save, rmdir, remove


class ConfigAPI:
Expand Down Expand Up @@ -191,3 +191,27 @@ def appending_recursive_dict_update(d, u):
appending_recursive_dict_update(settings, settings_user)

return Settings(settings)

def migrate(self):
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
# Migration system
# TODO: A prettier refactoring of migrators would be nice
from conans.client.migrations import ClientMigrator
migrator = ClientMigrator(self.home(), conan_version)
migrator.migrate()

def clean(self):
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
contents = os.listdir(self.home())
for content in contents:
# keep packages
if content not in ("p", "version.txt"):
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved
content_path = os.path.join(self.home(), content)
ConanOutput().debug(f"Removing {content_path}")
if os.path.isdir(content_path):
rmdir(content_path)
else:
remove(content_path)
# CHECK: This also generates a remotes.json that is not there after a conan profile show?
self.migrate()

def reinit(self):
self._new_config = None
4 changes: 4 additions & 0 deletions conan/api/subapi/local.py
Original file line number Diff line number Diff line change
Expand Up @@ -118,3 +118,7 @@ def inspect(self, conanfile_path, remotes, lockfile, name=None, version=None, us
conanfile = app.loader.load_named(conanfile_path, name=name, version=version, user=user,
channel=channel, remotes=remotes, graph_lock=lockfile)
return conanfile

def reinit(self):
# TODO: Check if this is risky
self.editable_packages = EditablePackages(self._conan_api.home_folder)
3 changes: 3 additions & 0 deletions conan/api/subapi/remotes.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,9 @@ def __init__(self, conan_api):
# Wraps an http_requester to inject proxies, certs, etc
self._requester = ConanRequester(self.conan_api.config.global_conf, self.conan_api.cache_folder)

def reinit(self):
self._requester = ConanRequester(self.conan_api.config.global_conf, self.conan_api.cache_folder)
AbrilRBS marked this conversation as resolved.
Show resolved Hide resolved

def list(self, pattern=None, only_enabled=True):
"""
Obtain a list of ``Remote`` objects matching the pattern.
Expand Down
10 changes: 10 additions & 0 deletions conan/cli/commands/config.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
from conan.api.input import UserInput
from conan.api.model import Remote
from conan.api.output import cli_out_write
from conan.cli.command import conan_command, conan_subcommand, OnceArgument
Expand Down Expand Up @@ -132,3 +133,12 @@ def config_show(conan_api, parser, subparser, *args):
args = parser.parse_args(*args)

return conan_api.config.show(args.pattern)


@conan_subcommand()
def config_clean(conan_api, parser, subparser, *args):
"""
Clean the configuration files in the Conan home folder. (Keeping installed packages)
"""
parser.parse_args(*args)
conan_api.config.clean()
40 changes: 40 additions & 0 deletions test/integration/command/config_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -201,3 +201,43 @@ def test_config_show():
tc.run("config show zlib/*:foo")
assert "zlib/*:user.mycategory:foo" in tc.out
assert "zlib/*:user.myothercategory:foo" in tc.out


def test_config_clean():
tc = TestClient(light=True)
tc.run("profile detect --name=foo")
tc.run("remote add bar http://fakeurl")
tc.save_home({"global.conf": "core.upload:retry=7\n",
"extensions/compatibility/mycomp.py": "",
"extensions/commands/cmd_foo.py": "",
})
tc.run("config clean")
tc.run("profile list")
assert "foo" not in tc.out
tc.run("remote list")
assert "bar" not in tc.out
tc.run("config show core.upload:retry")
assert "7" not in tc.out
assert not os.path.exists(os.path.join(tc.cache_folder, "extensions"))


def test_config_reinit():
custom_global_conf = "core.upload:retry=7"
global_conf_folder = temp_folder()
with open(os.path.join(global_conf_folder, "global.conf"), "w") as f:
f.write(custom_global_conf)

cache_folder = temp_folder()
conan_api = ConanAPI(cache_folder=cache_folder)
# Ensure reinitialization does not invalidate references
config_api = conan_api.config
assert config_api.global_conf.get("core.upload:retry", check_type=int) != 7

conan_api.config.install(global_conf_folder, verify_ssl=False)

# No effect yet, we haven't reinitialized the API after the config installation
assert config_api.global_conf.get("core.upload:retry", check_type=int) != 7

conan_api.reinit()

assert config_api.global_conf.get("core.upload:retry", check_type=int) == 7
Loading