User migration works (but the rest of the system isn't updated for new user setup...
authorChristopher Allan Webber <cwebber@dustycloud.org>
Sun, 22 May 2011 15:52:53 +0000 (10:52 -0500)
committerChristopher Allan Webber <cwebber@dustycloud.org>
Sun, 22 May 2011 15:52:53 +0000 (10:52 -0500)
mediagoblin/db/migrations.py [new file with mode: 0644]
mediagoblin/db/models.py
mediagoblin/gmg_commands/__init__.py
mediagoblin/gmg_commands/migrate.py [new file with mode: 0644]

diff --git a/mediagoblin/db/migrations.py b/mediagoblin/db/migrations.py
new file mode 100644 (file)
index 0000000..d035b15
--- /dev/null
@@ -0,0 +1,39 @@
+# GNU MediaGoblin -- federated, autonomous media hosting
+# Copyright (C) 2011 Free Software Foundation, Inc
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU Affero General Public License for more details.
+#
+# You should have received a copy of the GNU Affero General Public License
+# along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+from mongokit import DocumentMigration
+
+from mediagoblin import globals as mediagoblin_globals
+
+
+class MediaEntryMigration(DocumentMigration):
+    def allmigration01_uploader_to_reference(self):
+        """
+        Old MediaEntry['uploader'] accidentally embedded the User instead
+        of referencing it.  Fix that!
+        """
+        # uploader is an associative array
+        self.target = {'uploader': {'$type': 3}}
+        if not self.status:
+            for doc in self.collection.find(self.target):
+                self.update = {
+                    '$set': {
+                        'uploader': doc['uploader']['_id']}}
+                self.collection.update(
+                    self.target, self.update, multi=True, safe=True)
+
+
+MIGRATE_CLASSES = ['MediaEntry']
index 8e7889eba5a06f512bd7ab31f055d41843a7ef0d..3fc8d9e8d512b7fa7e4f5c6dfdfa39ed2ecde389 100644 (file)
@@ -21,6 +21,8 @@ from mongokit import Document, Set
 from mediagoblin import util
 from mediagoblin.auth import lib as auth_lib
 from mediagoblin import globals as mediagoblin_globals
+from mediagoblin.db import migrations
+from mediagoblin.db.util import ObjectId
 
 ###################
 # Custom validators
@@ -67,7 +69,7 @@ class MediaEntry(Document):
     __collection__ = 'media_entries'
 
     structure = {
-        'uploader': User,
+        'uploader': ObjectId,
         'title': unicode,
         'slug': unicode,
         'created': datetime.datetime,
@@ -99,6 +101,8 @@ class MediaEntry(Document):
         'created': datetime.datetime.utcnow,
         'state': u'unprocessed'}
 
+    migration_handler = migrations.MediaEntryMigration
+
     # Actually we should referene uniqueness by uploader, but we
     # should fix http://bugs.foocorp.net/issues/340 first.
     # indexes = [
index 9ece2ec5a3cf0c2040ee670175cd687fe3969101..d1f7bfc127992c1315287d2da69ae1c70ddd3753 100644 (file)
@@ -24,6 +24,10 @@ SUBCOMMAND_MAP = {
         'setup': 'mediagoblin.gmg_commands.shell:shell_parser_setup',
         'func': 'mediagoblin.gmg_commands.shell:shell',
         'help': 'Run a shell with some tools pre-setup'},
+    'migrate': {
+        'setup': 'mediagoblin.gmg_commands.migrate:migrate_parser_setup',
+        'func': 'mediagoblin.gmg_commands.migrate:migrate',
+        'help': 'Apply all unapplied bulk migrations to the database'},
     }
 
 
diff --git a/mediagoblin/gmg_commands/migrate.py b/mediagoblin/gmg_commands/migrate.py
new file mode 100644 (file)
index 0000000..e04fb34
--- /dev/null
@@ -0,0 +1,45 @@
+# GNU MediaGoblin -- federated, autonomous media hosting
+# Copyright (C) 2011 Free Software Foundation, Inc
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU Affero General Public License for more details.
+#
+# You should have received a copy of the GNU Affero General Public License
+# along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+
+from mediagoblin.db import migrations
+from mediagoblin.gmg_commands import util as commands_util
+from mediagoblin import globals as mgoblin_globals
+
+
+def migrate_parser_setup(subparser):
+    subparser.add_argument(
+        '-cf', '--conf_file', default='mediagoblin.ini',
+        help="Config file used to set up environment")
+    subparser.add_argument(
+        '-cs', '--app_section', default='app:mediagoblin',
+        help="Section of the config file where the app config is stored.")
+
+
+def migrate(args):
+    mgoblin_app = commands_util.setup_app(args)
+    print "Applying migrations..."
+
+    for model_name in migrations.MIGRATE_CLASSES:
+        model = getattr(mgoblin_app.db, model_name)
+
+        if not hasattr(model, 'migration_handler') or not model.collection:
+            continue
+
+        migration = model.migration_handler(model)
+        migration.migrate_all(collection=model.collection)
+            
+    print "... done."