1 # GNU MediaGoblin -- federated, autonomous media hosting
2 # Copyright (C) 2011, 2012 MediaGoblin contributors. See AUTHORS.
4 # This program is free software: you can redistribute it and/or modify
5 # it under the terms of the GNU Affero General Public License as published by
6 # the Free Software Foundation, either version 3 of the License, or
7 # (at your option) any later version.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU Affero General Public License for more details.
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
23 from functools
import wraps
25 from paste
.deploy
import loadapp
26 from webtest
import TestApp
28 from mediagoblin
import mg_globals
29 from mediagoblin
.db
.models
import User
, MediaEntry
, Collection
30 from mediagoblin
.tools
import testing
31 from mediagoblin
.init
.config
import read_mediagoblin_config
32 from mediagoblin
.db
.base
import Session
33 from mediagoblin
.meddleware
import BaseMeddleware
34 from mediagoblin
.auth
.lib
import bcrypt_gen_password_hash
35 from mediagoblin
.gmg_commands
.dbupdate
import run_dbupdate
38 MEDIAGOBLIN_TEST_DB_NAME
= u
'__mediagoblin_tests__'
39 TEST_SERVER_CONFIG
= pkg_resources
.resource_filename(
40 'mediagoblin.tests', 'test_paste.ini')
41 TEST_APP_CONFIG
= pkg_resources
.resource_filename(
42 'mediagoblin.tests', 'test_mgoblin_app.ini')
43 TEST_USER_DEV
= pkg_resources
.resource_filename(
44 'mediagoblin.tests', 'test_user_dev')
47 USER_DEV_DIRECTORIES_TO_SETUP
= ['media/public', 'media/queue']
50 class TestingMeddleware(BaseMeddleware
):
52 Meddleware for the Unit tests
54 It might make sense to perform some tests on all
55 requests/responses. Or prepare them in a special
56 manner. For example all html responses could be tested
57 for being valid html *after* being rendered.
59 This module is getting inserted at the front of the
60 meddleware list, which means: requests are handed here
61 first, responses last. So this wraps up the "normal"
64 If you need to add a test, either add it directly to
65 the appropiate process_request or process_response, or
66 create a new method and call it from process_*.
69 def process_response(self
, request
, response
):
70 # All following tests should be for html only!
71 if getattr(response
, 'content_type', None) != "text/html":
75 # If the template contains a reference to
76 # /mgoblin_static/ instead of using
77 # /request.staticdirect(), error out here.
78 # This could probably be implemented as a grep on
79 # the shipped templates easier...
80 if response
.text
.find("/mgoblin_static/") >= 0:
82 "Response HTML contains reference to /mgoblin_static/ "
83 "instead of staticdirect. Request was for: "
89 def get_app(request
, paste_config
=None, mgoblin_config
=None):
90 """Create a MediaGoblin app for testing.
93 - request: Not an http request, but a pytest fixture request. We
94 use this to make temporary directories that pytest
95 automatically cleans up as needed.
96 - paste_config: particular paste config used by this application.
97 - mgoblin_config: particular mediagoblin config used by this
100 paste_config
= paste_config
or TEST_SERVER_CONFIG
101 mgoblin_config
= mgoblin_config
or TEST_APP_CONFIG
103 # This is the directory we're copying the paste/mgoblin config stuff into
104 run_dir
= request
.config
._tmpdirhandler
.mktemp(
105 'mgoblin_app', numbered
=True)
106 user_dev_dir
= run_dir
.mkdir('test_user_dev').strpath
108 new_paste_config
= run_dir
.join('paste.ini').strpath
109 new_mgoblin_config
= run_dir
.join('mediagoblin.ini').strpath
110 shutil
.copyfile(paste_config
, new_paste_config
)
111 shutil
.copyfile(mgoblin_config
, new_mgoblin_config
)
116 # install user_dev directories
117 for directory
in USER_DEV_DIRECTORIES_TO_SETUP
:
118 full_dir
= os
.path
.join(user_dev_dir
, directory
)
119 os
.makedirs(full_dir
)
122 global_config
, validation_result
= read_mediagoblin_config(new_mgoblin_config
)
123 app_config
= global_config
['mediagoblin']
125 # Run database setup/migrations
126 run_dbupdate(app_config
, global_config
)
128 # setup app and return
130 'config:' + new_paste_config
)
132 # Insert the TestingMeddleware, which can do some
133 # sanity checks on every request/response.
134 # Doing it this way is probably not the cleanest way.
135 # We'll fix it, when we have plugins!
136 mg_globals
.app
.meddleware
.insert(0, TestingMeddleware(mg_globals
.app
))
138 app
= TestApp(test_app
)
143 def install_fixtures_simple(db
, fixtures
):
145 Very simply install fixtures in the database
147 for collection_name
, collection_fixtures
in fixtures
.iteritems():
148 collection
= db
[collection_name
]
149 for fixture
in collection_fixtures
:
150 collection
.insert(fixture
)
153 def assert_db_meets_expected(db
, expected
):
155 Assert a database contains the things we expect it to.
157 Objects are found via 'id', so you should make sure your document
161 - db: pymongo or mongokit database connection
162 - expected: the data we expect. Formatted like:
163 {'collection_name': [
165 'some_field': 'some_value'},]}
167 for collection_name
, collection_data
in expected
.iteritems():
168 collection
= db
[collection_name
]
169 for expected_document
in collection_data
:
170 document
= collection
.find_one({'id': expected_document
['id']})
171 assert document
is not None # make sure it exists
172 assert document
== expected_document
# make sure it matches
175 def fixture_add_user(username
=u
'chris', password
=u
'toast',
177 # Reuse existing user or create a new one
178 test_user
= User
.query
.filter_by(username
=username
).first()
179 if test_user
is None:
181 test_user
.username
= username
182 test_user
.email
= username
+ u
'@example.com'
183 if password
is not None:
184 test_user
.pw_hash
= bcrypt_gen_password_hash(password
)
186 test_user
.email_verified
= True
187 test_user
.status
= u
'active'
192 test_user
= User
.query
.filter_by(username
=username
).first()
194 # ... and detach from session:
195 Session
.expunge(test_user
)
200 def fixture_media_entry(title
=u
"Some title", slug
=None,
201 uploader
=None, save
=True, gen_slug
=True):
205 entry
.uploader
= uploader
or fixture_add_user().id
206 entry
.media_type
= u
'image'
209 entry
.generate_slug()
216 def fixture_add_collection(name
=u
"My first Collection", user
=None):
218 user
= fixture_add_user()
219 coll
= Collection
.query
.filter_by(creator
=user
.id, title
=name
).first()
223 coll
.creator
= user
.id
229 Session
.refresh(coll
)
231 # ... and detach from session:
232 Session
.expunge(coll
)