In order to actually test the new app, you need to update the database with the new schema.
However, Django's syncdb does not update existing tables, only adds missing ones.
There are a few database migration tools out there, but south is by far the most common one.
South excels on small changes, like adding a field or removing a constraint. In the early stages of app development, however, you might make rapid large changes and in the same time, not care too much about the existing data in the database (for that particular app).
So in the process of your development, you might want to do some kind of "app resetting", meaning - 'drop all the tables for this app and recreate them according to the new models definition'.
As common as it seemed to me, I couldn't find a solution for that procedure in neither the Django native api, nor in the community.
The closest options are sqlclear, which prints the sql statements to drop tables for an app and flush, which actually resets the entire database. Obviously these solutions are not south-friendly.
But I wanted something that actually resets a single app and also plays nicely with south.
Enters "south_reset", a management command that is just a few south commands sewn together, but I still found it useful enough to share.
The usage is pretty straight forward, just list the app names you want to reset.
The optional "soft" flag means that the migrations are merged to a single initial migration without actually running them (just faking it), so the data persists for the app. This is useful when you make lots of migrations and you want to get rid of the clutter, but still keep the existing data in the database.
Note that you should be very careful with this command if you are deploying the code somewhere, you might get ghost migrations.
So, with no further ado, here is the gist:
If you are not familiar with management commands, you need to put this script under a "management/commands" folder in any of your apps. more information here.
So, with no further ado, here is the gist:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from optparse import make_option | |
from os.path import dirname, join, abspath, basename | |
import shutil | |
import subprocess | |
from django.core.management.base import AppCommand | |
class Command(AppCommand): | |
help = "Reset south migrations for app" | |
option_list = AppCommand.option_list + ( | |
make_option('--soft', | |
action = 'store_true', | |
dest = 'soft', | |
default = False, | |
help = "Just merge the migrations and update the south DB accordingly, doesn't change actual app tables"), | |
make_option('--database', | |
action = 'store', | |
dest = 'database', | |
default = 'default', | |
help = 'Nominates a database to synchronize. Defaults to the "default" database.'), | |
) | |
def handle_app(self, app, soft = False, verbosity = 1, **options): | |
printer = Printer(verbosity = verbosity) | |
def manage_call(*a, **kw): | |
b = ['--%s%s' % (key, ('=' + value) if value!=True else '') for key, value in kw.iteritems() if value != False] | |
final_args = ['python', 'manage.py'] + list(a) + b | |
printer('Calling: '+ ' '.join(final_args), 3) | |
return subprocess.call(final_args) | |
app_name = basename(dirname(app.__file__)) | |
printer('Resetting %s (%s)' % (app_name, 'soft' if soft else 'hard')) | |
migrations_dir = abspath(join(dirname(app.__file__), 'migrations')) | |
printer('Rolling back migrations', 2) | |
manage_call('migrate', app_name, 'zero', fake = soft, verbosity = verbosity, database = options['database']) | |
printer('Deleting migrations folder', 2) | |
try: | |
shutil.rmtree(migrations_dir) | |
except Exception, ex: | |
printer('Failed to delete migrations folder, probably does not exist') | |
pass | |
printer('Creating new initial migration', 2) | |
manage_call('schemamigration', app_name, initial = True, verbosity = verbosity) | |
printer('Migrating to new initial', 2) | |
manage_call('migrate', app_name, fake = soft, verbosity = verbosity, database = options['database']) | |
printer('Done') | |
class Printer(object): | |
def __init__(self, verbosity = 1): | |
self.verbosity = int(verbosity) | |
def __call__(self, s, verbosity = 1, prefix = '***'): | |
if verbosity <= self.verbosity: | |
print prefix, s |