From b5d2bc237101b71aa6c601d804fb6e415b6b86d5 Mon Sep 17 00:00:00 2001 From: Erik Nyquist Date: Sun, 4 Dec 2022 17:09:41 -0800 Subject: [PATCH] Fix bug in object migration, and add test to cover it There was a bug that could result in some serialized data being lost during a migration from an older object version. --- tests/test_serializer.py | 62 +++++++++++++++++++++++++++++++++++++++- versionedobj/object.py | 14 ++++----- 2 files changed, 68 insertions(+), 8 deletions(-) diff --git a/tests/test_serializer.py b/tests/test_serializer.py index c1ff6cd..7c5f745 100644 --- a/tests/test_serializer.py +++ b/tests/test_serializer.py @@ -1192,7 +1192,7 @@ def migrate_none_to_100(attrs): self.assertEqual("1.0.3", result.target_version) self.assertEqual("1.0.0", result.version_reached) - def test_dict_multiple_migrations(self): + def test_dict_multiple_migrations_1(self): """ Tests that from_dict can successfully migrate an object that is 4 versions old, to the latest version (migration decorator) @@ -1252,6 +1252,66 @@ def migrate_102_to_103(attrs): self.assertEqual("1.0.3", result.target_version) self.assertEqual("1.0.3", result.version_reached) + def test_dict_multiple_migrations_2(self): + """ + Tests that from_dict can successfully migrate an object that is 3 versions old, + when there are 4 versions total, to the latest version (migration decorator) + """ + # Newest config, with version number added + class TestConfig(VersionedObject): + version = "1.0.3" + var1 = 1 + var2 = 2 + var3 = 3 + var4 = 4 + var5 = 5 + var6 = 6 + + # oldest config, different format, and without version number + fake_config = {'version': '1.0.1', 'var1': 11, 'var2': 12, 'var3': 113, 'var4': 114} + + # Add migration from unversioned to 1.0.0 + @migration(TestConfig, None, "1.0.0") + def migrate_none_to_100(attrs): + attrs['var3'] = 13 + return attrs + + # Add migration from 1.0.0 to 1.0.1 + @migration(TestConfig, "1.0.0", "1.0.1") + def migrate_100_to_101(attrs): + attrs['var4'] = 14 + return attrs + + # Add migration from 1.0.1 to 1.0.2 + @migration(TestConfig, "1.0.1", "1.0.2") + def migrate_101_to_102(attrs): + attrs['var5'] = 15 + return attrs + + # Add migration from 1.0.2 to 1.0.3 + @migration(TestConfig, "1.0.2", "1.0.3") + def migrate_102_to_103(attrs): + attrs['var6'] = 16 + return attrs + + # Load oldest config from dict + ser = Serializer() + cfg = TestConfig() + result = ser.from_dict(fake_config, cfg) + + # Verify all migrations were performed + self.assertEqual(11, cfg.var1) + self.assertEqual(12, cfg.var2) + self.assertEqual(113, cfg.var3) + self.assertEqual(114, cfg.var4) + self.assertEqual(15, cfg.var5) + self.assertEqual(16, cfg.var6) + + self.assertEqual(True, result.success) + self.assertEqual("1.0.1", result.old_version) + self.assertEqual("1.0.3", result.target_version) + self.assertEqual("1.0.3", result.version_reached) + def test_json_multiple_migrations(self): """ Tests that from_json can successfully migrate an object that is 4 versions old, diff --git a/versionedobj/object.py b/versionedobj/object.py index 805fb13..4d89fb9 100644 --- a/versionedobj/object.py +++ b/versionedobj/object.py @@ -195,7 +195,7 @@ def _vobj__populate_instance(self): def _vobj__migrate(cls, version, attrs): old_version = attrs.get('version', None) version_before_migration = old_version - version_after_migration = old_version + current_version = old_version result = None @@ -204,17 +204,17 @@ def _vobj__migrate(cls, version, attrs): # Attempt migrations for fromversion, toversion, migrate in cls._vobj__migrations: - if fromversion == version_after_migration: + if fromversion == current_version: attrs = migrate(attrs) - version_after_migration = toversion - if toversion == version: - break + current_version = toversion + if toversion == version: + break - if version_after_migration != version: + if current_version != version: result.success = False - result.version_reached = version_after_migration + result.version_reached = current_version return result, attrs