Description
Hi,
wouldn't it be awesome if we could update custom types or product types field definitions with a migration strategy to prevent data loss? As far I know, It doesn't exist any assistant in migrating, for example, a ProductVariant
attribute between different field types
.
Scope:
It will affect custom field definitions (Types
, ProductTypes
). That means migration support for product attributes
, custom attributes
of all supported resources. For the base fields, it's not necessary since the type will never change.
Use cases:
- A user might want to change the field
String
toLocalizedString
- A user might want to change the field
String
toNumber
- A user might want to change the field
String
toSetOfStrings
With migrationActions
I think about a companion to control how the old data is preserved.
That feature request comes from the daily practice experience with CT. Our product is very new and the model is constantly evolving.
I think this feature would bring a lot of value to the platform because it allows writing migrations scripts in a convenient and safe way.
That migration tool can be implemented in a very generic way so we don't need to respect every resource type individually.
It would require that "syncActions" can handle changes in custom attributes. I have seen that the current implementation respects only base fields or the implementation is just not complete.
Out of scope:
Updates in base fields are covered by sync-actions
whereby the implementation doesn't cover all cases e.g there is no support for custom fields
in types or product-types. That half
complete implementation is really hard to track.
It's also really hard to find out what sync
really means in the context of the "resourceType". As a customer, I would expect that sync checks EVERYTHING or at least the documentation should clarify the exact behavior.
Related issues:
Some code to illustrate the migration approach:
1. Determine attribute changes
import { typeConversionMap } from '@commercetools/migration-actions';
const updateActions = productTypeSync.buildActions(next, previous, { preservePreviousFields: true });
// for example the attribute type has changed from "String" to "Number"
// we will create a new field to hold the new value + saving type and name differences in the attribute name
updateActions = [
{
action: 'addAttributeDefinition',
attribute: {
name: 'previous-attribute-name__string-number__next-attribute-name',
type: { name: 'Number' }
}
}
];
2. Apply changes and create temporary fields to hold the new value + conversion information
client.execute(updates);
3. Fetch records which should be migrated
// for example
const productA = {
attributes: [
{ name: 'previous-attribute-name__string-number__next-attribute-name', value: null },
{ name: 'previous-attribute-name', value: '2' }
]
};
4. We will create actions to migrate all fields
With the convention "old__string-number__new" or even "old__oldType-newType__old" we know how to convert that field.
const typeConversionHelper = {
[[typeConversionMap.String][typeConversionMap.Number]]: old => parseInt(old),
[[typeConversionMap.String][typeConversionMap.ListString]]: old => [old.toString()],
[[typeConversionMap.String][typeConversionMap.LocalizedString]]: old => { en: old }
};
const dataActions = migrationSync.buildActions([productA], { conversionHelper });
dataActions = {
url: 'https://api/products/....',
body: {
actions: [
{
action: 'setAttribute',
name: 'previous-attribute-name__string-number__next-attribute-name',
value: 2
}
]
}
};
5. Determine changes based on the naming convention and remove old fields and rename temporary fields
const dataActions = migrationSync.migrationCleanActions([productA]);
dataActions = [
{
action: 'removeAttributeDefinition',
name: 'previous-attribute-name'
},
{
action: 'changeAttributeName',
attributeName: 'previous-attribute-name__string-number__next-attribute-name',
newAttributeName: 'next-attribute-name'
}
];