From 5e70be84e01d8859d0d3c0767413248f1c3e0186 Mon Sep 17 00:00:00 2001 From: Chris DeCarolis Date: Mon, 16 Dec 2024 15:45:02 -0800 Subject: [PATCH] Split loader file into multiple different files --- .../load_asset_checks_from_modules.py | 6 +- .../load_assets_from_modules.py | 299 +----------------- .../definitions/module_loaders/object_list.py | 230 ++++++++++++++ .../_core/definitions/module_loaders/utils.py | 99 ++++++ 4 files changed, 338 insertions(+), 296 deletions(-) create mode 100644 python_modules/dagster/dagster/_core/definitions/module_loaders/object_list.py create mode 100644 python_modules/dagster/dagster/_core/definitions/module_loaders/utils.py diff --git a/python_modules/dagster/dagster/_core/definitions/module_loaders/load_asset_checks_from_modules.py b/python_modules/dagster/dagster/_core/definitions/module_loaders/load_asset_checks_from_modules.py index 13389bb51a491..ce1ab2bd9a9ec 100644 --- a/python_modules/dagster/dagster/_core/definitions/module_loaders/load_asset_checks_from_modules.py +++ b/python_modules/dagster/dagster/_core/definitions/module_loaders/load_asset_checks_from_modules.py @@ -9,10 +9,8 @@ check_opt_coercible_to_asset_key_prefix_param, ) from dagster._core.definitions.assets import AssetsDefinition -from dagster._core.definitions.module_loaders.load_assets_from_modules import ( - LoadedAssetsList, - find_modules_in_package, -) +from dagster._core.definitions.module_loaders.object_list import LoadedAssetsList +from dagster._core.definitions.module_loaders.utils import find_modules_in_package def load_asset_checks_from_modules( diff --git a/python_modules/dagster/dagster/_core/definitions/module_loaders/load_assets_from_modules.py b/python_modules/dagster/dagster/_core/definitions/module_loaders/load_assets_from_modules.py index 603271f29fe7b..319f329fe3735 100644 --- a/python_modules/dagster/dagster/_core/definitions/module_loaders/load_assets_from_modules.py +++ b/python_modules/dagster/dagster/_core/definitions/module_loaders/load_assets_from_modules.py @@ -1,28 +1,11 @@ import inspect -import pkgutil -from collections import defaultdict -from functools import cached_property from importlib import import_module from types import ModuleType -from typing import ( - Callable, - Dict, - Iterable, - Iterator, - Mapping, - Optional, - Sequence, - Tuple, - Type, - Union, - cast, -) +from typing import Iterable, Iterator, Optional, Sequence, Tuple, Type, Union import dagster._check as check -from dagster._core.definitions.asset_checks import AssetChecksDefinition, has_only_asset_checks +from dagster._core.definitions.asset_checks import has_only_asset_checks from dagster._core.definitions.asset_key import ( - AssetCheckKey, - AssetKey, CoercibleToAssetKeyPrefix, check_opt_coercible_to_asset_key_prefix_param, ) @@ -35,9 +18,13 @@ AutomationCondition, ) from dagster._core.definitions.freshness_policy import FreshnessPolicy +from dagster._core.definitions.module_loaders.object_list import LoadedAssetsList +from dagster._core.definitions.module_loaders.utils import ( + LoadableAssetTypes, + find_modules_in_package, +) from dagster._core.definitions.source_asset import SourceAsset from dagster._core.definitions.utils import resolve_automation_condition -from dagster._core.errors import DagsterInvalidDefinitionError def find_objects_in_module_of_types( @@ -64,109 +51,6 @@ def find_subclasses_in_module( yield value -LoadableAssetTypes = Union[AssetsDefinition, AssetSpec, SourceAsset, CacheableAssetsDefinition] -KeyScopedAssetObjects = (AssetsDefinition, AssetSpec, SourceAsset) - - -class LoadedAssetsList: - def __init__( - self, - assets_per_module: Mapping[str, Sequence[LoadableAssetTypes]], - ): - self.assets_per_module = assets_per_module - self._do_collision_detection() - - @classmethod - def from_modules(cls, modules: Iterable[ModuleType]) -> "LoadedAssetsList": - return cls( - { - module.__name__: list( - find_objects_in_module_of_types( - module, - (AssetsDefinition, SourceAsset, CacheableAssetsDefinition, AssetSpec), - ) - ) - for module in modules - }, - ) - - @cached_property - def flat_object_list(self) -> Sequence[LoadableAssetTypes]: - return [ - asset_object for objects in self.assets_per_module.values() for asset_object in objects - ] - - @cached_property - def objects_by_id(self) -> Dict[int, LoadableAssetTypes]: - return {id(asset_object): asset_object for asset_object in self.flat_object_list} - - @cached_property - def deduped_objects(self) -> Sequence[LoadableAssetTypes]: - return list(self.objects_by_id.values()) - - @cached_property - def assets_defs(self) -> Sequence[AssetsDefinition]: - return [asset for asset in self.deduped_objects if isinstance(asset, AssetsDefinition)] - - @cached_property - def source_assets(self) -> Sequence[SourceAsset]: - return [asset for asset in self.deduped_objects if isinstance(asset, SourceAsset)] - - @cached_property - def module_name_by_id(self) -> Dict[int, str]: - return { - id(asset_object): module_name - for module_name, objects in self.assets_per_module.items() - for asset_object in objects - } - - @cached_property - def objects_by_key(self) -> Mapping[AssetKey, Sequence[Union[SourceAsset, AssetsDefinition]]]: - objects_by_key = defaultdict(list) - for asset_object in self.flat_object_list: - if not isinstance(asset_object, KeyScopedAssetObjects): - continue - for key in key_iterator(asset_object): - objects_by_key[key].append(asset_object) - return objects_by_key - - def _do_collision_detection(self) -> None: - for key, asset_objects in self.objects_by_key.items(): - # If there is more than one asset_object in the list for a given key, and the objects do not refer to the same asset_object in memory, we have a collision. - num_distinct_objects_for_key = len( - set(id(asset_object) for asset_object in asset_objects) - ) - if len(asset_objects) > 1 and num_distinct_objects_for_key > 1: - asset_objects_str = ", ".join( - set(self.module_name_by_id[id(asset_object)] for asset_object in asset_objects) - ) - raise DagsterInvalidDefinitionError( - f"Asset key {key.to_user_string()} is defined multiple times. Definitions found in modules: {asset_objects_str}." - ) - - def to_post_load(self) -> "ResolvedAssetObjectList": - return ResolvedAssetObjectList(self.deduped_objects) - - -def key_iterator( - asset: Union[AssetsDefinition, SourceAsset, AssetSpec], included_targeted_keys: bool = False -) -> Iterator[AssetKey]: - return ( - iter( - [ - *asset.keys, - *( - [check_key.asset_key for check_key in asset.check_keys] - if included_targeted_keys - else [] - ), - ] - ) - if isinstance(asset, AssetsDefinition) - else iter([asset.key]) - ) - - def load_assets_from_modules( modules: Iterable[ModuleType], group_name: Optional[str] = None, @@ -380,172 +264,3 @@ def load_assets_from_package_name( source_key_prefix=source_key_prefix, include_specs=include_specs, ) - - -def find_modules_in_package(package_module: ModuleType) -> Iterable[ModuleType]: - yield package_module - if package_module.__file__: - for _, modname, is_pkg in pkgutil.walk_packages( - package_module.__path__, prefix=package_module.__name__ + "." - ): - submodule = import_module(modname) - if is_pkg: - yield from find_modules_in_package(submodule) - else: - yield submodule - else: - raise ValueError( - f"Tried to find modules in package {package_module}, but its __file__ is None" - ) - - -def replace_keys_in_asset( - asset: Union[AssetsDefinition, AssetSpec, SourceAsset], - key_replacements: Mapping[AssetKey, AssetKey], - check_key_replacements: Mapping[AssetCheckKey, AssetCheckKey], -) -> Union[AssetsDefinition, AssetSpec, SourceAsset]: - if isinstance(asset, SourceAsset): - return asset.with_attributes(key=key_replacements.get(asset.key, asset.key)) - if isinstance(asset, AssetSpec): - return asset.replace_attributes( - key=key_replacements.get(asset.key, asset.key), - ) - else: - updated_object = asset.with_attributes( - output_asset_key_replacements={ - key: key_replacements.get(key, key) for key in asset.keys - }, - output_check_key_replacements={ - key: check_key_replacements.get(key, key) for key in asset.check_keys - }, - input_asset_key_replacements={ - key: key_replacements.get(key, key) for key in asset.keys_by_input_name.values() - }, - ) - return updated_object - - -class ResolvedAssetObjectList: - def __init__( - self, - loaded_objects: Sequence[LoadableAssetTypes], - ): - self.loaded_objects = loaded_objects - - @cached_property - def assets_defs_and_specs(self) -> Sequence[Union[AssetsDefinition, AssetSpec]]: - return [ - asset - for asset in self.loaded_objects - if (isinstance(asset, AssetsDefinition) and asset.keys) or isinstance(asset, AssetSpec) - ] - - @cached_property - def assets_defs(self) -> Sequence[AssetsDefinition]: - return [asset for asset in self.loaded_objects if isinstance(asset, AssetsDefinition)] - - @cached_property - def checks_defs(self) -> Sequence[AssetChecksDefinition]: - return [ - cast(AssetChecksDefinition, asset) - for asset in self.loaded_objects - if isinstance(asset, AssetsDefinition) and has_only_asset_checks(asset) - ] - - @cached_property - def assets_defs_specs_and_checks_defs( - self, - ) -> Sequence[Union[AssetsDefinition, AssetSpec, AssetChecksDefinition]]: - return [*self.assets_defs_and_specs, *self.checks_defs] - - @cached_property - def source_assets(self) -> Sequence[SourceAsset]: - return [asset for asset in self.loaded_objects if isinstance(asset, SourceAsset)] - - @cached_property - def cacheable_assets(self) -> Sequence[CacheableAssetsDefinition]: - return [ - asset for asset in self.loaded_objects if isinstance(asset, CacheableAssetsDefinition) - ] - - def get_objects( - self, filter_fn: Callable[[LoadableAssetTypes], bool] - ) -> Sequence[LoadableAssetTypes]: - return [asset for asset in self.loaded_objects if filter_fn(asset)] - - def assets_with_loadable_prefix( - self, key_prefix: CoercibleToAssetKeyPrefix - ) -> "ResolvedAssetObjectList": - # There is a tricky edge case here where if a non-cacheable asset depends on a cacheable asset, - # and the assets are prefixed, the non-cacheable asset's dependency will not be prefixed since - # at prefix-time it is not known that its dependency is one of the cacheable assets. - # https://github.com/dagster-io/dagster/pull/10389#pullrequestreview-1170913271 - result_list = [] - all_asset_keys = { - key - for asset_object in self.assets_defs_specs_and_checks_defs - for key in key_iterator(asset_object, included_targeted_keys=True) - } - all_check_keys = { - check_key for asset_object in self.assets_defs for check_key in asset_object.check_keys - } - - key_replacements = {key: key.with_prefix(key_prefix) for key in all_asset_keys} - check_key_replacements = { - check_key: check_key.with_asset_key_prefix(key_prefix) for check_key in all_check_keys - } - for asset_object in self.loaded_objects: - if isinstance(asset_object, CacheableAssetsDefinition): - result_list.append(asset_object.with_prefix_for_all(key_prefix)) - elif isinstance(asset_object, AssetsDefinition): - result_list.append( - replace_keys_in_asset(asset_object, key_replacements, check_key_replacements) - ) - else: - # We don't replace the key for SourceAssets. - result_list.append(asset_object) - return ResolvedAssetObjectList(result_list) - - def assets_with_source_prefix( - self, key_prefix: CoercibleToAssetKeyPrefix - ) -> "ResolvedAssetObjectList": - result_list = [] - key_replacements = { - source_asset.key: source_asset.key.with_prefix(key_prefix) - for source_asset in self.source_assets - } - for asset_object in self.loaded_objects: - if isinstance(asset_object, KeyScopedAssetObjects): - result_list.append( - replace_keys_in_asset(asset_object, key_replacements, check_key_replacements={}) - ) - else: - result_list.append(asset_object) - return ResolvedAssetObjectList(result_list) - - def with_attributes( - self, - key_prefix: Optional[CoercibleToAssetKeyPrefix], - source_key_prefix: Optional[CoercibleToAssetKeyPrefix], - group_name: Optional[str], - freshness_policy: Optional[FreshnessPolicy], - automation_condition: Optional[AutomationCondition], - backfill_policy: Optional[BackfillPolicy], - ) -> "ResolvedAssetObjectList": - assets_list = self.assets_with_loadable_prefix(key_prefix) if key_prefix else self - assets_list = ( - assets_list.assets_with_source_prefix(source_key_prefix) - if source_key_prefix - else assets_list - ) - return_list = [] - for asset in assets_list.loaded_objects: - return_list.append( - asset.with_attributes( - group_name=group_name, - freshness_policy=freshness_policy, - automation_condition=automation_condition, - backfill_policy=backfill_policy, - ) - ) - return ResolvedAssetObjectList(return_list) diff --git a/python_modules/dagster/dagster/_core/definitions/module_loaders/object_list.py b/python_modules/dagster/dagster/_core/definitions/module_loaders/object_list.py new file mode 100644 index 0000000000000..cc3cf6a1ca6c5 --- /dev/null +++ b/python_modules/dagster/dagster/_core/definitions/module_loaders/object_list.py @@ -0,0 +1,230 @@ +from collections import defaultdict +from functools import cached_property +from types import ModuleType +from typing import Callable, Dict, Iterable, Mapping, Optional, Sequence, Union, cast + +from dagster._core.definitions.asset_checks import AssetChecksDefinition, has_only_asset_checks +from dagster._core.definitions.asset_key import AssetKey, CoercibleToAssetKeyPrefix +from dagster._core.definitions.asset_spec import AssetSpec +from dagster._core.definitions.assets import AssetsDefinition +from dagster._core.definitions.backfill_policy import BackfillPolicy +from dagster._core.definitions.cacheable_assets import CacheableAssetsDefinition +from dagster._core.definitions.declarative_automation.automation_condition import ( + AutomationCondition, +) +from dagster._core.definitions.freshness_policy import FreshnessPolicy +from dagster._core.definitions.module_loaders.utils import ( + KeyScopedAssetObjects, + LoadableAssetTypes, + find_objects_in_module_of_types, + key_iterator, + replace_keys_in_asset, +) +from dagster._core.definitions.source_asset import SourceAsset +from dagster._core.errors import DagsterInvalidDefinitionError + + +class LoadedAssetsList: + def __init__( + self, + assets_per_module: Mapping[str, Sequence[LoadableAssetTypes]], + ): + self.assets_per_module = assets_per_module + self._do_collision_detection() + + @classmethod + def from_modules(cls, modules: Iterable[ModuleType]) -> "LoadedAssetsList": + return cls( + { + module.__name__: list( + find_objects_in_module_of_types( + module, + (AssetsDefinition, SourceAsset, CacheableAssetsDefinition, AssetSpec), + ) + ) + for module in modules + }, + ) + + @cached_property + def flat_object_list(self) -> Sequence[LoadableAssetTypes]: + return [ + asset_object for objects in self.assets_per_module.values() for asset_object in objects + ] + + @cached_property + def objects_by_id(self) -> Dict[int, LoadableAssetTypes]: + return {id(asset_object): asset_object for asset_object in self.flat_object_list} + + @cached_property + def deduped_objects(self) -> Sequence[LoadableAssetTypes]: + return list(self.objects_by_id.values()) + + @cached_property + def assets_defs(self) -> Sequence[AssetsDefinition]: + return [asset for asset in self.deduped_objects if isinstance(asset, AssetsDefinition)] + + @cached_property + def source_assets(self) -> Sequence[SourceAsset]: + return [asset for asset in self.deduped_objects if isinstance(asset, SourceAsset)] + + @cached_property + def module_name_by_id(self) -> Dict[int, str]: + return { + id(asset_object): module_name + for module_name, objects in self.assets_per_module.items() + for asset_object in objects + } + + @cached_property + def objects_by_key(self) -> Mapping[AssetKey, Sequence[Union[SourceAsset, AssetsDefinition]]]: + objects_by_key = defaultdict(list) + for asset_object in self.flat_object_list: + if not isinstance(asset_object, KeyScopedAssetObjects): + continue + for key in key_iterator(asset_object): + objects_by_key[key].append(asset_object) + return objects_by_key + + def _do_collision_detection(self) -> None: + for key, asset_objects in self.objects_by_key.items(): + # If there is more than one asset_object in the list for a given key, and the objects do not refer to the same asset_object in memory, we have a collision. + num_distinct_objects_for_key = len( + set(id(asset_object) for asset_object in asset_objects) + ) + if len(asset_objects) > 1 and num_distinct_objects_for_key > 1: + asset_objects_str = ", ".join( + set(self.module_name_by_id[id(asset_object)] for asset_object in asset_objects) + ) + raise DagsterInvalidDefinitionError( + f"Asset key {key.to_user_string()} is defined multiple times. Definitions found in modules: {asset_objects_str}." + ) + + def to_post_load(self) -> "ResolvedAssetObjectList": + return ResolvedAssetObjectList(self.deduped_objects) + + +class ResolvedAssetObjectList: + def __init__( + self, + loaded_objects: Sequence[LoadableAssetTypes], + ): + self.loaded_objects = loaded_objects + + @cached_property + def assets_defs_and_specs(self) -> Sequence[Union[AssetsDefinition, AssetSpec]]: + return [ + asset + for asset in self.loaded_objects + if (isinstance(asset, AssetsDefinition) and asset.keys) or isinstance(asset, AssetSpec) + ] + + @cached_property + def assets_defs(self) -> Sequence[AssetsDefinition]: + return [asset for asset in self.loaded_objects if isinstance(asset, AssetsDefinition)] + + @cached_property + def checks_defs(self) -> Sequence[AssetChecksDefinition]: + return [ + cast(AssetChecksDefinition, asset) + for asset in self.loaded_objects + if isinstance(asset, AssetsDefinition) and has_only_asset_checks(asset) + ] + + @cached_property + def assets_defs_specs_and_checks_defs( + self, + ) -> Sequence[Union[AssetsDefinition, AssetSpec, AssetChecksDefinition]]: + return [*self.assets_defs_and_specs, *self.checks_defs] + + @cached_property + def source_assets(self) -> Sequence[SourceAsset]: + return [asset for asset in self.loaded_objects if isinstance(asset, SourceAsset)] + + @cached_property + def cacheable_assets(self) -> Sequence[CacheableAssetsDefinition]: + return [ + asset for asset in self.loaded_objects if isinstance(asset, CacheableAssetsDefinition) + ] + + def get_objects( + self, filter_fn: Callable[[LoadableAssetTypes], bool] + ) -> Sequence[LoadableAssetTypes]: + return [asset for asset in self.loaded_objects if filter_fn(asset)] + + def assets_with_loadable_prefix( + self, key_prefix: CoercibleToAssetKeyPrefix + ) -> "ResolvedAssetObjectList": + # There is a tricky edge case here where if a non-cacheable asset depends on a cacheable asset, + # and the assets are prefixed, the non-cacheable asset's dependency will not be prefixed since + # at prefix-time it is not known that its dependency is one of the cacheable assets. + # https://github.com/dagster-io/dagster/pull/10389#pullrequestreview-1170913271 + result_list = [] + all_asset_keys = { + key + for asset_object in self.assets_defs_specs_and_checks_defs + for key in key_iterator(asset_object, included_targeted_keys=True) + } + all_check_keys = { + check_key for asset_object in self.assets_defs for check_key in asset_object.check_keys + } + + key_replacements = {key: key.with_prefix(key_prefix) for key in all_asset_keys} + check_key_replacements = { + check_key: check_key.with_asset_key_prefix(key_prefix) for check_key in all_check_keys + } + for asset_object in self.loaded_objects: + if isinstance(asset_object, CacheableAssetsDefinition): + result_list.append(asset_object.with_prefix_for_all(key_prefix)) + elif isinstance(asset_object, AssetsDefinition): + result_list.append( + replace_keys_in_asset(asset_object, key_replacements, check_key_replacements) + ) + else: + # We don't replace the key for SourceAssets. + result_list.append(asset_object) + return ResolvedAssetObjectList(result_list) + + def assets_with_source_prefix( + self, key_prefix: CoercibleToAssetKeyPrefix + ) -> "ResolvedAssetObjectList": + result_list = [] + key_replacements = { + source_asset.key: source_asset.key.with_prefix(key_prefix) + for source_asset in self.source_assets + } + for asset_object in self.loaded_objects: + if isinstance(asset_object, KeyScopedAssetObjects): + result_list.append( + replace_keys_in_asset(asset_object, key_replacements, check_key_replacements={}) + ) + else: + result_list.append(asset_object) + return ResolvedAssetObjectList(result_list) + + def with_attributes( + self, + key_prefix: Optional[CoercibleToAssetKeyPrefix], + source_key_prefix: Optional[CoercibleToAssetKeyPrefix], + group_name: Optional[str], + freshness_policy: Optional[FreshnessPolicy], + automation_condition: Optional[AutomationCondition], + backfill_policy: Optional[BackfillPolicy], + ) -> "ResolvedAssetObjectList": + assets_list = self.assets_with_loadable_prefix(key_prefix) if key_prefix else self + assets_list = ( + assets_list.assets_with_source_prefix(source_key_prefix) + if source_key_prefix + else assets_list + ) + return_list = [] + for asset in assets_list.loaded_objects: + return_list.append( + asset.with_attributes( + group_name=group_name, + freshness_policy=freshness_policy, + automation_condition=automation_condition, + backfill_policy=backfill_policy, + ) + ) + return ResolvedAssetObjectList(return_list) diff --git a/python_modules/dagster/dagster/_core/definitions/module_loaders/utils.py b/python_modules/dagster/dagster/_core/definitions/module_loaders/utils.py new file mode 100644 index 0000000000000..33a8341eb14e8 --- /dev/null +++ b/python_modules/dagster/dagster/_core/definitions/module_loaders/utils.py @@ -0,0 +1,99 @@ +import pkgutil +from importlib import import_module +from types import ModuleType +from typing import Iterable, Iterator, Mapping, Tuple, Type, Union + +from dagster._core.definitions.asset_key import AssetCheckKey, AssetKey +from dagster._core.definitions.asset_spec import AssetSpec +from dagster._core.definitions.assets import AssetsDefinition +from dagster._core.definitions.cacheable_assets import CacheableAssetsDefinition +from dagster._core.definitions.source_asset import SourceAsset + +LoadableAssetTypes = Union[AssetsDefinition, AssetSpec, SourceAsset, CacheableAssetsDefinition] +KeyScopedAssetObjects = (AssetsDefinition, AssetSpec, SourceAsset) + + +def find_objects_in_module_of_types( + module: ModuleType, + types: Union[Type, Tuple[Type, ...]], +) -> Iterator: + """Yields instances or subclasses of the given type(s).""" + for attr in dir(module): + value = getattr(module, attr) + if isinstance(value, types): + yield value + elif isinstance(value, list) and all(isinstance(el, types) for el in value): + yield from value + + +def find_subclasses_in_module( + module: ModuleType, + types: Union[Type, Tuple[Type, ...]], +) -> Iterator: + """Yields instances or subclasses of the given type(s).""" + for attr in dir(module): + value = getattr(module, attr) + if isinstance(value, type) and issubclass(value, types): + yield value + + +def key_iterator( + asset: Union[AssetsDefinition, SourceAsset, AssetSpec], included_targeted_keys: bool = False +) -> Iterator[AssetKey]: + return ( + iter( + [ + *asset.keys, + *( + [check_key.asset_key for check_key in asset.check_keys] + if included_targeted_keys + else [] + ), + ] + ) + if isinstance(asset, AssetsDefinition) + else iter([asset.key]) + ) + + +def find_modules_in_package(package_module: ModuleType) -> Iterable[ModuleType]: + yield package_module + if package_module.__file__: + for _, modname, is_pkg in pkgutil.walk_packages( + package_module.__path__, prefix=package_module.__name__ + "." + ): + submodule = import_module(modname) + if is_pkg: + yield from find_modules_in_package(submodule) + else: + yield submodule + else: + raise ValueError( + f"Tried to find modules in package {package_module}, but its __file__ is None" + ) + + +def replace_keys_in_asset( + asset: Union[AssetsDefinition, AssetSpec, SourceAsset], + key_replacements: Mapping[AssetKey, AssetKey], + check_key_replacements: Mapping[AssetCheckKey, AssetCheckKey], +) -> Union[AssetsDefinition, AssetSpec, SourceAsset]: + if isinstance(asset, SourceAsset): + return asset.with_attributes(key=key_replacements.get(asset.key, asset.key)) + if isinstance(asset, AssetSpec): + return asset.replace_attributes( + key=key_replacements.get(asset.key, asset.key), + ) + else: + updated_object = asset.with_attributes( + output_asset_key_replacements={ + key: key_replacements.get(key, key) for key in asset.keys + }, + output_check_key_replacements={ + key: check_key_replacements.get(key, key) for key in asset.check_keys + }, + input_asset_key_replacements={ + key: key_replacements.get(key, key) for key in asset.keys_by_input_name.values() + }, + ) + return updated_object