2012-03-08 17:14:44 +01:00
|
|
|
import logging
|
2019-03-06 12:19:05 +01:00
|
|
|
from nose.plugins.skip import SkipTest
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
from ckan import model
|
2019-03-06 12:19:05 +01:00
|
|
|
from ckan.model import Session
|
2012-03-08 17:14:44 +01:00
|
|
|
from ckan.lib.base import config
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
# TODO: remove references to old tests
|
2016-01-11 22:35:11 +01:00
|
|
|
try:
|
|
|
|
from ckan.tests import CreateTestData
|
|
|
|
except ImportError:
|
|
|
|
from ckan.tests.legacy import CreateTestData
|
2015-04-07 14:31:45 +02:00
|
|
|
try:
|
|
|
|
from ckan.tests.functional.base import FunctionalTestCase
|
|
|
|
except ImportError:
|
|
|
|
from ckan.tests.legacy.functional.base import FunctionalTestCase
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
from ckanext.harvest.model import HarvestSource, HarvestJob, setup as harvest_model_setup
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class HarvestAuthBaseCase():
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
2012-12-14 15:52:19 +01:00
|
|
|
raise SkipTest()
|
2012-03-08 17:14:44 +01:00
|
|
|
harvest_model_setup()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def teardown_class(cls):
|
|
|
|
pass
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
def _test_auth_not_allowed(self, user_name=None, source=None, status=401):
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
if not source:
|
|
|
|
# Create harvest source
|
2019-03-06 12:19:05 +01:00
|
|
|
source = HarvestSource(url=u'http://test-source.com', type='ckan')
|
2012-03-08 17:14:44 +01:00
|
|
|
Session.add(source)
|
|
|
|
Session.commit()
|
|
|
|
|
|
|
|
if user_name:
|
|
|
|
extra_environ = {'REMOTE_USER': user_name.encode('utf8')}
|
|
|
|
else:
|
|
|
|
extra_environ = {}
|
|
|
|
|
|
|
|
# List
|
2019-03-06 12:19:05 +01:00
|
|
|
self.app.get('/harvest', status=status, extra_environ=extra_environ)
|
2012-03-08 17:14:44 +01:00
|
|
|
# Create
|
2019-03-06 12:19:05 +01:00
|
|
|
self.app.get('/harvest/new', status=status, extra_environ=extra_environ)
|
2012-03-08 17:14:44 +01:00
|
|
|
# Read
|
2019-03-06 12:19:05 +01:00
|
|
|
self.app.get('/harvest/%s' % source.id, status=status, extra_environ=extra_environ)
|
2012-03-08 17:14:44 +01:00
|
|
|
# Edit
|
2019-03-06 12:19:05 +01:00
|
|
|
self.app.get('/harvest/edit/%s' % source.id, status=status, extra_environ=extra_environ)
|
2012-03-08 17:14:44 +01:00
|
|
|
# Refresh
|
2019-03-06 12:19:05 +01:00
|
|
|
self.app.get('/harvest/refresh/%s' % source.id, status=status, extra_environ=extra_environ)
|
2012-03-08 17:14:44 +01:00
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
def _test_auth_allowed(self, user_name, auth_profile=None):
|
2012-03-08 17:14:44 +01:00
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
extra_environ = {'REMOTE_USER': user_name.encode('utf8')}
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
# List
|
|
|
|
res = self.app.get('/harvest', extra_environ=extra_environ)
|
|
|
|
assert 'Harvesting Sources' in res
|
|
|
|
|
|
|
|
# Create
|
|
|
|
res = self.app.get('/harvest/new', extra_environ=extra_environ)
|
|
|
|
assert 'New harvest source' in res
|
|
|
|
if auth_profile == 'publisher':
|
|
|
|
assert 'publisher_id' in res
|
|
|
|
else:
|
2019-03-06 12:19:05 +01:00
|
|
|
assert 'publisher_id' not in res
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
fv = res.forms['source-new']
|
|
|
|
fv['url'] = u'http://test-source.com'
|
|
|
|
fv['type'] = u'ckan'
|
|
|
|
fv['title'] = u'Test harvest source'
|
|
|
|
fv['description'] = u'Test harvest source'
|
|
|
|
fv['config'] = u'{"a":1,"b":2}'
|
|
|
|
|
|
|
|
if auth_profile == 'publisher':
|
|
|
|
fv['publisher_id'] = self.publisher1.id
|
|
|
|
|
|
|
|
res = fv.submit('save', extra_environ=extra_environ)
|
2019-03-06 12:19:05 +01:00
|
|
|
assert 'Error' not in res, res
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
source = Session.query(HarvestSource).first()
|
|
|
|
assert source.url == u'http://test-source.com'
|
|
|
|
assert source.type == u'ckan'
|
|
|
|
|
|
|
|
# Read
|
|
|
|
res = self.app.get('/harvest/%s' % source.id, extra_environ=extra_environ)
|
|
|
|
assert 'Harvest Source Details' in res
|
|
|
|
assert source.id in res
|
|
|
|
assert source.title in res
|
|
|
|
|
|
|
|
# Edit
|
|
|
|
res = self.app.get('/harvest/edit/%s' % source.id, extra_environ=extra_environ)
|
|
|
|
assert 'Edit harvest source' in res
|
|
|
|
if auth_profile == 'publisher':
|
|
|
|
assert 'publisher_id' in res
|
|
|
|
else:
|
2019-03-06 12:19:05 +01:00
|
|
|
assert 'publisher_id' not in res
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
fv = res.forms['source-new']
|
|
|
|
fv['title'] = u'Test harvest source Updated'
|
|
|
|
|
|
|
|
res = fv.submit('save', extra_environ=extra_environ)
|
2019-03-06 12:19:05 +01:00
|
|
|
assert 'Error' not in res, res
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
source = Session.query(HarvestSource).first()
|
|
|
|
assert source.title == u'Test harvest source Updated'
|
|
|
|
|
|
|
|
# Refresh
|
|
|
|
res = self.app.get('/harvest/refresh/%s' % source.id, extra_environ=extra_environ)
|
|
|
|
|
|
|
|
job = Session.query(HarvestJob).first()
|
|
|
|
assert job.source_id == source.id
|
|
|
|
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
class TestAuthDefaultProfile(FunctionalTestCase, HarvestAuthBaseCase):
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
2019-03-06 12:19:05 +01:00
|
|
|
if (config.get('ckan.harvest.auth.profile', '') != ''):
|
2012-03-08 17:14:44 +01:00
|
|
|
raise SkipTest('Skipping default auth profile tests. Set ckan.harvest.auth.profile = \'\' to run them')
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
super(TestAuthDefaultProfile, cls).setup_class()
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
def setup(self):
|
|
|
|
CreateTestData.create()
|
|
|
|
self.sysadmin_user = model.User.get('testsysadmin')
|
|
|
|
self.normal_user = model.User.get('annafan')
|
|
|
|
|
|
|
|
def teardown(self):
|
|
|
|
model.repo.rebuild_db()
|
|
|
|
|
|
|
|
def test_auth_default_profile_sysadmin(self):
|
|
|
|
self._test_auth_allowed(self.sysadmin_user.name)
|
|
|
|
|
|
|
|
def test_auth_default_profile_normal(self):
|
|
|
|
self._test_auth_not_allowed(self.normal_user.name)
|
|
|
|
|
|
|
|
def test_auth_default_profile_notloggedin(self):
|
|
|
|
self._test_auth_not_allowed(status=302)
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
|
|
|
|
class TestAuthPublisherProfile(FunctionalTestCase, HarvestAuthBaseCase):
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
|
|
|
if (config.get('ckan.harvest.auth.profile') != 'publisher'):
|
|
|
|
raise SkipTest('Skipping publisher auth profile tests. Set ckan.harvest.auth.profile = \'publisher\' to run them')
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
super(TestAuthPublisherProfile, cls).setup_class()
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
def setup(self):
|
|
|
|
|
|
|
|
model.Session.remove()
|
|
|
|
CreateTestData.create(auth_profile='publisher')
|
|
|
|
self.sysadmin_user = model.User.get('testsysadmin')
|
2019-03-06 12:19:05 +01:00
|
|
|
self.normal_user = model.User.get('annafan') # Does not belong to a publisher
|
2012-03-08 17:14:44 +01:00
|
|
|
self.publisher1_user = model.User.by_name('russianfan')
|
|
|
|
self.publisher2_user = model.User.by_name('tester')
|
|
|
|
|
|
|
|
# Create two Publishers
|
2019-03-06 12:19:05 +01:00
|
|
|
model.repo.new_revision()
|
|
|
|
self.publisher1 = model.Group(name=u'test-publisher1', title=u'Test Publihser 1', type=u'publisher')
|
2012-03-08 17:14:44 +01:00
|
|
|
Session.add(self.publisher1)
|
2019-03-06 12:19:05 +01:00
|
|
|
self.publisher2 = model.Group(name=u'test-publisher2', title=u'Test Publihser 2', type=u'publisher')
|
2012-03-08 17:14:44 +01:00
|
|
|
Session.add(self.publisher2)
|
|
|
|
|
2019-03-06 12:19:05 +01:00
|
|
|
member1 = model.Member(table_name='user',
|
|
|
|
table_id=self.publisher1_user.id,
|
|
|
|
group=self.publisher1,
|
|
|
|
capacity='admin')
|
2012-03-08 17:14:44 +01:00
|
|
|
Session.add(member1)
|
2019-03-06 12:19:05 +01:00
|
|
|
member2 = model.Member(table_name='user',
|
|
|
|
table_id=self.publisher2_user.id,
|
|
|
|
group=self.publisher2,
|
|
|
|
capacity='admin')
|
2012-03-08 17:14:44 +01:00
|
|
|
Session.add(member2)
|
|
|
|
|
|
|
|
Session.commit()
|
|
|
|
|
|
|
|
def teardown(self):
|
|
|
|
model.repo.rebuild_db()
|
|
|
|
|
|
|
|
def test_auth_publisher_profile_normal(self):
|
|
|
|
self._test_auth_not_allowed(self.normal_user.name)
|
|
|
|
|
|
|
|
def test_auth_publisher_profile_notloggedin(self):
|
|
|
|
self._test_auth_not_allowed(status=302)
|
|
|
|
|
|
|
|
def test_auth_publisher_profile_sysadmin(self):
|
2019-03-06 12:19:05 +01:00
|
|
|
self._test_auth_allowed(self.sysadmin_user.name, auth_profile='publisher')
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
def test_auth_publisher_profile_publisher(self):
|
2019-03-06 12:19:05 +01:00
|
|
|
self._test_auth_allowed(self.publisher1_user.name, auth_profile='publisher')
|
2012-03-08 17:14:44 +01:00
|
|
|
|
|
|
|
def test_auth_publisher_profile_different_publisher(self):
|
|
|
|
|
|
|
|
# Create a source for publisher 1
|
2019-03-06 12:19:05 +01:00
|
|
|
source = HarvestSource(url=u'http://test-source.com', type='ckan',
|
2012-03-08 17:14:44 +01:00
|
|
|
publisher_id=self.publisher1.id)
|
|
|
|
Session.add(source)
|
|
|
|
Session.commit()
|
|
|
|
|
|
|
|
extra_environ = {'REMOTE_USER': self.publisher2_user.name.encode('utf8')}
|
|
|
|
|
|
|
|
# List (Publihsers can see the sources list)
|
|
|
|
res = self.app.get('/harvest', extra_environ=extra_environ)
|
|
|
|
assert 'Harvesting Sources' in res
|
|
|
|
# Create
|
|
|
|
res = self.app.get('/harvest/new', extra_environ=extra_environ)
|
|
|
|
assert 'New harvest source' in res
|
|
|
|
assert 'publisher_id' in res
|
|
|
|
|
|
|
|
# Check that this publihser is not allowed to manage sources from other publishers
|
|
|
|
status = 401
|
|
|
|
# Read
|
|
|
|
res = self.app.get('/harvest/%s' % source.id, status=status, extra_environ=extra_environ)
|
|
|
|
# Edit
|
|
|
|
res = self.app.get('/harvest/edit/%s' % source.id, status=status, extra_environ=extra_environ)
|
|
|
|
# Refresh
|
|
|
|
res = self.app.get('/harvest/refresh/%s' % source.id, status=status, extra_environ=extra_environ)
|