diff --git a/conda_build/build.py b/conda_build/build.py index 45f64995f2..28ffc04a70 100644 --- a/conda_build/build.py +++ b/conda_build/build.py @@ -2308,7 +2308,7 @@ def create_build_envs(m: MetaData, notest): m.config._merge_build_host = m.build_is_host if m.is_cross and not m.build_is_host: - host_actions = environ.get_install_actions( + host_precs = environ.get_package_records( m.config.host_prefix, tuple(host_ms_deps), "host", @@ -2325,7 +2325,7 @@ def create_build_envs(m: MetaData, notest): ) environ.create_env( m.config.host_prefix, - host_actions, + host_precs, env="host", config=m.config, subdir=m.config.host_subdir, @@ -2334,7 +2334,7 @@ def create_build_envs(m: MetaData, notest): ) if m.build_is_host: build_ms_deps.extend(host_ms_deps) - build_actions = environ.get_install_actions( + build_precs = environ.get_package_records( m.config.build_prefix, tuple(build_ms_deps), "build", @@ -2360,7 +2360,7 @@ def create_build_envs(m: MetaData, notest): *utils.ensure_list(m.get_value("requirements/run", [])), ] # make sure test deps are available before taking time to create build env - environ.get_install_actions( + environ.get_package_records( m.config.test_prefix, tuple(test_run_ms_deps), "test", @@ -2397,7 +2397,7 @@ def create_build_envs(m: MetaData, notest): ): environ.create_env( m.config.build_prefix, - build_actions, + build_precs, env="build", config=m.config, subdir=m.config.build_subdir, @@ -2435,8 +2435,8 @@ def build( return default_return log = utils.get_logger(__name__) - host_actions = [] - build_actions = [] + host_precs = [] + build_precs = [] output_metas = [] with utils.path_prepended(m.config.build_prefix): @@ -2779,7 +2779,7 @@ def build( host_ms_deps = m.ms_depends("host") sub_build_ms_deps = m.ms_depends("build") if m.is_cross and not m.build_is_host: - host_actions = environ.get_install_actions( + host_precs = environ.get_package_records( m.config.host_prefix, tuple(host_ms_deps), "host", @@ -2796,7 +2796,7 @@ def build( ) environ.create_env( m.config.host_prefix, - host_actions, + host_precs, env="host", config=m.config, subdir=subdir, @@ -2806,7 +2806,7 @@ def build( else: # When not cross-compiling, the build deps aggregate 'build' and 'host'. sub_build_ms_deps.extend(host_ms_deps) - build_actions = environ.get_install_actions( + build_precs = environ.get_package_records( m.config.build_prefix, tuple(sub_build_ms_deps), "build", @@ -2823,7 +2823,7 @@ def build( ) environ.create_env( m.config.build_prefix, - build_actions, + build_precs, env="build", config=m.config, subdir=m.config.build_subdir, @@ -3481,7 +3481,7 @@ def test( utils.rm_rf(metadata.config.test_prefix) try: - actions = environ.get_install_actions( + precs = environ.get_package_records( metadata.config.test_prefix, tuple(specs), "host", @@ -3523,7 +3523,7 @@ def test( with env_var("CONDA_PATH_CONFLICT", conflict_verbosity, reset_context): environ.create_env( metadata.config.test_prefix, - actions, + precs, config=metadata.config, env="host", subdir=subdir, @@ -3819,7 +3819,7 @@ def build_tree( with TemporaryDirectory( prefix="_", suffix=r_string ) as tmpdir: - actions = environ.get_install_actions( + precs = environ.get_package_records( tmpdir, specs, env="run", @@ -3839,7 +3839,7 @@ def build_tree( # make sure to download that package to the local cache if not there local_file = execute_download_actions( meta, - actions, + precs, "host", package_subset=[dep], require_files=True, diff --git a/conda_build/environ.py b/conda_build/environ.py index f9397355fc..10a77daa2f 100644 --- a/conda_build/environ.py +++ b/conda_build/environ.py @@ -34,6 +34,7 @@ LockError, MatchSpec, NoPackagesFoundError, + PackageRecord, PaddingError, ProgressiveFetchExtract, TemporaryDirectory, @@ -45,6 +46,7 @@ reset_context, root_dir, ) +from .deprecations import deprecated from .exceptions import BuildLockError, DependencyNeedsBuildingError from .features import feature_list from .index import get_build_index @@ -851,6 +853,7 @@ def package_specs(self): last_index_ts = 0 +@deprecated("24.1.0", "24.3.0", addendum="Use `get_package_records` instead.") def get_install_actions( prefix, specs, @@ -1001,9 +1004,44 @@ def get_install_actions( return actions +def get_package_records( + prefix, + specs, + env, + retries=0, + subdir=None, + verbose=True, + debug=False, + locking=True, + bldpkgs_dirs=None, + timeout=900, + disable_pip=False, + max_env_retry=3, + output_folder=None, + channel_urls=None, +): + return get_install_actions( + prefix=prefix, + specs=specs, + env=env, + retries=retries, + subdir=subdir, + verbose=verbose, + debug=debug, + locking=locking, + bldpkgs_dirs=bldpkgs_dirs, + timeout=timeout, + disable_pip=disable_pip, + max_env_retry=max_env_retry, + output_folder=output_folder, + channel_urls=channel_urls, + ).get(LINK_ACTION, []) + + +@deprecated.argument("24.1.0", "24.3.0", "specs_or_actions", rename="specs_or_precs") def create_env( prefix, - specs_or_actions, + specs_or_precs, env, config, subdir, @@ -1031,17 +1069,20 @@ def create_env( # if os.path.isdir(prefix): # utils.rm_rf(prefix) - if specs_or_actions: # Don't waste time if there is nothing to do + if specs_or_precs: # Don't waste time if there is nothing to do log.debug("Creating environment in %s", prefix) - log.debug(str(specs_or_actions)) + log.debug(str(specs_or_precs)) if not locks: locks = utils.get_conda_operation_locks(config) try: with utils.try_acquire_locks(locks, timeout=config.timeout): - # input is a list - it's specs in MatchSpec format - if not hasattr(specs_or_actions, "keys"): - specs = list(set(specs_or_actions)) + # input is a list of specs in MatchSpec format + if not ( + hasattr(specs_or_precs, "keys") + or isinstance(specs_or_precs[0], PackageRecord) + ): + specs = list(set(specs_or_precs)) actions = get_install_actions( prefix, tuple(specs), @@ -1058,7 +1099,10 @@ def create_env( channel_urls=tuple(config.channel_urls), ) else: - actions = specs_or_actions + if not hasattr(specs_or_precs, "keys"): + actions = {LINK_ACTION: specs_or_precs} + else: + actions = specs_or_precs index, _, _ = get_build_index( subdir=subdir, bldpkgs_dir=config.bldpkgs_dir, @@ -1070,13 +1114,13 @@ def create_env( timeout=config.timeout, ) utils.trim_empty_keys(actions) - _display_actions(actions) + _display_actions(prefix, actions) if utils.on_win: for k, v in os.environ.items(): os.environ[k] = str(v) with env_var("CONDA_QUIET", not config.verbose, reset_context): with env_var("CONDA_JSON", not config.verbose, reset_context): - _execute_actions(actions) + _execute_actions(prefix, actions) except ( SystemExit, PaddingError, @@ -1136,7 +1180,7 @@ def create_env( ) create_env( prefix, - specs_or_actions, + specs_or_precs, config=config, subdir=subdir, env=env, @@ -1167,7 +1211,7 @@ def create_env( ) create_env( prefix, - specs_or_actions, + specs_or_precs, config=config, subdir=subdir, env=env, @@ -1205,7 +1249,7 @@ def create_env( ) create_env( prefix, - specs_or_actions, + specs_or_precs, config=config, subdir=subdir, env=env, @@ -1314,12 +1358,11 @@ def install_actions(prefix, index, specs): del install_actions -def _execute_actions(actions): +def _execute_actions(prefix, actions): # This is copied over from https://github.com/conda/conda/blob/23.11.0/conda/plan.py#L575 # but reduced to only the functionality actually used within conda-build. - assert PREFIX_ACTION in actions and actions[PREFIX_ACTION] - prefix = actions[PREFIX_ACTION] + assert prefix if LINK_ACTION not in actions: log.debug(f"action {LINK_ACTION} not in actions") @@ -1348,11 +1391,10 @@ def _execute_actions(actions): unlink_link_transaction.execute() -def _display_actions(actions): +def _display_actions(prefix, actions): # This is copied over from https://github.com/conda/conda/blob/23.11.0/conda/plan.py#L58 # but reduced to only the functionality actually used within conda-build. - prefix = actions.get(PREFIX_ACTION) builder = ["", "## Package Plan ##\n"] if prefix: builder.append(" environment location: %s" % prefix) diff --git a/conda_build/index.py b/conda_build/index.py index aebc28fe21..ba23eaf19e 100644 --- a/conda_build/index.py +++ b/conda_build/index.py @@ -61,6 +61,7 @@ human_bytes, url_path, ) +from .deprecations import deprecated from .utils import ( CONDA_PACKAGE_EXTENSION_V1, CONDA_PACKAGE_EXTENSION_V2, @@ -112,7 +113,8 @@ def map(self, func, *iterables): local_subdir = "" local_output_folder = "" cached_channels = [] -channel_data = {} +_channel_data = {} +deprecated.constant("24.1.0", "24.3.0", "channel_data", _channel_data) # TODO: support for libarchive seems to have broken ability to use multiple threads here. @@ -151,7 +153,7 @@ def get_build_index( global local_output_folder global cached_index global cached_channels - global channel_data + global _channel_data mtime = 0 channel_urls = list(utils.ensure_list(channel_urls)) @@ -248,7 +250,7 @@ def get_build_index( while retry < max_retries: try: with open(channeldata_file, "r+") as f: - channel_data[channel.name] = json.load(f) + _channel_data[channel.name] = json.load(f) break except (OSError, JSONDecodeError): time.sleep(0.2) @@ -257,24 +259,24 @@ def get_build_index( # download channeldata.json for url if not context.offline: try: - channel_data[channel.name] = utils.download_channeldata( + _channel_data[channel.name] = utils.download_channeldata( channel.base_url + "/channeldata.json" ) except CondaHTTPError: continue # collapse defaults metachannel back into one superchannel, merging channeldata - if channel.base_url in context.default_channels and channel_data.get( + if channel.base_url in context.default_channels and _channel_data.get( channel.name ): packages = superchannel.get("packages", {}) - packages.update(channel_data[channel.name]) + packages.update(_channel_data[channel.name]) superchannel["packages"] = packages - channel_data["defaults"] = superchannel + _channel_data["defaults"] = superchannel local_index_timestamp = os.path.getmtime(index_file) local_subdir = subdir local_output_folder = output_folder cached_channels = channel_urls - return cached_index, local_index_timestamp, channel_data + return cached_index, local_index_timestamp, _channel_data def _ensure_valid_channel(local_folder, subdir): @@ -366,11 +368,11 @@ def _make_seconds(timestamp): # ========================================================================== -REPODATA_VERSION = 1 -CHANNELDATA_VERSION = 1 -REPODATA_JSON_FN = "repodata.json" -REPODATA_FROM_PKGS_JSON_FN = "repodata_from_packages.json" -CHANNELDATA_FIELDS = ( +_REPODATA_VERSION = 1 +_CHANNELDATA_VERSION = 1 +_REPODATA_JSON_FN = "repodata.json" +_REPODATA_FROM_PKGS_JSON_FN = "repodata_from_packages.json" +_CHANNELDATA_FIELDS = ( "description", "dev_url", "doc_url", @@ -401,6 +403,13 @@ def _make_seconds(timestamp): "recipe_origin", "commits", ) +deprecated.constant("24.1.0", "24.3.0", "REPODATA_VERSION", _REPODATA_VERSION) +deprecated.constant("24.1.0", "24.3.0", "CHANNELDATA_VERSION", _CHANNELDATA_VERSION) +deprecated.constant("24.1.0", "24.3.0", "REPODATA_JSON_FN", _REPODATA_JSON_FN) +deprecated.constant( + "24.1.0", "24.3.0", "REPODATA_FROM_PKGS_JSON_FN", _REPODATA_FROM_PKGS_JSON_FN +) +deprecated.constant("24.1.0", "24.3.0", "CHANNELDATA_FIELDS", _CHANNELDATA_FIELDS) def _clear_newline_chars(record, field_name): @@ -412,6 +421,9 @@ def _clear_newline_chars(record, field_name): record[field_name] = record[field_name][0].strip().replace("\n", " ") +@deprecated( + "24.1.0", "24.5.0", addendum="Use `conda_index._apply_instructions` instead." +) def _apply_instructions(subdir, repodata, instructions): repodata.setdefault("removed", []) utils.merge_or_update_dict( @@ -871,6 +883,7 @@ def _build_current_repodata(subdir, repodata, pins): return new_repodata +@deprecated("24.1.0", "24.3.0") class ChannelIndex: def __init__( self, @@ -951,7 +964,7 @@ def index( self._write_repodata( subdir, repodata_from_packages, - REPODATA_FROM_PKGS_JSON_FN, + _REPODATA_FROM_PKGS_JSON_FN, ) # Step 3. Apply patch instructions. @@ -968,7 +981,7 @@ def index( t2.set_description("Writing patched repodata") t2.update() self._write_repodata( - subdir, patched_repodata, REPODATA_JSON_FN + subdir, patched_repodata, _REPODATA_JSON_FN ) t2.set_description("Building current_repodata subset") t2.update() @@ -1000,7 +1013,7 @@ def index( def index_subdir(self, subdir, index_file=None, verbose=False, progress=False): subdir_path = join(self.channel_root, subdir) self._ensure_dirs(subdir) - repodata_json_path = join(subdir_path, REPODATA_FROM_PKGS_JSON_FN) + repodata_json_path = join(subdir_path, _REPODATA_FROM_PKGS_JSON_FN) if verbose: log.info("Building repodata for %s" % subdir_path) @@ -1158,7 +1171,7 @@ def index_subdir(self, subdir, index_file=None, verbose=False, progress=False): "info": { "subdir": subdir, }, - "repodata_version": REPODATA_VERSION, + "repodata_version": _REPODATA_VERSION, "removed": sorted(list(ignore_set)), } finally: @@ -1465,11 +1478,11 @@ def _add_extra_path(extra_paths, path): } extra_paths = OrderedDict() - _add_extra_path(extra_paths, join(subdir_path, REPODATA_JSON_FN)) - _add_extra_path(extra_paths, join(subdir_path, REPODATA_JSON_FN + ".bz2")) - _add_extra_path(extra_paths, join(subdir_path, REPODATA_FROM_PKGS_JSON_FN)) + _add_extra_path(extra_paths, join(subdir_path, _REPODATA_JSON_FN)) + _add_extra_path(extra_paths, join(subdir_path, _REPODATA_JSON_FN + ".bz2")) + _add_extra_path(extra_paths, join(subdir_path, _REPODATA_FROM_PKGS_JSON_FN)) _add_extra_path( - extra_paths, join(subdir_path, REPODATA_FROM_PKGS_JSON_FN + ".bz2") + extra_paths, join(subdir_path, _REPODATA_FROM_PKGS_JSON_FN + ".bz2") ) # _add_extra_path(extra_paths, join(subdir_path, "repodata2.json")) _add_extra_path(extra_paths, join(subdir_path, "patch_instructions.json")) @@ -1603,7 +1616,7 @@ def _replace_if_newer_and_present(pd, data, erec, data_newer, k): channel_data.update( { - "channeldata_version": CHANNELDATA_VERSION, + "channeldata_version": _CHANNELDATA_VERSION, "subdirs": sorted( list(set(channel_data.get("subdirs", []) + [subdir])) ), diff --git a/conda_build/render.py b/conda_build/render.py index c75838a65b..a46130f4ed 100644 --- a/conda_build/render.py +++ b/conda_build/render.py @@ -35,6 +35,7 @@ pkgs_dirs, specs_from_url, ) +from .deprecations import deprecated from .environ import LINK_ACTION from .exceptions import DependencyNeedsBuildingError from .index import get_build_index @@ -90,6 +91,7 @@ def bldpkg_path(m): return path +@deprecated("24.1.0", "24.3.0") def actions_to_pins(actions): if LINK_ACTION in actions: return [package_record_to_requirement(prec) for prec in actions[LINK_ACTION]] @@ -182,7 +184,9 @@ def get_env_dependencies( else: raise - specs = actions_to_pins(actions) + specs = [ + package_record_to_requirement(prec) for prec in actions.get(LINK_ACTION, []) + ] return ( utils.ensure_list( (specs + subpackages + pass_through_deps) @@ -325,7 +329,8 @@ def _read_specs_from_package(pkg_loc, pkg_dist): return specs -def execute_download_actions(m, actions, env, package_subset=None, require_files=False): +@deprecated.argument("24.1.0", "24.3.0", "actions", rename="precs") +def execute_download_actions(m, precs, env, package_subset=None, require_files=False): subdir = getattr(m.config, f"{env}_subdir") index, _, _ = get_build_index( subdir=subdir, @@ -354,7 +359,8 @@ def execute_download_actions(m, actions, env, package_subset=None, require_files pkg_files = {} - precs = actions.get(LINK_ACTION, []) + if hasattr(precs, "keys"): + precs = precs.get(LINK_ACTION, []) if isinstance(package_subset, PackageRecord): package_subset = [package_subset] else: @@ -403,18 +409,20 @@ def execute_download_actions(m, actions, env, package_subset=None, require_files return pkg_files -def get_upstream_pins(m: MetaData, actions, env): +@deprecated.argument("24.1.0", "24.3.0", "actions", rename="precs") +def get_upstream_pins(m: MetaData, precs, env): """Download packages from specs, then inspect each downloaded package for additional downstream dependency specs. Return these additional specs.""" env_specs = m.get_value(f"requirements/{env}", []) explicit_specs = [req.split(" ")[0] for req in env_specs] if env_specs else [] - linked_packages = actions.get(LINK_ACTION, []) - linked_packages = [prec for prec in linked_packages if prec.name in explicit_specs] + if hasattr(precs, "keys"): + precs = precs.get(LINK_ACTION, []) + precs = [prec for prec in precs if prec.name in explicit_specs] ignore_pkgs_list = utils.ensure_list(m.get_value("build/ignore_run_exports_from")) ignore_list = utils.ensure_list(m.get_value("build/ignore_run_exports")) additional_specs = {} - for prec in linked_packages: + for prec in precs: if any(prec.name in req.split(" ")[0] for req in ignore_pkgs_list): continue run_exports = None @@ -428,7 +436,7 @@ def get_upstream_pins(m: MetaData, actions, env): if run_exports is None: loc, dist = execute_download_actions( m, - actions, + precs, env=env, package_subset=[prec], )[prec]