aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--mediagoblin/db/migrations.py39
-rw-r--r--mediagoblin/db/models.py15
-rw-r--r--mediagoblin/decorators.py4
-rw-r--r--mediagoblin/gmg_commands/__init__.py66
-rw-r--r--mediagoblin/gmg_commands/migrate.py45
-rw-r--r--mediagoblin/gmg_commands/shell.py54
-rw-r--r--mediagoblin/gmg_commands/util.py45
-rw-r--r--mediagoblin/submit/views.py2
-rw-r--r--mediagoblin/templates/mediagoblin/user_pages/media.html27
-rw-r--r--mediagoblin/user_pages/views.py2
10 files changed, 225 insertions, 74 deletions
diff --git a/mediagoblin/db/migrations.py b/mediagoblin/db/migrations.py
new file mode 100644
index 00000000..d035b15b
--- /dev/null
+++ b/mediagoblin/db/migrations.py
@@ -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']
diff --git a/mediagoblin/db/models.py b/mediagoblin/db/models.py
index 8e7889eb..37420834 100644
--- a/mediagoblin/db/models.py
+++ b/mediagoblin/db/models.py
@@ -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 = [
@@ -123,17 +127,22 @@ class MediaEntry(Document):
Use a slug if we have one, else use our '_id'.
"""
+ uploader = self.uploader()
+
if self.get('slug'):
return urlgen(
'mediagoblin.user_pages.media_home',
- user=self['uploader']['username'],
+ user=uploader['username'],
media=self['slug'])
else:
return urlgen(
'mediagoblin.user_pages.media_home',
- user=self['uploader']['username'],
+ user=uploader['username'],
media=unicode(self['_id']))
+ def uploader(self):
+ return self.db.User.find_one({'_id': self['uploader']})
+
REGISTER_MODELS = [MediaEntry, User]
diff --git a/mediagoblin/decorators.py b/mediagoblin/decorators.py
index 34a471cb..ff3f0b5e 100644
--- a/mediagoblin/decorators.py
+++ b/mediagoblin/decorators.py
@@ -80,7 +80,7 @@ def get_user_media_entry(controller):
media = request.db.MediaEntry.find_one(
{'slug': request.matchdict['media'],
'state': 'processed',
- 'uploader._id': user['_id']})
+ 'uploader': user['_id']})
# no media via slug? Grab it via ObjectId
if not media:
@@ -88,7 +88,7 @@ def get_user_media_entry(controller):
media = request.db.MediaEntry.find_one(
{'_id': ObjectId(request.matchdict['media']),
'state': 'processed',
- 'uploader._id': user['_id']})
+ 'uploader': user['_id']})
except InvalidId:
return exc.HTTPNotFound()
diff --git a/mediagoblin/gmg_commands/__init__.py b/mediagoblin/gmg_commands/__init__.py
index 04e2ab6c..d1f7bfc1 100644
--- a/mediagoblin/gmg_commands/__init__.py
+++ b/mediagoblin/gmg_commands/__init__.py
@@ -14,71 +14,23 @@
# 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/>.
-import code
import argparse
-import os
-from paste.deploy.loadwsgi import NicerConfigParser
-
-from mediagoblin.celery_setup import setup_celery_from_config
-from mediagoblin import app, util
-from mediagoblin import globals as mgoblin_globals
+from mediagoblin import util as mg_util
SUBCOMMAND_MAP = {
'shell': {
- 'setup': 'mediagoblin.gmg_commands:shell_parser_setup',
- 'func': 'mediagoblin.gmg_commands:shell',
+ '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'},
}
-def shell_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.")
-
-
-SHELL_BANNER = """\
-GNU MediaGoblin shell!
-----------------------
-Available vars:
- - mgoblin_app: instantiated mediagoblin application
- - mgoblin_globals: mediagoblin.globals
- - db: database instance
-"""
-
-
-def shell(args):
- """
- """
- # Duplicated from from_celery.py, remove when we have the generic util
- parser = NicerConfigParser(args.conf_file)
- parser.read(args.conf_file)
- parser._defaults.setdefault(
- 'here', os.path.dirname(os.path.abspath(args.conf_file)))
- parser._defaults.setdefault(
- '__file__', os.path.abspath(args.conf_file))
-
- mgoblin_section = dict(parser.items(args.app_section))
- mgoblin_conf = dict(
- [(section_name, dict(parser.items(section_name)))
- for section_name in parser.sections()])
-
- mgoblin_app = app.paste_app_factory(
- mgoblin_conf, **mgoblin_section)
-
- code.interact(
- banner=SHELL_BANNER,
- local={
- 'mgoblin_app': mgoblin_app,
- 'mgoblin_globals': mgoblin_globals,
- 'db': mgoblin_globals.database})
-
-
def main_cli():
parser = argparse.ArgumentParser(
description='GNU MediaGoblin utilities.')
@@ -91,8 +43,8 @@ def main_cli():
else:
subparser = subparsers.add_parser(command_name)
- setup_func = util.import_component(command_struct['setup'])
- exec_func = util.import_component(command_struct['func'])
+ setup_func = mg_util.import_component(command_struct['setup'])
+ exec_func = mg_util.import_component(command_struct['func'])
setup_func(subparser)
diff --git a/mediagoblin/gmg_commands/migrate.py b/mediagoblin/gmg_commands/migrate.py
new file mode 100644
index 00000000..e04fb343
--- /dev/null
+++ b/mediagoblin/gmg_commands/migrate.py
@@ -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."
diff --git a/mediagoblin/gmg_commands/shell.py b/mediagoblin/gmg_commands/shell.py
new file mode 100644
index 00000000..9c0259de
--- /dev/null
+++ b/mediagoblin/gmg_commands/shell.py
@@ -0,0 +1,54 @@
+# 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/>.
+
+
+import code
+
+from mediagoblin import globals as mgoblin_globals
+from mediagoblin.gmg_commands import util as commands_util
+
+
+def shell_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.")
+
+
+SHELL_BANNER = """\
+GNU MediaGoblin shell!
+----------------------
+Available vars:
+ - mgoblin_app: instantiated mediagoblin application
+ - mgoblin_globals: mediagoblin.globals
+ - db: database instance
+"""
+
+
+def shell(args):
+ """
+ Setup a shell for the user
+ """
+ mgoblin_app = commands_util.setup_app(args)
+
+ code.interact(
+ banner=SHELL_BANNER,
+ local={
+ 'mgoblin_app': mgoblin_app,
+ 'mgoblin_globals': mgoblin_globals,
+ 'db': mgoblin_globals.database})
diff --git a/mediagoblin/gmg_commands/util.py b/mediagoblin/gmg_commands/util.py
new file mode 100644
index 00000000..41a21a1e
--- /dev/null
+++ b/mediagoblin/gmg_commands/util.py
@@ -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/>.
+
+
+import os
+
+from paste.deploy.loadwsgi import NicerConfigParser
+
+from mediagoblin import app
+
+
+def setup_app(args):
+ """
+ Setup the application after reading the mediagoblin config files
+ """
+ # Duplicated from from_celery.py, remove when we have the generic util
+ parser = NicerConfigParser(args.conf_file)
+ parser.read(args.conf_file)
+ parser._defaults.setdefault(
+ 'here', os.path.dirname(os.path.abspath(args.conf_file)))
+ parser._defaults.setdefault(
+ '__file__', os.path.abspath(args.conf_file))
+
+ mgoblin_section = dict(parser.items(args.app_section))
+ mgoblin_conf = dict(
+ [(section_name, dict(parser.items(section_name)))
+ for section_name in parser.sections()])
+
+ mgoblin_app = app.paste_app_factory(
+ mgoblin_conf, **mgoblin_section)
+
+ return mgoblin_app
diff --git a/mediagoblin/submit/views.py b/mediagoblin/submit/views.py
index 95a416e2..262f2b12 100644
--- a/mediagoblin/submit/views.py
+++ b/mediagoblin/submit/views.py
@@ -46,7 +46,7 @@ def submit_start(request):
entry['title'] = request.POST['title'] or unicode(splitext(filename)[0])
entry['description'] = request.POST.get('description')
entry['media_type'] = u'image' # heh
- entry['uploader'] = request.user
+ entry['uploader'] = request.user['_id']
# Save, just so we can get the entry id for the sake of using
# it to generate the file path
diff --git a/mediagoblin/templates/mediagoblin/user_pages/media.html b/mediagoblin/templates/mediagoblin/user_pages/media.html
index e07cee44..036bf726 100644
--- a/mediagoblin/templates/mediagoblin/user_pages/media.html
+++ b/mediagoblin/templates/mediagoblin/user_pages/media.html
@@ -20,20 +20,27 @@
{# temporarily, an "image gallery" that isn't one really ;) #}
{% if media %}
- <h1>Media details for <a
- href="{{ request.urlgen('mediagoblin.user_pages.user_home',
- user= media.uploader.username) }}">{{media.uploader.username}}</a>
- / {{media.title}}
+ <h1>
+ Media details for
+ <a href="{{ request.urlgen(
+ 'mediagoblin.user_pages.user_home',
+ user=media.uploader().username) }}">
+ {{- media.uploader().username }}</a>
+ / {{media.title}}
</h1>
<div>
<img src="{{ request.app.public_store.file_url(
media.media_files.main) }}" />
-
- <br/>Uploaded on {{ "%4d-%02d-%02d"|format(media.created.year,
- media.created.month,media.created.day)}} by <a
- href="{{ request.urlgen('mediagoblin.user_pages.user_home',
- user= media.uploader.username) }}">{{media.uploader.username}}</a>
- <br/>Description: {{media.description}}
+ <br />
+ Uploaded on
+ {{ "%4d-%02d-%02d"|format(media.created.year,
+ media.created.month, media.created.day) }}
+ by
+ <a href="{{ request.urlgen('mediagoblin.user_pages.user_home',
+ user= media.uploader().username) }}">
+ {{- media.uploader().username }}</a>
+ <br />
+ Description: {{ media.description }}
</div>
{% else %}
<p>Sorry, no such media found.<p/>
diff --git a/mediagoblin/user_pages/views.py b/mediagoblin/user_pages/views.py
index 0d9833cd..41bdb402 100644
--- a/mediagoblin/user_pages/views.py
+++ b/mediagoblin/user_pages/views.py
@@ -31,7 +31,7 @@ def user_home(request, page):
return exc.HTTPNotFound()
cursor = request.db.MediaEntry.find(
- {'uploader': user,
+ {'uploader': user['_id'],
'state': 'processed'}).sort('created', DESCENDING)
pagination = Pagination(page, cursor)