Browse Source

Refactor migration code

pull/6/head
Marcio Mazza 10 years ago
parent
commit
5241066ab1
  1. 196
      legacy/migration.py

196
legacy/migration.py

@ -85,33 +85,38 @@ def info(msg):
def warn(msg): def warn(msg):
print('WARNING! ' + msg) print('WARNING! ' + msg)
special_transforms = {}
def get_fk_related(field, value, label=None):
def special(model, fieldname): if value is not None:
def wrap(function): try:
special_transforms[model._meta.get_field(fieldname)] = function value = field.related_model.objects.get(id=value)
return function except ObjectDoesNotExist:
return wrap msg = 'FK [%s] not found for value %s ' \
'(in %s %s)' % (
field.name, value,
field.model.__name__, label or '---')
if value == 0:
# we interpret FK == 0 as actually FK == NONE
value = None
warn(msg + ' => using NONE for zero value')
else:
value = make_stub(field.related_model, value)
warn(msg + ' => STUB CREATED')
else:
assert value
return value
@special(Parlamentar, 'unidade_deliberativa') def get_field(model, fieldname):
def none_to_false(obj, value): return model._meta.get_field(fieldname)
# Field is defined as not null in legacy db, but data includes null values
# => transform None to False
if value is None:
warn('null converted to False')
return bool(value)
@special(Participacao, 'composicao') def get_participacao_composicao(obj):
def get_participacao_composicao(obj, value):
# value parameter is ignored
new = Composicao() new = Composicao()
for new_field, value in [('comissao', obj.cod_comissao), for new_field, value in [('comissao', obj.cod_comissao),
('periodo', obj.cod_periodo_comp)]: ('periodo', obj.cod_periodo_comp)]:
model_field = Composicao._meta.get_field(new_field) model_field = Composicao._meta.get_field(new_field)
value = get_related_if_foreignkey(model_field, '???', value) value = get_fk_related(model_field, value)
setattr(new, new_field, value) setattr(new, new_field, value)
previous = Composicao.objects.filter(comissao=new.comissao, periodo=new.periodo) previous = Composicao.objects.filter(comissao=new.comissao, periodo=new.periodo)
if previous: if previous:
@ -122,27 +127,25 @@ def get_participacao_composicao(obj, value):
return new return new
def migrate(obj=appconfs, count_limit=None): SPECIAL_FIELD_MIGRATIONS = {
# warning: model/app migration order is of utmost importance get_field(Participacao, 'composicao'): get_participacao_composicao}
to_delete = []
_do_migrate(obj, to_delete, count_limit)
# exclude logically deleted in legacy base
info('Deleting models with ind_excluido...')
for obj in to_delete:
obj.delete()
def build_special_field_migration(field, get_old_field_value):
def _do_migrate(obj, to_delete, count_limit=None): if field == get_field(Parlamentar, 'unidade_deliberativa'):
if isinstance(obj, AppConfig):
_do_migrate(obj.models.values(), to_delete, count_limit) def none_to_false(obj):
elif isinstance(obj, ModelBase): value = get_old_field_value(obj)
migrate_model(obj, to_delete, count_limit) # Field is defined as not null in legacy db, but data includes null values
elif hasattr(obj, '__iter__'): # => transform None to False
for item in obj: if value is None:
_do_migrate(item, to_delete, count_limit) warn('null converted to False')
else: return bool(value)
raise TypeError('Parameter must be a Model, AppConfig or a sequence of them') return none_to_false
elif field in SPECIAL_FIELD_MIGRATIONS:
return SPECIAL_FIELD_MIGRATIONS[field]
def exec_sql(sql, db='default'): def exec_sql(sql, db='default'):
@ -181,76 +184,99 @@ def make_stub(model, id):
return new return new
def migrate_model(model, to_delete, count_limit=None): class DataMigrator(object):
def __init__(self):
self.field_renames, self.model_renames = get_renames()
def field_migrations(self, model):
renames = self.field_renames[model]
for field in model._meta.fields:
old_field_name = renames.get(field.name)
def get_old_field_value(old):
return getattr(old, old_field_name)
legacy_model_name = model_renames.get(model, model.__name__) special = build_special_field_migration(field, get_old_field_value)
if legacy_model_name.upper() == 'IGNORE': if special:
print('Model ignored: %s' % model.__name__) yield field, special
return elif field.name in renames:
def get_fk_value(old):
old_value = get_old_field_value(old)
old_type = type(old) # not necessarily a model
if hasattr(old_type, '_meta') and \
old_type._meta.pk.name != 'id':
label = old.pk
else:
label = '-- WITHOUT PK --'
return get_fk_related(field, old_value, label)
if isinstance(field, models.ForeignKey):
yield field, get_fk_value
else:
yield field, get_old_field_value
def migrate(self, obj=appconfs):
# warning: model/app migration order is of utmost importance
self.to_delete = []
info('Starting %s migration...' % obj)
self._do_migrate(obj)
# exclude logically deleted in legacy base
info('Deleting models with ind_excluido...')
for obj in self.to_delete:
obj.delete()
def _do_migrate(self, obj):
if isinstance(obj, AppConfig):
models_to_migrate = (model for model in obj.models.values()
if model in self.field_renames)
self._do_migrate(models_to_migrate)
elif isinstance(obj, ModelBase):
self.migrate_model(obj)
elif hasattr(obj, '__iter__'):
for item in obj:
self._do_migrate(item)
else:
raise TypeError('Parameter must be a Model, AppConfig or a sequence of them')
def migrate_model(self, model):
print('Migrating %s...' % model.__name__) print('Migrating %s...' % model.__name__)
legacy_model_name = self.model_renames.get(model, model.__name__)
legacy_model = legacy_app.get_model(legacy_model_name)
legacy_pk_name = legacy_model._meta.pk.name
# clear all model entries # clear all model entries
model.objects.all().delete() model.objects.all().delete()
legacy_model = legacy_app.get_model(legacy_model_name)
old_pk_name = legacy_model._meta.pk.name
# setup migration strategy for tables with or without a pk # setup migration strategy for tables with or without a pk
if old_pk_name == 'id': if legacy_pk_name == 'id':
# There is no pk in the legacy table # There is no pk in the legacy table
def save(new, old):
def get_old_pk(old):
return '-- WITHOUT PK --'
def save(new, id):
new.save() new.save()
old_records = iter_sql_records( old_records = iter_sql_records(
'select * from ' + legacy_model._meta.db_table, 'legacy') 'select * from ' + legacy_model._meta.db_table, 'legacy')
else: else:
def get_old_pk(old): def save(new, old):
return getattr(old, old_pk_name) save_with_id(new, getattr(old, legacy_pk_name))
save = save_with_id
old_records = legacy_model.objects.all().order_by(old_pk_name)[:count_limit] old_records = legacy_model.objects.all().order_by(legacy_pk_name)
# convert old records to new ones # convert old records to new ones
for old in old_records: for old in old_records:
old_pk = get_old_pk(old)
new = model() new = model()
for new_field, old_field in field_renames[model].items(): for new_field, get_value in self.field_migrations(model):
value = getattr(old, old_field) setattr(new, new_field.name, get_value(old))
model_field = model._meta.get_field(new_field) save(new, old)
transform = special_transforms.get(model_field)
if transform:
value = transform(old, value)
else:
# check for a relation
value = get_related_if_foreignkey(model_field, old_pk, value)
setattr(new, new_field, value)
save(new, old_pk)
if getattr(old, 'ind_excluido', False): if getattr(old, 'ind_excluido', False):
to_delete.append(new) self.to_delete.append(new)
def get_related_if_foreignkey(model_field, old_pk, value): # CHECKS #####################################################################
if isinstance(model_field, models.ForeignKey) and value is not None:
try:
value = model_field.related_model.objects.get(id=value)
except ObjectDoesNotExist:
msg = 'FK [%s (%s) : %s] not found for value %s' % (
model_field.model.__name__, old_pk, model_field.name, value)
if value == 0:
# we interpret FK == 0 as actually FK == NONE
value = None
warn(msg + ' => NONE for zero value')
else:
value = make_stub(model_field.related_model, value)
warn(msg + ' => STUB CREATED')
else:
assert value
return value
def get_ind_excluido(obj): def get_ind_excluido(obj):
legacy_model = legacy_app.get_model(type(obj).__name__) legacy_model = legacy_app.get_model(type(obj).__name__)

Loading…
Cancel
Save