From c2dad36a470f05ed2442b492b76a17dccdfcc439 Mon Sep 17 00:00:00 2001 From: "aziz.karabashov" Date: Thu, 8 Feb 2024 14:52:30 +0500 Subject: [PATCH 1/5] Add schema compatibility document --- docs/design/schema-compatibility.md | 63 +++++++++++++++++++++++++++++ 1 file changed, 63 insertions(+) create mode 100644 docs/design/schema-compatibility.md diff --git a/docs/design/schema-compatibility.md b/docs/design/schema-compatibility.md new file mode 100644 index 000000000..80cca3a22 --- /dev/null +++ b/docs/design/schema-compatibility.md @@ -0,0 +1,63 @@ +# Support for forward and backward compatibility in DCL schemes + +Schema changes can cover a wide range of modifications with varying impacts on application compatibility and data integrity. Below are use cases with strategies to manage schema changes and ensure compatibility. + +## Strategy for Compatible Changes + +For changes that are backward-compatible, such as adding optional fields or extending enumerations: + +**Strategy steps:** + +- One time actions: + - Add a version field to DCL schema to track the schema version. +- For each update: + - Update the schema. + - Probably need to update transactions and queries. + +This strategy is straightforward and quick to implement, but only suitable for compatible changes. + +## Strategy for Convertible Changes + +For changes that affect compatibility but can be converted, like field type changes, renaming fields, and changing enumerations: + +**Strategy steps:** + +- For each update: + - Create a new schema version and state + - Migrate older states to newer schema version. + - Implement transactions and queries capable of converting data between the latest and older schema version. + +The specific implementation details will vary with each change, each time requires an individual approach. + +## Strategy for Handling Non-Convertible Changes + +For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields: + +**Strategy steps:** + +- One time actions: + - Create a more flexible, generic schema structure to hold a wide range of data formats (Can be used [Any](https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/any.proto) as described in [ADR-19](https://docs.cosmos.network/v0.47/build/architecture/adr-019-protobuf-state-encoding#usage-of-any-to-encode-interfaces)) + - Migrate old states to the newer, generic schema. + - Implement transactions and queries that can handle data according to its version, including mechanisms for converting generic values into the corresponding schema version. +- For each update: + - Create a new schema version + - Probably need to update transactions and queries + +While offering a robust solution for handling radical changes, this method requires careful planning and development, which can potentially take a significant amount of time. + +## Strategy for Structural Changes + +For major organizational changes to the schema, such as splitting or merging schemas: + +**Strategy steps:** + +- For each update: + - Create a new schema version and state + - Migrate older states to newer schema version. + - Implement transactions and queries specifically designed for the reorganized schema. + +These types of changes are relatively rare. Implementing backward compatibility and migration can be complex. + +## Conclusion + +To lay the foundation for future compatibility improvements, it's a good idea to start by adding a version field to each schema. This step not only provides compatible updates to schemas, but also prepares the schemas for seamless integration with upcoming compatibility enhancements. From f516cf6420e3cd1a2fb9c94d2836fd0fbc3c94d5 Mon Sep 17 00:00:00 2001 From: "aziz.karabashov" Date: Fri, 9 Feb 2024 15:26:29 +0500 Subject: [PATCH 2/5] Revise and clarify the schema compatibility document --- docs/design/schema-compatibility.md | 51 +++++++++++++++++++++-------- 1 file changed, 37 insertions(+), 14 deletions(-) diff --git a/docs/design/schema-compatibility.md b/docs/design/schema-compatibility.md index 80cca3a22..c2a966c18 100644 --- a/docs/design/schema-compatibility.md +++ b/docs/design/schema-compatibility.md @@ -2,34 +2,54 @@ Schema changes can cover a wide range of modifications with varying impacts on application compatibility and data integrity. Below are use cases with strategies to manage schema changes and ensure compatibility. -## Strategy for Compatible Changes +## Non Breaking Changes + +This section provides potential strategies for maintaining backward compatibility for changes that either do not break compatibility or can be converted without loss of information. + +### Strategy for Compatible Changes For changes that are backward-compatible, such as adding optional fields or extending enumerations: **Strategy steps:** - One time actions: - - Add a version field to DCL schema to track the schema version. + - Add an optional version field to all DCL schema to track the schema version. - For each update: - - Update the schema. + - Update the schema by introducing compatible changes (such as adding a new optional field). - Probably need to update transactions and queries. + - DCL doesn't fulfill the Schema version automatically + - It will be up to the transaction submitter (Vendor) to specify a correct Schema version + - If Schema Version is not set - then the initial version (version 0 or 1) is assumed + - It will be up to the client application to process the Schema version This strategy is straightforward and quick to implement, but only suitable for compatible changes. -## Strategy for Convertible Changes +### Strategy for Convertible Changes -For changes that affect compatibility but can be converted, like field type changes, renaming fields, and changing enumerations: +For changes that affect compatibility but can be converted, like renaming fields, and changing enumerations: **Strategy steps:** - For each update: - - Create a new schema version and state + - Create a new version of a Schema and state (a new .proto file) - Migrate older states to newer schema version. - - Implement transactions and queries capable of converting data between the latest and older schema version. + - Remove the states associated with the older schema versions. + - Implement transactions and queries for the new schema version. + - Update older transactions and queries to converting data between the latest and older schema version, ensuring backward compatibility. + - There will be separated API for each version of the schema, for example:: + - models/vid/pid + - modelsV2/vid/pid + - modelsV3/vid/pid The specific implementation details will vary with each change, each time requires an individual approach. -## Strategy for Handling Non-Convertible Changes +Support for the Light Client feature will not be extended to legacy APIs due to on-the-fly data migration (as the data is not in the State and proofs can be generated). + +## Breaking changes + +This section provides potential strategies for maintaining backward compatibility for breaking changes. However, it is unlikely that these options will be implemented in production, as we do not plan to support backward compatibility for breaking changes. + +### Strategy for Handling Non-Convertible Changes For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields: @@ -38,23 +58,26 @@ For significant changes that directly impact compatibility, such as adding manda - One time actions: - Create a more flexible, generic schema structure to hold a wide range of data formats (Can be used [Any](https://github.com/protocolbuffers/protobuf/blob/main/src/google/protobuf/any.proto) as described in [ADR-19](https://docs.cosmos.network/v0.47/build/architecture/adr-019-protobuf-state-encoding#usage-of-any-to-encode-interfaces)) - Migrate old states to the newer, generic schema. - - Implement transactions and queries that can handle data according to its version, including mechanisms for converting generic values into the corresponding schema version. + - Remove the states associated with the older schema versions. + - Optioanlly can be implemented queries for requesting schemas with any return type - For each update: - - Create a new schema version - - Probably need to update transactions and queries + - Create a new Schema version (a new .proto file) + - Implement transactions and queries that can handle data according to its version, including mechanisms for converting generic values into the corresponding schema version. While offering a robust solution for handling radical changes, this method requires careful planning and development, which can potentially take a significant amount of time. -## Strategy for Structural Changes +### Strategy for Structural Changes For major organizational changes to the schema, such as splitting or merging schemas: **Strategy steps:** - For each update: - - Create a new schema version and state + - Create a new version of a Schema and state (a new .proto file) - Migrate older states to newer schema version. - - Implement transactions and queries specifically designed for the reorganized schema. + - Remove the states associated with the older schema versions. + - Implement transactions and queries for the new schema version. + - Implement transactions and queries that are backward compatible, specifically designed for the reorganized schema. These types of changes are relatively rare. Implementing backward compatibility and migration can be complex. From 7aa8adbf4e384247157d21cb2afd5c645c2317c5 Mon Sep 17 00:00:00 2001 From: "aziz.karabashov" Date: Sat, 10 Feb 2024 14:39:30 +0500 Subject: [PATCH 3/5] Update schema compatibilty document structure --- docs/design/schema-compatibility.md | 78 +++++++++++++++++------------ 1 file changed, 45 insertions(+), 33 deletions(-) diff --git a/docs/design/schema-compatibility.md b/docs/design/schema-compatibility.md index c2a966c18..694246b98 100644 --- a/docs/design/schema-compatibility.md +++ b/docs/design/schema-compatibility.md @@ -2,56 +2,53 @@ Schema changes can cover a wide range of modifications with varying impacts on application compatibility and data integrity. Below are use cases with strategies to manage schema changes and ensure compatibility. -## Non Breaking Changes - -This section provides potential strategies for maintaining backward compatibility for changes that either do not break compatibility or can be converted without loss of information. +## Multiple versions can live in parallel ### Strategy for Compatible Changes For changes that are backward-compatible, such as adding optional fields or extending enumerations: +**Description:** + +Add an optional version field to all DCL schema to track the schema version. + +This strategy is straightforward and quick to implement, but only suitable for compatible changes. + **Strategy steps:** - One time actions: - - Add an optional version field to all DCL schema to track the schema version. + - Add an optional version field to all DCL schema - For each update: - Update the schema by introducing compatible changes (such as adding a new optional field). - - Probably need to update transactions and queries. + - Update update transactions and queries if needed. - DCL doesn't fulfill the Schema version automatically - It will be up to the transaction submitter (Vendor) to specify a correct Schema version - If Schema Version is not set - then the initial version (version 0 or 1) is assumed - It will be up to the client application to process the Schema version -This strategy is straightforward and quick to implement, but only suitable for compatible changes. +### Strategy for Non-Compatible Changes + +For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields, splitting or merging schemas, changing enumerations: -### Strategy for Convertible Changes +#### Option 1: Separate Schemas for Each Version -For changes that affect compatibility but can be converted, like renaming fields, and changing enumerations: +**Description:** + +Each version has its distinct schema, state and its own queries/requests. This strategy eliminates the need for data migration and allows different schema versions to coexist seamlessly. **Strategy steps:** - For each update: - Create a new version of a Schema and state (a new .proto file) - - Migrate older states to newer schema version. - - Remove the states associated with the older schema versions. - Implement transactions and queries for the new schema version. - - Update older transactions and queries to converting data between the latest and older schema version, ensuring backward compatibility. - - There will be separated API for each version of the schema, for example:: - - models/vid/pid - - modelsV2/vid/pid - - modelsV3/vid/pid - -The specific implementation details will vary with each change, each time requires an individual approach. - -Support for the Light Client feature will not be extended to legacy APIs due to on-the-fly data migration (as the data is not in the State and proofs can be generated). -## Breaking changes +#### Option 2: Generic Schema Storage (Not Recommended for Production) -This section provides potential strategies for maintaining backward compatibility for breaking changes. However, it is unlikely that these options will be implemented in production, as we do not plan to support backward compatibility for breaking changes. +**Description:** -### Strategy for Handling Non-Convertible Changes +Implement a flexible, generic schema structure that can support a wide range of data formats. -For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields: +While offering a robust solution for handling radical changes, this method requires careful planning and development, which can potentially take a significant amount of time. **Strategy steps:** @@ -64,23 +61,38 @@ For significant changes that directly impact compatibility, such as adding manda - Create a new Schema version (a new .proto file) - Implement transactions and queries that can handle data according to its version, including mechanisms for converting generic values into the corresponding schema version. -While offering a robust solution for handling radical changes, this method requires careful planning and development, which can potentially take a significant amount of time. +## New version replaces the legacy one (V2 replaces V1) -### Strategy for Structural Changes +### Strategy for Compatible changes (Not keeping backward compatibility in API) -For major organizational changes to the schema, such as splitting or merging schemas: +For changes that are backward-compatible, such as adding optional fields or extending enumerations: + +**Description:** + +This strategy focuses on updating the schema without ensuring backward compatibility at the API level. Since the schemas are compatible, there will likely be no need for migration. **Strategy steps:** - For each update: - - Create a new version of a Schema and state (a new .proto file) - - Migrate older states to newer schema version. - - Remove the states associated with the older schema versions. - - Implement transactions and queries for the new schema version. - - Implement transactions and queries that are backward compatible, specifically designed for the reorganized schema. + - Update the schema by introducing compatible changes (such as adding a new optional field). + - Migrate old states to the newer if needed. + - Update transactions and queries if needed. + +### Strategy for Non-Compatible changes (Not keeping backward compatibility in API) -These types of changes are relatively rare. Implementing backward compatibility and migration can be complex. +For changes that affect compatibility, like adding mandatory fields + +**Description:** + +This strategy focuses on updating the schema without ensuring backward compatibility at the API level. Since the schemas are not compatible, migration is carried out manually through a special transaction. + +**Strategy steps:** + +- For each update: + - Update the schema by introducing changes. + - Update transactions and queries if needed. + - Add a new transaction to fulfill new required fields (essentially this is a manual migration via transactions) ## Conclusion -To lay the foundation for future compatibility improvements, it's a good idea to start by adding a version field to each schema. This step not only provides compatible updates to schemas, but also prepares the schemas for seamless integration with upcoming compatibility enhancements. +To lay the foundation for future compatibility improvements, it's a good idea to start by adding a version field to each schema. For subsequent changes, we will then select the most appropriate strategy based on the nature of these changes. From a202f16f5fbfc1851894d085337973c71e1f0273 Mon Sep 17 00:00:00 2001 From: "aziz.karabashov" Date: Mon, 12 Feb 2024 11:42:48 +0500 Subject: [PATCH 4/5] Update schema compatibility strategies --- docs/design/schema-compatibility.md | 29 +++++++++++++++++++++++++---- 1 file changed, 25 insertions(+), 4 deletions(-) diff --git a/docs/design/schema-compatibility.md b/docs/design/schema-compatibility.md index 694246b98..d98d480b4 100644 --- a/docs/design/schema-compatibility.md +++ b/docs/design/schema-compatibility.md @@ -6,7 +6,7 @@ Schema changes can cover a wide range of modifications with varying impacts on a ### Strategy for Compatible Changes -For changes that are backward-compatible, such as adding optional fields or extending enumerations: +For changes that are backward-compatible, such as adding optional fields or extending enumerations. **Description:** @@ -28,7 +28,7 @@ This strategy is straightforward and quick to implement, but only suitable for c ### Strategy for Non-Compatible Changes -For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields, splitting or merging schemas, changing enumerations: +For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields, splitting or merging schemas, changing enumerations. #### Option 1: Separate Schemas for Each Version @@ -65,7 +65,7 @@ While offering a robust solution for handling radical changes, this method requi ### Strategy for Compatible changes (Not keeping backward compatibility in API) -For changes that are backward-compatible, such as adding optional fields or extending enumerations: +For changes that are backward-compatible, such as adding optional fields or extending enumerations. **Description:** @@ -80,7 +80,7 @@ This strategy focuses on updating the schema without ensuring backward compatibi ### Strategy for Non-Compatible changes (Not keeping backward compatibility in API) -For changes that affect compatibility, like adding mandatory fields +For changes that affect compatibility, like adding mandatory fields. **Description:** @@ -93,6 +93,27 @@ This strategy focuses on updating the schema without ensuring backward compatibi - Update transactions and queries if needed. - Add a new transaction to fulfill new required fields (essentially this is a manual migration via transactions) +### Strategy for Non-Compatible changes (Keeping backward compatibility in API) + +For changes that affect compatibility but can be converted, adding mandatory fields and changing enumerations. + +**Description:** + +The main idea of this strategy is the dynamically converting newer schemas into older ones. However, this method is only possible if there is compatibility between the newer and legacy schemas, allowing them to be converted to each other. Due to the on-the-fly data conversion, this approach does not support the Light Client in legacy APIs because the converted data is not stored in the state, preventing the generation of proofs. + +**Strategy steps:** + +- For each update: + - Create a new version of a Schema and state (a new .proto file) + - Migrate older states to newer schema version. + - Remove the states associated with the older schema versions. + - Implement transactions and queries for the new schema version. + - Update older transactions and queries to converting data between the latest and older schema version, ensuring backward compatibility. + - There will be separated API for each version of the schema, for example:: + - models/vid/pid + - modelsV2/vid/pid + - modelsV3/vid/pid + ## Conclusion To lay the foundation for future compatibility improvements, it's a good idea to start by adding a version field to each schema. For subsequent changes, we will then select the most appropriate strategy based on the nature of these changes. From d78cfbfb9a1e88feac946616fadf96b2072395db Mon Sep 17 00:00:00 2001 From: "aziz.karabashov" Date: Sat, 17 Feb 2024 13:00:20 +0500 Subject: [PATCH 5/5] Change the structure of the schema compatibility document --- docs/design/schema-compatibility.md | 71 +++++++++++++++-------------- 1 file changed, 36 insertions(+), 35 deletions(-) diff --git a/docs/design/schema-compatibility.md b/docs/design/schema-compatibility.md index d98d480b4..d2c280f18 100644 --- a/docs/design/schema-compatibility.md +++ b/docs/design/schema-compatibility.md @@ -2,17 +2,16 @@ Schema changes can cover a wide range of modifications with varying impacts on application compatibility and data integrity. Below are use cases with strategies to manage schema changes and ensure compatibility. -## Multiple versions can live in parallel +## I. Multiple versions can live in parallel -### Strategy for Compatible Changes +### 1. Strategy for Compatible Changes For changes that are backward-compatible, such as adding optional fields or extending enumerations. -**Description:** - -Add an optional version field to all DCL schema to track the schema version. +#### Option A: Add an optional version field to all DCL schema -This strategy is straightforward and quick to implement, but only suitable for compatible changes. +**Description:** +Implement an optional version field in all DCL schemas to track the schema version. This approach is simple and quick to execute, suitable primarily for compatible updates. **Strategy steps:** @@ -20,20 +19,19 @@ This strategy is straightforward and quick to implement, but only suitable for c - Add an optional version field to all DCL schema - For each update: - Update the schema by introducing compatible changes (such as adding a new optional field). - - Update update transactions and queries if needed. + - Update transactions and queries if needed. - DCL doesn't fulfill the Schema version automatically - It will be up to the transaction submitter (Vendor) to specify a correct Schema version - If Schema Version is not set - then the initial version (version 0 or 1) is assumed - It will be up to the client application to process the Schema version -### Strategy for Non-Compatible Changes +### 2. Strategy for Non-Compatible Changes For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields, splitting or merging schemas, changing enumerations. -#### Option 1: Separate Schemas for Each Version +#### Option B: Separate Schemas for Each Version **Description:** - Each version has its distinct schema, state and its own queries/requests. This strategy eliminates the need for data migration and allows different schema versions to coexist seamlessly. **Strategy steps:** @@ -42,10 +40,9 @@ Each version has its distinct schema, state and its own queries/requests. This s - Create a new version of a Schema and state (a new .proto file) - Implement transactions and queries for the new schema version. -#### Option 2: Generic Schema Storage (Not Recommended for Production) +#### Option C: Generic Schema Storage (Not Recommended for Production) **Description:** - Implement a flexible, generic schema structure that can support a wide range of data formats. While offering a robust solution for handling radical changes, this method requires careful planning and development, which can potentially take a significant amount of time. @@ -61,14 +58,15 @@ While offering a robust solution for handling radical changes, this method requi - Create a new Schema version (a new .proto file) - Implement transactions and queries that can handle data according to its version, including mechanisms for converting generic values into the corresponding schema version. -## New version replaces the legacy one (V2 replaces V1) +## II. New version replaces the legacy one (V2 replaces V1) -### Strategy for Compatible changes (Not keeping backward compatibility in API) +### 1. Strategy for Compatible or Convertible changes -For changes that are backward-compatible, such as adding optional fields or extending enumerations. +For changes that are backward-compatible, such as adding optional or mandatory fields or extending enumerations -**Description:** +#### Option D: Not keeping backward compatibility in API +**Description:** This strategy focuses on updating the schema without ensuring backward compatibility at the API level. Since the schemas are compatible, there will likely be no need for migration. **Strategy steps:** @@ -78,27 +76,9 @@ This strategy focuses on updating the schema without ensuring backward compatibi - Migrate old states to the newer if needed. - Update transactions and queries if needed. -### Strategy for Non-Compatible changes (Not keeping backward compatibility in API) - -For changes that affect compatibility, like adding mandatory fields. +#### Option E: Keeping backward compatibility in API **Description:** - -This strategy focuses on updating the schema without ensuring backward compatibility at the API level. Since the schemas are not compatible, migration is carried out manually through a special transaction. - -**Strategy steps:** - -- For each update: - - Update the schema by introducing changes. - - Update transactions and queries if needed. - - Add a new transaction to fulfill new required fields (essentially this is a manual migration via transactions) - -### Strategy for Non-Compatible changes (Keeping backward compatibility in API) - -For changes that affect compatibility but can be converted, adding mandatory fields and changing enumerations. - -**Description:** - The main idea of this strategy is the dynamically converting newer schemas into older ones. However, this method is only possible if there is compatibility between the newer and legacy schemas, allowing them to be converted to each other. Due to the on-the-fly data conversion, this approach does not support the Light Client in legacy APIs because the converted data is not stored in the state, preventing the generation of proofs. **Strategy steps:** @@ -114,6 +94,27 @@ The main idea of this strategy is the dynamically converting newer schemas into - modelsV2/vid/pid - modelsV3/vid/pid +### 2. Strategy for Non-Compatible changes + +For significant changes that directly impact compatibility, such as adding mandatory fields or removing fields, splitting or merging schemas, changing enumerations. + +#### Optiona F: Not keeping backward compatibility in API + +**Description:** +This strategy focuses on updating the schema without ensuring backward compatibility at the API level. Since the schemas are not compatible, migration is carried out manually through a special transaction. + +**Strategy steps:** + +- For each update: + - Update the schema by introducing changes. + - Update transactions and queries if needed. + - Add a new transaction to fulfill new required fields (essentially this is a manual migration via transactions) + +#### Option G: Keeping backward compatibility in API + +**Description:** +It's not possible to replace an old version here. [Multiple versions can live in parallel: Strategy for Non-Compatible Changes](#2-strategy-for-non-compatible-changes) options should be used instead. + ## Conclusion To lay the foundation for future compatibility improvements, it's a good idea to start by adding a version field to each schema. For subsequent changes, we will then select the most appropriate strategy based on the nature of these changes.