nodejs icon indicating copy to clipboard operation
nodejs copied to clipboard

Implement "migrationActions" to allow migration of data / data conversions between syncActions updates

Open StarpTech opened this issue 4 years ago • 3 comments

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 to LocalizedString
  • A user might want to change the field String to Number
  • A user might want to change the field String to SetOfStrings

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:

https://github.com/commercetools/nodejs/issues/355

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'
  }
];

StarpTech avatar Oct 19 '19 22:10 StarpTech

@StarpTech I really like your idea a lot. And until now I never thought about this use case😄 Can you perhaps explain what you mean with base fields?

katmatt avatar Oct 25 '19 11:10 katmatt

Hi @katmatt, with base fields I mean all already existing attributes from the commercetools resources.

StarpTech avatar Oct 25 '19 16:10 StarpTech

@StarpTech I see. The only comment I can make at the moment is that we very likely will add support for custom fields next year to the sync-actions module. And as part of that we will add some docs that describe which properties of each resource type are supported. And that would allow you to implement this feature on your own 😄

katmatt avatar Oct 28 '19 09:10 katmatt