From e61206f342fb43aceab2de4ce888bc4d2b376068 Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 08:28:07 -0800 Subject: [PATCH 1/6] [GCU] Implementing DryRun by printing pathc-sorter steps --- config/main.py | 10 ++++++++++ generic_config_updater/change_applier.py | 14 ++++++++++++-- generic_config_updater/generic_updater.py | 2 +- generic_config_updater/gu_common.py | 14 ++++++++++++-- 4 files changed, 35 insertions(+), 5 deletions(-) diff --git a/config/main.py b/config/main.py index f914e64653..ef22bd53ce 100644 --- a/config/main.py +++ b/config/main.py @@ -1164,6 +1164,10 @@ def load(filename, yes): log.log_info("'load' executing...") clicommon.run_command(command, display_cmd=True) +def print_dry_run_message(dry_run): + if dry_run: + click.secho("** DRY RUN EXECUTION **.", fg="yellow", underline=True) + @config.command('apply-patch') @click.argument('patch-file-path', type=str, required=True) @click.option('-f', '--format', type=click.Choice([e.name for e in ConfigFormat]), @@ -1182,6 +1186,8 @@ def apply_patch(ctx, patch_file_path, format, dry_run, ignore_non_yang_tables, i : Path to the patch file on the file-system.""" try: + print_dry_run_message(dry_run) + with open(patch_file_path, 'r') as fh: text = fh.read() patch_as_json = json.loads(text) @@ -1214,6 +1220,8 @@ def replace(ctx, target_file_path, format, dry_run, ignore_non_yang_tables, igno : Path to the target file on the file-system.""" try: + print_dry_run_message(dry_run) + with open(target_file_path, 'r') as fh: target_config_as_text = fh.read() target_config = json.loads(target_config_as_text) @@ -1241,6 +1249,8 @@ def rollback(ctx, checkpoint_name, dry_run, ignore_non_yang_tables, ignore_path, : The checkpoint name, use `config list-checkpoints` command to see available checkpoints.""" try: + print_dry_run_message(dry_run) + GenericUpdater().rollback(checkpoint_name, verbose, dry_run, ignore_non_yang_tables, ignore_path) click.secho("Config rolled back successfully.", fg="cyan", underline=True) diff --git a/generic_config_updater/change_applier.py b/generic_config_updater/change_applier.py index 23b9383814..35dbf25f60 100644 --- a/generic_config_updater/change_applier.py +++ b/generic_config_updater/change_applier.py @@ -6,7 +6,7 @@ import tempfile from collections import defaultdict from swsscommon.swsscommon import ConfigDBConnector -from .gu_common import genericUpdaterLogging +from .gu_common import genericUpdaterLogging, DryRunConfigWrapper SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__)) UPDATER_CONF_FILE = f"{SCRIPT_DIR}/generic_config_updater.conf.json" @@ -59,7 +59,12 @@ class ChangeApplier: updater_conf = None - def __init__(self): + def __init__(self, config_wrapper): + self.config_wrapper = config_wrapper + if type(self.config_wrapper) is DryRunConfigWrapper: + # No need to init the rest of the parameters + return + self.config_db = get_config_db() if (not ChangeApplier.updater_conf) and os.path.exists(UPDATER_CONF_FILE): with open(UPDATER_CONF_FILE, "r") as s: @@ -121,6 +126,11 @@ def _report_mismatch(self, run_data, upd_data): def apply(self, change): + # TODO: DryRun to also include service validation. + if type(self.config_wrapper) is DryRunConfigWrapper: + self.config_wrapper.apply_change_to_config_db(change) + return + run_data = self._get_running_config() upd_data = prune_empty_table(change.apply(copy.deepcopy(run_data))) upd_keys = defaultdict(dict) diff --git a/generic_config_updater/generic_updater.py b/generic_config_updater/generic_updater.py index a4ea6f5ee6..9f3912f40c 100644 --- a/generic_config_updater/generic_updater.py +++ b/generic_config_updater/generic_updater.py @@ -34,7 +34,7 @@ def __init__(self, self.config_wrapper = config_wrapper if config_wrapper is not None else ConfigWrapper() self.patch_wrapper = patch_wrapper if patch_wrapper is not None else PatchWrapper() self.patchsorter = patchsorter if patchsorter is not None else StrictPatchSorter(self.config_wrapper, self.patch_wrapper) - self.changeapplier = changeapplier if changeapplier is not None else ChangeApplier() + self.changeapplier = changeapplier if changeapplier is not None else ChangeApplier(self.config_wrapper) def apply(self, patch): self.logger.log_notice("Patch application starting.") diff --git a/generic_config_updater/gu_common.py b/generic_config_updater/gu_common.py index ea4954ba24..37f942e21d 100644 --- a/generic_config_updater/gu_common.py +++ b/generic_config_updater/gu_common.py @@ -152,9 +152,19 @@ def remove_empty_tables(self, config): return config_with_non_empty_tables class DryRunConfigWrapper(ConfigWrapper): - # TODO: implement DryRunConfigWrapper # This class will simulate all read/write operations to ConfigDB on a virtual storage unit. - pass + def __init__(self, yang_dir = YANG_DIR): + self.yang_dir = YANG_DIR + super().__init__(yang_dir) + self.logger = genericUpdaterLogging.get_logger(title="** DryRun", print_all_to_console=True) + self.imitated_config_db = super().get_config_db_as_json() + + def apply_change_to_config_db(self, change): + self.logger.log_notice(f"Would apply {change}") + self.imitated_config_db = change.apply(self.imitated_config_db) + + def get_config_db_as_json(self): + return self.imitated_config_db class PatchWrapper: def __init__(self, config_wrapper=None): From 4e81809b70fa60a8e635601fab87f6ebef7ba86b Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 16:02:01 -0800 Subject: [PATCH 2/6] addressing comments --- generic_config_updater/change_applier.py | 22 +++++----- generic_config_updater/generic_updater.py | 28 +++++++++--- generic_config_updater/gu_common.py | 9 ++-- .../change_applier_test.py | 16 ++++++- .../generic_updater_test.py | 11 +++++ .../generic_config_updater/gu_common_test.py | 44 +++++++++++++++++++ 6 files changed, 108 insertions(+), 22 deletions(-) diff --git a/generic_config_updater/change_applier.py b/generic_config_updater/change_applier.py index 35dbf25f60..6997dcabd8 100644 --- a/generic_config_updater/change_applier.py +++ b/generic_config_updater/change_applier.py @@ -55,16 +55,21 @@ def prune_empty_table(data): return data -class ChangeApplier: - - updater_conf = None +class DryRunChangeApplier: def __init__(self, config_wrapper): self.config_wrapper = config_wrapper - if type(self.config_wrapper) is DryRunConfigWrapper: - # No need to init the rest of the parameters - return + + def apply(self, change): + self.config_wrapper.apply_change_to_config_db(change) + + +class ChangeApplier: + + updater_conf = None + + def __init__(self): self.config_db = get_config_db() if (not ChangeApplier.updater_conf) and os.path.exists(UPDATER_CONF_FILE): with open(UPDATER_CONF_FILE, "r") as s: @@ -126,11 +131,6 @@ def _report_mismatch(self, run_data, upd_data): def apply(self, change): - # TODO: DryRun to also include service validation. - if type(self.config_wrapper) is DryRunConfigWrapper: - self.config_wrapper.apply_change_to_config_db(change) - return - run_data = self._get_running_config() upd_data = prune_empty_table(change.apply(copy.deepcopy(run_data))) upd_keys = defaultdict(dict) diff --git a/generic_config_updater/generic_updater.py b/generic_config_updater/generic_updater.py index 9f3912f40c..ee7af65620 100644 --- a/generic_config_updater/generic_updater.py +++ b/generic_config_updater/generic_updater.py @@ -5,7 +5,7 @@ DryRunConfigWrapper, PatchWrapper, genericUpdaterLogging from .patch_sorter import StrictPatchSorter, NonStrictPatchSorter, ConfigSplitter, \ TablesWithoutYangConfigSplitter, IgnorePathsFromYangConfigSplitter -from .change_applier import ChangeApplier +from .change_applier import ChangeApplier, DryRunChangeApplier CHECKPOINTS_DIR = "/etc/sonic/checkpoints" CHECKPOINT_EXT = ".cp.json" @@ -34,7 +34,7 @@ def __init__(self, self.config_wrapper = config_wrapper if config_wrapper is not None else ConfigWrapper() self.patch_wrapper = patch_wrapper if patch_wrapper is not None else PatchWrapper() self.patchsorter = patchsorter if patchsorter is not None else StrictPatchSorter(self.config_wrapper, self.patch_wrapper) - self.changeapplier = changeapplier if changeapplier is not None else ChangeApplier(self.config_wrapper) + self.changeapplier = changeapplier if changeapplier is not None else ChangeApplier() def apply(self, patch): self.logger.log_notice("Patch application starting.") @@ -299,9 +299,13 @@ class GenericUpdateFactory: def create_patch_applier(self, config_format, verbose, dry_run, ignore_non_yang_tables, ignore_paths): self.init_verbose_logging(verbose) config_wrapper = self.get_config_wrapper(dry_run) + change_applier = self.get_change_applier(dry_run, config_wrapper) patch_wrapper = PatchWrapper(config_wrapper) patch_sorter = self.get_patch_sorter(ignore_non_yang_tables, ignore_paths, config_wrapper, patch_wrapper) - patch_applier = PatchApplier(config_wrapper=config_wrapper, patchsorter=patch_sorter, patch_wrapper=patch_wrapper) + patch_applier = PatchApplier(config_wrapper=config_wrapper, + patchsorter=patch_sorter, + patch_wrapper=patch_wrapper, + changeapplier=change_applier) if config_format == ConfigFormat.CONFIGDB: pass @@ -320,9 +324,13 @@ def create_config_replacer(self, config_format, verbose, dry_run, ignore_non_yan self.init_verbose_logging(verbose) config_wrapper = self.get_config_wrapper(dry_run) + change_applier = self.get_change_applier(dry_run, config_wrapper) patch_wrapper = PatchWrapper(config_wrapper) patch_sorter = self.get_patch_sorter(ignore_non_yang_tables, ignore_paths, config_wrapper, patch_wrapper) - patch_applier = PatchApplier(config_wrapper=config_wrapper, patchsorter=patch_sorter, patch_wrapper=patch_wrapper) + patch_applier = PatchApplier(config_wrapper=config_wrapper, + patchsorter=patch_sorter, + patch_wrapper=patch_wrapper, + changeapplier=change_applier) config_replacer = ConfigReplacer(patch_applier=patch_applier, config_wrapper=config_wrapper) if config_format == ConfigFormat.CONFIGDB: @@ -342,9 +350,13 @@ def create_config_rollbacker(self, verbose, dry_run=False, ignore_non_yang_table self.init_verbose_logging(verbose) config_wrapper = self.get_config_wrapper(dry_run) + change_applier = self.get_change_applier(dry_run, config_wrapper) patch_wrapper = PatchWrapper(config_wrapper) patch_sorter = self.get_patch_sorter(ignore_non_yang_tables, ignore_paths, config_wrapper, patch_wrapper) - patch_applier = PatchApplier(config_wrapper=config_wrapper, patchsorter=patch_sorter, patch_wrapper=patch_wrapper) + patch_applier = PatchApplier(config_wrapper=config_wrapper, + patchsorter=patch_sorter, + patch_wrapper=patch_wrapper, + changeapplier=change_applier) config_replacer = ConfigReplacer(config_wrapper=config_wrapper, patch_applier=patch_applier) config_rollbacker = FileSystemConfigRollbacker(config_wrapper = config_wrapper, config_replacer = config_replacer) @@ -363,6 +375,12 @@ def get_config_wrapper(self, dry_run): else: return ConfigWrapper() + def get_change_applier(self, dry_run, config_wrapper): + if dry_run: + return DryRunChangeApplier(config_wrapper) + else: + return ChangeApplier() + def get_patch_sorter(self, ignore_non_yang_tables, ignore_paths, config_wrapper, patch_wrapper): if not ignore_non_yang_tables and not ignore_paths: return StrictPatchSorter(config_wrapper, patch_wrapper) diff --git a/generic_config_updater/gu_common.py b/generic_config_updater/gu_common.py index 37f942e21d..dbf702dcb0 100644 --- a/generic_config_updater/gu_common.py +++ b/generic_config_updater/gu_common.py @@ -153,11 +153,12 @@ def remove_empty_tables(self, config): class DryRunConfigWrapper(ConfigWrapper): # This class will simulate all read/write operations to ConfigDB on a virtual storage unit. - def __init__(self, yang_dir = YANG_DIR): - self.yang_dir = YANG_DIR - super().__init__(yang_dir) + def __init__(self, initial_imitated_config_db = None): + super().__init__() self.logger = genericUpdaterLogging.get_logger(title="** DryRun", print_all_to_console=True) - self.imitated_config_db = super().get_config_db_as_json() + self.imitated_config_db = copy.deepcopy(initial_imitated_config_db) \ + if initial_imitated_config_db \ + else super().get_config_db_as_json() def apply_change_to_config_db(self, change): self.logger.log_notice(f"Would apply {change}") diff --git a/tests/generic_config_updater/change_applier_test.py b/tests/generic_config_updater/change_applier_test.py index b734485ffd..55df84c855 100644 --- a/tests/generic_config_updater/change_applier_test.py +++ b/tests/generic_config_updater/change_applier_test.py @@ -4,7 +4,7 @@ import os import unittest from collections import defaultdict -from unittest.mock import patch +from unittest.mock import patch, Mock, call import generic_config_updater.change_applier import generic_config_updater.services_validator @@ -269,4 +269,16 @@ def test_change_apply(self, mock_set, mock_db, mock_os_sys): debug_print("all good for applier") - +class TestDryRunChangeApplier(unittest.TestCase): + def test_apply__calls_apply_change_to_config_db(self): + # Arrange + change = Mock() + config_wrapper = Mock() + applier = generic_config_updater.change_applier.DryRunChangeApplier(config_wrapper) + + # Act + applier.apply(change) + + # Assert + applier.config_wrapper.apply_change_to_config_db.assert_has_calls([call(change)]) + diff --git a/tests/generic_config_updater/generic_updater_test.py b/tests/generic_config_updater/generic_updater_test.py index 1a8151f398..2f75b9ca4c 100644 --- a/tests/generic_config_updater/generic_updater_test.py +++ b/tests/generic_config_updater/generic_updater_test.py @@ -7,6 +7,7 @@ import generic_config_updater.generic_updater as gu import generic_config_updater.patch_sorter as ps +import generic_config_updater.change_applier as ca # import sys # sys.path.insert(0,'../../generic_config_updater') @@ -420,8 +421,11 @@ def validate_create_patch_applier(self, params, expected_decorators): self.assertIsInstance(patch_applier, gu.PatchApplier) if params["dry_run"]: self.assertIsInstance(patch_applier.config_wrapper, gu.DryRunConfigWrapper) + self.assertIsInstance(patch_applier.changeapplier, ca.DryRunChangeApplier) + self.assertIsInstance(patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) else: self.assertIsInstance(patch_applier.config_wrapper, gu.ConfigWrapper) + self.assertIsInstance(patch_applier.changeapplier, ca.ChangeApplier) if params["ignore_non_yang_tables"] or params["ignore_paths"]: self.assertIsInstance(patch_applier.patchsorter, ps.NonStrictPatchSorter) @@ -451,9 +455,12 @@ def validate_create_config_replacer(self, params, expected_decorators): if params["dry_run"]: self.assertIsInstance(config_replacer.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance(config_replacer.patch_applier.config_wrapper, gu.DryRunConfigWrapper) + self.assertIsInstance(config_replacer.patch_applier.changeapplier, ca.DryRunChangeApplier) + self.assertIsInstance(config_replacer.patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) else: self.assertIsInstance(config_replacer.config_wrapper, gu.ConfigWrapper) self.assertIsInstance(config_replacer.patch_applier.config_wrapper, gu.ConfigWrapper) + self.assertIsInstance(config_replacer.patch_applier.changeapplier, ca.ChangeApplier) if params["ignore_non_yang_tables"] or params["ignore_paths"]: self.assertIsInstance(config_replacer.patch_applier.patchsorter, ps.NonStrictPatchSorter) @@ -482,11 +489,15 @@ def validate_create_config_rollbacker(self, params, expected_decorators): self.assertIsInstance(config_rollbacker.config_replacer.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance( config_rollbacker.config_replacer.patch_applier.config_wrapper, gu.DryRunConfigWrapper) + self.assertIsInstance(config_rollbacker.config_replacer.patch_applier.changeapplier, ca.DryRunChangeApplier) + self.assertIsInstance( + config_rollbacker.config_replacer.patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) else: self.assertIsInstance(config_rollbacker.config_wrapper, gu.ConfigWrapper) self.assertIsInstance(config_rollbacker.config_replacer.config_wrapper, gu.ConfigWrapper) self.assertIsInstance( config_rollbacker.config_replacer.patch_applier.config_wrapper, gu.ConfigWrapper) + self.assertIsInstance(config_rollbacker.config_replacer.patch_applier.changeapplier, ca.ChangeApplier) if params["ignore_non_yang_tables"] or params["ignore_paths"]: self.assertIsInstance(config_rollbacker.config_replacer.patch_applier.patchsorter, ps.NonStrictPatchSorter) diff --git a/tests/generic_config_updater/gu_common_test.py b/tests/generic_config_updater/gu_common_test.py index 56cebe786b..7d0c9124bc 100644 --- a/tests/generic_config_updater/gu_common_test.py +++ b/tests/generic_config_updater/gu_common_test.py @@ -7,6 +7,50 @@ from .gutest_helpers import create_side_effect_dict, Files import generic_config_updater.gu_common as gu_common +class TestDryRunConfigWrapper(unittest.TestCase): + def test_get_config_db_as_json__returns_imitated_config_db(self): + # Arrange + config_wrapper = gu_common.DryRunConfigWrapper(Files.CONFIG_DB_AS_JSON) + expected = Files.CONFIG_DB_AS_JSON + + # Act + actual = config_wrapper.get_config_db_as_json() + + # Assert + self.assertDictEqual(expected, actual) + + def test_get_sonic_yang_as_json__returns_imitated_config_db_as_yang(self): + # Arrange + config_wrapper = gu_common.DryRunConfigWrapper(Files.CONFIG_DB_AS_JSON) + expected = Files.SONIC_YANG_AS_JSON + + # Act + actual = config_wrapper.get_sonic_yang_as_json() + + # Assert + self.assertDictEqual(expected, actual) + + def test_apply_change_to_config_db__multiple_calls__changes_imitated_config_db(self): + # Arrange + imitated_config_db = Files.CONFIG_DB_AS_JSON + config_wrapper = gu_common.DryRunConfigWrapper(imitated_config_db) + + changes = [gu_common.JsonChange(jsonpatch.JsonPatch([{'op':'remove', 'path':'/VLAN'}])), + gu_common.JsonChange(jsonpatch.JsonPatch([{'op':'remove', 'path':'/ACL_TABLE'}])), + gu_common.JsonChange(jsonpatch.JsonPatch([{'op':'remove', 'path':'/PORT'}])) + ] + + expected = imitated_config_db + for change in changes: + # Act + config_wrapper.apply_change_to_config_db(change) + + actual = config_wrapper.get_config_db_as_json() + expected = change.apply(expected) + + # Assert + self.assertDictEqual(expected, actual) + class TestConfigWrapper(unittest.TestCase): def setUp(self): self.config_wrapper_mock = gu_common.ConfigWrapper() From 21b528af7fb96333f0a1d7c81560958fb1ad9ae9 Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 17:30:50 -0800 Subject: [PATCH 3/6] moving call to get running config to method instead of ctor --- generic_config_updater/gu_common.py | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/generic_config_updater/gu_common.py b/generic_config_updater/gu_common.py index dbf702dcb0..8f8f509e17 100644 --- a/generic_config_updater/gu_common.py +++ b/generic_config_updater/gu_common.py @@ -156,17 +156,23 @@ class DryRunConfigWrapper(ConfigWrapper): def __init__(self, initial_imitated_config_db = None): super().__init__() self.logger = genericUpdaterLogging.get_logger(title="** DryRun", print_all_to_console=True) - self.imitated_config_db = copy.deepcopy(initial_imitated_config_db) \ - if initial_imitated_config_db \ - else super().get_config_db_as_json() + self.imitated_config_db = copy.deepcopy(initial_imitated_config_db) def apply_change_to_config_db(self, change): + self._init_imitated_config_db_if_none() self.logger.log_notice(f"Would apply {change}") self.imitated_config_db = change.apply(self.imitated_config_db) def get_config_db_as_json(self): + self._init_imitated_config_db_if_none() return self.imitated_config_db + def _init_imitated_config_db_if_none(self): + # if there is no initial imitated config_db and it is the first time calling this method + if self.imitated_config_db is None: + self.imitated_config_db = super().get_config_db_as_json() + + class PatchWrapper: def __init__(self, config_wrapper=None): self.config_wrapper = config_wrapper if config_wrapper is not None else ConfigWrapper() From dd9db9efca6b00d95cd24e74de90e4a8fe2679ce Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 17:45:16 -0800 Subject: [PATCH 4/6] remove unused import --- generic_config_updater/change_applier.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/generic_config_updater/change_applier.py b/generic_config_updater/change_applier.py index 6997dcabd8..3786e1a372 100644 --- a/generic_config_updater/change_applier.py +++ b/generic_config_updater/change_applier.py @@ -6,7 +6,7 @@ import tempfile from collections import defaultdict from swsscommon.swsscommon import ConfigDBConnector -from .gu_common import genericUpdaterLogging, DryRunConfigWrapper +from .gu_common import genericUpdaterLogging SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__)) UPDATER_CONF_FILE = f"{SCRIPT_DIR}/generic_config_updater.conf.json" From 50729fee107f310e9c80094f85e3ca2da83ff640 Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 17:47:20 -0800 Subject: [PATCH 5/6] remove dot --- config/main.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/main.py b/config/main.py index ef22bd53ce..370958856b 100644 --- a/config/main.py +++ b/config/main.py @@ -1166,7 +1166,7 @@ def load(filename, yes): def print_dry_run_message(dry_run): if dry_run: - click.secho("** DRY RUN EXECUTION **.", fg="yellow", underline=True) + click.secho("** DRY RUN EXECUTION **", fg="yellow", underline=True) @config.command('apply-patch') @click.argument('patch-file-path', type=str, required=True) From e5b45ffc4ec6cb89d4e8ff2707038940164b276e Mon Sep 17 00:00:00 2001 From: ghooo Date: Thu, 16 Dec 2021 22:09:55 -0800 Subject: [PATCH 6/6] fixing unit-test --- tests/generic_config_updater/generic_updater_test.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tests/generic_config_updater/generic_updater_test.py b/tests/generic_config_updater/generic_updater_test.py index 2f75b9ca4c..aab2eae275 100644 --- a/tests/generic_config_updater/generic_updater_test.py +++ b/tests/generic_config_updater/generic_updater_test.py @@ -422,7 +422,7 @@ def validate_create_patch_applier(self, params, expected_decorators): if params["dry_run"]: self.assertIsInstance(patch_applier.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance(patch_applier.changeapplier, ca.DryRunChangeApplier) - self.assertIsInstance(patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) + self.assertIsInstance(patch_applier.changeapplier.config_wrapper, gu.DryRunConfigWrapper) else: self.assertIsInstance(patch_applier.config_wrapper, gu.ConfigWrapper) self.assertIsInstance(patch_applier.changeapplier, ca.ChangeApplier) @@ -456,7 +456,7 @@ def validate_create_config_replacer(self, params, expected_decorators): self.assertIsInstance(config_replacer.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance(config_replacer.patch_applier.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance(config_replacer.patch_applier.changeapplier, ca.DryRunChangeApplier) - self.assertIsInstance(config_replacer.patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) + self.assertIsInstance(config_replacer.patch_applier.changeapplier.config_wrapper, gu.DryRunConfigWrapper) else: self.assertIsInstance(config_replacer.config_wrapper, gu.ConfigWrapper) self.assertIsInstance(config_replacer.patch_applier.config_wrapper, gu.ConfigWrapper) @@ -491,7 +491,7 @@ def validate_create_config_rollbacker(self, params, expected_decorators): config_rollbacker.config_replacer.patch_applier.config_wrapper, gu.DryRunConfigWrapper) self.assertIsInstance(config_rollbacker.config_replacer.patch_applier.changeapplier, ca.DryRunChangeApplier) self.assertIsInstance( - config_rollbacker.config_replacer.patch_applier.changeapplier.config_wrapper, ca.DryRunConfigWrapper) + config_rollbacker.config_replacer.patch_applier.changeapplier.config_wrapper, gu.DryRunConfigWrapper) else: self.assertIsInstance(config_rollbacker.config_wrapper, gu.ConfigWrapper) self.assertIsInstance(config_rollbacker.config_replacer.config_wrapper, gu.ConfigWrapper)