Migrating non-reference fields to reference fields
Summarize
Summary of Migrating Non-Reference Fields to Reference Fields
This guide outlines the process for migrating non-reference fields to reference fields in ServiceNow, emphasizing the importance of preservingsysidsinstead of storing display names as strings. The migration is facilitated through background scripts and jobs designed to manage the data transformation effectively.
Show less
Key Features
- Migration Script: A background script is provided to create migration records for relevant tables, ensuring only valid tables are processed.
- Reference Copy Job: The Archive Reference Copy job updates display values to sysids in batches of 10,000 records, based on the archived timestamp.
- Field Type Change: Once migration is complete, the ArchiveRefJob converts field types from string to reference.
- Error Handling: The guide includes steps to manage node failures during migration, ensuring continuity by updating job statuses and re-triggering processes as needed.
Key Outcomes
By following this migration process, ServiceNow customers can ensure that their data integrity is maintained, with reference fields correctly populated. This transition enhances data relationships and improves overall system functionality. Customers can expect a streamlined process for updating field types and resolving any potential migration issues efficiently.
Manually preserve sys_ids in reference fields instead of storing the display name as a string.
Determining tables to migrate
var tables = GlideDBObjectManager.get().getAllExtensions(current.table);
for (i = 0; i < tables.size(); i++) {
var gr = new GlideRecord('sys_archive_ref_migration');
gr.addQuery('table', 'ar_' + tables.get(i));
gr.query();
if (!gr.next()) {
if (GlideTableDescriptor.isValid('ar_' + tables.get(i))) {
var gr2 = new GlideRecord("sys_archive_ref_migration");
gr2.initialize();
gr2.setValue('rule', current.sys_id);
gr2.setValue('table', 'ar_' + tables.get(i));
gr2.setValue('reference_migration_progress', 'waiting');
gr2.insert();
}
}
}
//Also get insert related records tables as well
var map = new GlideRecord('sys_archive');
map.addQuery('table', current.table);
map.query();
if (map.next()) {
var id = map.getValue('sys_id');
if (!(id === undefined)) {
var related = new GlideRecord('sys_archive_related');
related.addQuery('archive_map', id);
related.addQuery('action', 'archive');
related.query();
while (related.next()) {
if (!GlideTableDescriptor.isValid('ar_' + related.getValue('table'))) {
gs.log('Related Record table: ' + related.getValue('table') + ' not created yet');
continue;
}
var gr3 = new GlideRecord("sys_archive_ref_migration");
gr3.initialize();
gr3.setValue('rule', current.sys_id);
gr3.setValue('table', 'ar_' + related.getValue('table'));
gr3.setValue('reference_migration_progress', 'waiting');
gr3.insert();
}
}
}Copying archive references
For each table that goes through the migration process, the Archive Reference Copy (RefCopyJob) job identifies the sys_id of the reference fields and updates the display value to be the correct sys_id. The job configures 10k records at a time unless there are more than 10k records with the same timestamp. The migration progress relies on the archived timestamp.
Changing field types from string to reference
After the tables associated with an archive rule completely migrate, the ArchiveRefJob job executes. This job changes the sys_dictionary types of the archive table from string to reference.
Fixing node failures for RefCopyJob and ArchiveRefJob
If node failure occurs while these jobs are running, it leaves the status of the data migration in an improper state. If the RefCopyJob fails, it can leave a table in a migrating status. You can verify this condition by checking if rows in the sys_archive_ref_migration are stuck in migrating status for an extraordinary amount of time. Update the specific row status from migrating to waiting and the RefCopyJob continues the data migration on the table when the job runs again.
GlideRecord trigger = new GlideRecord('sys_trigger');
trigger.initialize();
trigger.setValue('state', 0);
trigger.setValue('trigger_type', 0);
trigger.setValue('next_action', new GlideDateTime());
trigger.setValue('job_context', 'fcRuleId=' + ruleId);
trigger.setValue('name', 'Job Reference Migration' + ' Node - ' + new GlideClusterSynchronizer().getSystemID());
trigger.setValue('trigger_class', 'com.glide.db.auxiliary.job.ArchiveRefJob');
trigger.insert();