2012-11-30 18:02:06 +01:00
|
|
|
import json
|
2012-11-30 15:03:04 +01:00
|
|
|
import copy
|
2013-09-17 17:49:19 +02:00
|
|
|
import factories
|
|
|
|
import unittest
|
2012-11-30 15:03:04 +01:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
try:
|
|
|
|
from ckan.tests import factories as ckan_factories
|
|
|
|
from ckan.tests.helpers import _get_test_app, reset_db
|
|
|
|
except ImportError:
|
|
|
|
from ckan.new_tests import factories as ckan_factories
|
|
|
|
from ckan.new_tests.helpers import _get_test_app, reset_db
|
2012-12-14 15:52:19 +01:00
|
|
|
from ckan import plugins as p
|
2013-09-17 17:49:19 +02:00
|
|
|
from ckan.plugins import toolkit
|
2015-10-21 18:26:57 +02:00
|
|
|
from ckan import model
|
|
|
|
|
2012-12-14 15:52:19 +01:00
|
|
|
from ckanext.harvest.interfaces import IHarvester
|
2012-11-30 15:03:04 +01:00
|
|
|
import ckanext.harvest.model as harvest_model
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
|
|
|
|
def call_action_api(action, apikey=None, status=200, **kwargs):
|
2015-04-07 14:31:45 +02:00
|
|
|
'''POST an HTTP request to the CKAN API and return the result.
|
|
|
|
|
|
|
|
Any additional keyword arguments that you pass to this function as **kwargs
|
|
|
|
are posted as params to the API.
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
package_dict = call_action_api('package_create', apikey=apikey,
|
2015-04-07 14:31:45 +02:00
|
|
|
name='my_package')
|
|
|
|
assert package_dict['name'] == 'my_package'
|
|
|
|
|
|
|
|
num_followers = post(app, 'user_follower_count', id='annafan')
|
|
|
|
|
|
|
|
If you are expecting an error from the API and want to check the contents
|
|
|
|
of the error dict, you have to use the status param otherwise an exception
|
|
|
|
will be raised:
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
error_dict = call_action_api('group_activity_list', status=403,
|
2015-04-07 14:31:45 +02:00
|
|
|
id='invalid_id')
|
|
|
|
assert error_dict['message'] == 'Access Denied'
|
|
|
|
|
|
|
|
:param action: the action to post to, e.g. 'package_create'
|
|
|
|
:type action: string
|
|
|
|
|
|
|
|
:param apikey: the API key to put in the Authorization header of the post
|
|
|
|
(optional, default: None)
|
|
|
|
:type apikey: string
|
|
|
|
|
|
|
|
:param status: the HTTP status code expected in the response from the CKAN
|
|
|
|
API, e.g. 403, if a different status code is received an exception will
|
|
|
|
be raised (optional, default: 200)
|
|
|
|
:type status: int
|
|
|
|
|
|
|
|
:param **kwargs: any other keyword arguments passed to this function will
|
|
|
|
be posted to the API as params
|
|
|
|
|
|
|
|
:raises paste.fixture.AppError: if the HTTP status code of the response
|
|
|
|
from the CKAN API is different from the status param passed to this
|
|
|
|
function
|
|
|
|
|
|
|
|
:returns: the 'result' or 'error' dictionary from the CKAN API response
|
|
|
|
:rtype: dictionary
|
|
|
|
|
|
|
|
'''
|
|
|
|
params = json.dumps(kwargs)
|
2015-10-21 18:26:57 +02:00
|
|
|
app = _get_test_app()
|
2015-04-07 14:31:45 +02:00
|
|
|
response = app.post('/api/action/{0}'.format(action), params=params,
|
2015-10-21 18:26:57 +02:00
|
|
|
extra_environ={'Authorization': str(apikey)},
|
|
|
|
status=status)
|
2015-04-07 14:31:45 +02:00
|
|
|
|
|
|
|
if status in (200,):
|
|
|
|
assert response.json['success'] is True
|
|
|
|
return response.json['result']
|
|
|
|
else:
|
|
|
|
assert response.json['success'] is False
|
|
|
|
return response.json['error']
|
|
|
|
|
2013-09-17 17:49:19 +02:00
|
|
|
|
2012-12-14 15:52:19 +01:00
|
|
|
class MockHarvesterForActionTests(p.SingletonPlugin):
|
|
|
|
p.implements(IHarvester)
|
2015-10-21 18:26:57 +02:00
|
|
|
|
2012-12-14 15:52:19 +01:00
|
|
|
def info(self):
|
2015-10-21 18:26:57 +02:00
|
|
|
return {'name': 'test-for-action',
|
|
|
|
'title': 'Test for action',
|
|
|
|
'description': 'test'}
|
2012-12-14 15:52:19 +01:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
def validate_config(self, config):
|
2012-12-14 15:52:19 +01:00
|
|
|
if not config:
|
|
|
|
return config
|
|
|
|
|
|
|
|
try:
|
|
|
|
config_obj = json.loads(config)
|
|
|
|
|
|
|
|
if 'custom_option' in config_obj:
|
2015-10-21 18:26:57 +02:00
|
|
|
if not isinstance(config_obj['custom_option'], list):
|
2012-12-14 15:52:19 +01:00
|
|
|
raise ValueError('custom_option must be a list')
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
except ValueError, e:
|
2012-12-14 15:52:19 +01:00
|
|
|
raise e
|
|
|
|
|
|
|
|
return config
|
|
|
|
|
|
|
|
def gather_stage(self, harvest_job):
|
|
|
|
return []
|
|
|
|
|
|
|
|
def fetch_stage(self, harvest_object):
|
|
|
|
return True
|
|
|
|
|
|
|
|
def import_stage(self, harvest_object):
|
|
|
|
return True
|
2012-11-30 15:03:04 +01:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
|
|
|
|
class FunctionalTestBaseWithoutClearBetweenTests(object):
|
|
|
|
''' Functional tests should normally derive from
|
|
|
|
ckan.lib.helpers.FunctionalTestBase, but these are legacy tests so this
|
|
|
|
class is a compromise. This version doesn't call reset_db before every
|
|
|
|
test, because these tests are designed with fixtures created in
|
|
|
|
setup_class.'''
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
|
|
|
reset_db()
|
|
|
|
harvest_model.setup()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def teardown_class(cls):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class HarvestSourceActionBase(FunctionalTestBaseWithoutClearBetweenTests):
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
2015-10-21 18:26:57 +02:00
|
|
|
super(HarvestSourceActionBase, cls).setup_class()
|
2012-11-30 15:03:04 +01:00
|
|
|
harvest_model.setup()
|
2015-10-21 18:26:57 +02:00
|
|
|
|
|
|
|
cls.sysadmin = ckan_factories.Sysadmin()
|
|
|
|
|
|
|
|
cls.default_source_dict = {
|
|
|
|
"url": "http://test.action.com",
|
|
|
|
"name": "test-source-action",
|
|
|
|
"title": "Test source action",
|
|
|
|
"notes": "Test source action desc",
|
|
|
|
"source_type": "test-for-action",
|
|
|
|
"frequency": "MANUAL",
|
|
|
|
"config": json.dumps({"custom_option": ["a", "b"]})
|
2012-11-30 15:03:04 +01:00
|
|
|
}
|
|
|
|
|
2015-04-07 14:31:45 +02:00
|
|
|
if not p.plugin_loaded('test_action_harvester'):
|
|
|
|
p.load('test_action_harvester')
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def teardown_class(cls):
|
2015-10-21 18:26:57 +02:00
|
|
|
super(HarvestSourceActionBase, cls).teardown_class()
|
2012-11-30 15:03:04 +01:00
|
|
|
|
2015-04-07 14:31:45 +02:00
|
|
|
p.unload('test_action_harvester')
|
|
|
|
|
2012-11-30 15:03:04 +01:00
|
|
|
def test_invalid_missing_values(self):
|
|
|
|
|
|
|
|
source_dict = {}
|
|
|
|
if 'id' in self.default_source_dict:
|
|
|
|
source_dict['id'] = self.default_source_dict['id']
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api(self.action,
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-12-14 15:52:19 +01:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
for key in ('name', 'title', 'url', 'source_type'):
|
2012-11-30 15:03:04 +01:00
|
|
|
assert result[key] == [u'Missing value']
|
|
|
|
|
|
|
|
def test_invalid_unknown_type(self):
|
|
|
|
|
|
|
|
source_dict = copy.deepcopy(self.default_source_dict)
|
|
|
|
source_dict['source_type'] = 'unknown'
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api(self.action,
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
assert 'source_type' in result
|
|
|
|
assert u'Unknown harvester type' in result['source_type'][0]
|
|
|
|
|
|
|
|
def test_invalid_unknown_frequency(self):
|
|
|
|
wrong_frequency = 'ANNUALLY'
|
|
|
|
source_dict = copy.deepcopy(self.default_source_dict)
|
|
|
|
source_dict['frequency'] = wrong_frequency
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api(self.action,
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
assert 'frequency' in result
|
|
|
|
assert u'Frequency {0} not recognised'.format(wrong_frequency) in result['frequency'][0]
|
|
|
|
|
2012-11-30 18:02:06 +01:00
|
|
|
def test_invalid_wrong_configuration(self):
|
|
|
|
|
|
|
|
source_dict = copy.deepcopy(self.default_source_dict)
|
|
|
|
source_dict['config'] = 'not_json'
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api(self.action,
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-11-30 18:02:06 +01:00
|
|
|
|
|
|
|
assert 'config' in result
|
|
|
|
assert u'Error parsing the configuration options: No JSON object could be decoded' in result['config'][0]
|
|
|
|
|
|
|
|
source_dict['config'] = json.dumps({'custom_option': 'not_a_list'})
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api(self.action,
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-11-30 18:02:06 +01:00
|
|
|
|
|
|
|
assert 'config' in result
|
|
|
|
assert u'Error parsing the configuration options: custom_option must be a list' in result['config'][0]
|
|
|
|
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
class TestHarvestSourceActionCreate(HarvestSourceActionBase):
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.action = 'harvest_source_create'
|
|
|
|
|
|
|
|
def test_create(self):
|
|
|
|
|
|
|
|
source_dict = self.default_source_dict
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api('harvest_source_create',
|
|
|
|
apikey=self.sysadmin['apikey'], **source_dict)
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
for key in source_dict.keys():
|
|
|
|
assert source_dict[key] == result[key]
|
|
|
|
|
|
|
|
# Check that source was actually created
|
|
|
|
source = harvest_model.HarvestSource.get(result['id'])
|
|
|
|
assert source.url == source_dict['url']
|
|
|
|
assert source.type == source_dict['source_type']
|
|
|
|
|
|
|
|
# Trying to create a source with the same URL fails
|
|
|
|
source_dict = copy.deepcopy(self.default_source_dict)
|
|
|
|
source_dict['name'] = 'test-source-action-new'
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api('harvest_source_create',
|
|
|
|
apikey=self.sysadmin['apikey'], status=409,
|
|
|
|
**source_dict)
|
2012-12-14 15:52:19 +01:00
|
|
|
|
2012-11-30 15:03:04 +01:00
|
|
|
assert 'url' in result
|
|
|
|
assert u'There already is a Harvest Source for this URL' in result['url'][0]
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
|
2012-11-30 15:03:04 +01:00
|
|
|
class TestHarvestSourceActionUpdate(HarvestSourceActionBase):
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
|
|
|
|
|
|
|
cls.action = 'harvest_source_update'
|
2012-12-14 15:52:19 +01:00
|
|
|
|
2012-11-30 15:03:04 +01:00
|
|
|
super(TestHarvestSourceActionUpdate, cls).setup_class()
|
|
|
|
|
|
|
|
# Create a source to udpate
|
|
|
|
source_dict = cls.default_source_dict
|
2015-10-21 18:26:57 +02:00
|
|
|
result = call_action_api('harvest_source_create',
|
|
|
|
apikey=cls.sysadmin['apikey'], **source_dict)
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
cls.default_source_dict['id'] = result['id']
|
|
|
|
|
|
|
|
def test_update(self):
|
|
|
|
|
|
|
|
source_dict = self.default_source_dict
|
|
|
|
source_dict.update({
|
2015-10-21 18:26:57 +02:00
|
|
|
"url": "http://test.action.updated.com",
|
|
|
|
"name": "test-source-action-updated",
|
|
|
|
"title": "Test source action updated",
|
|
|
|
"notes": "Test source action desc updated",
|
|
|
|
"source_type": "test",
|
|
|
|
"frequency": "MONTHLY",
|
|
|
|
"config": json.dumps({"custom_option": ["c", "d"]})
|
|
|
|
})
|
|
|
|
|
|
|
|
result = call_action_api('harvest_source_update',
|
|
|
|
apikey=self.sysadmin['apikey'], **source_dict)
|
2012-11-30 15:03:04 +01:00
|
|
|
|
|
|
|
for key in source_dict.keys():
|
|
|
|
assert source_dict[key] == result[key]
|
|
|
|
|
|
|
|
# Check that source was actually updated
|
|
|
|
source = harvest_model.HarvestSource.get(result['id'])
|
|
|
|
assert source.url == source_dict['url']
|
|
|
|
assert source.type == source_dict['source_type']
|
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
|
2013-09-17 17:49:19 +02:00
|
|
|
class TestHarvestObject(unittest.TestCase):
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
2015-10-21 18:26:57 +02:00
|
|
|
reset_db()
|
2013-09-17 17:49:19 +02:00
|
|
|
harvest_model.setup()
|
|
|
|
|
|
|
|
def test_create(self):
|
2015-10-21 18:26:57 +02:00
|
|
|
job = factories.HarvestJobObj()
|
2013-09-17 17:49:19 +02:00
|
|
|
|
|
|
|
context = {
|
2015-10-21 18:26:57 +02:00
|
|
|
'model': model,
|
|
|
|
'session': model.Session,
|
2013-09-17 17:49:19 +02:00
|
|
|
'ignore_auth': True,
|
|
|
|
}
|
|
|
|
data_dict = {
|
2015-10-21 18:26:57 +02:00
|
|
|
'guid': 'guid',
|
|
|
|
'content': 'content',
|
|
|
|
'job_id': job.id,
|
|
|
|
'extras': {'a key': 'a value'},
|
2013-09-17 17:49:19 +02:00
|
|
|
}
|
|
|
|
harvest_object = toolkit.get_action('harvest_object_create')(
|
|
|
|
context, data_dict)
|
|
|
|
|
|
|
|
# fetch the object from database to check it was created
|
|
|
|
created_object = harvest_model.HarvestObject.get(harvest_object['id'])
|
|
|
|
assert created_object.guid == harvest_object['guid'] == data_dict['guid']
|
|
|
|
|
|
|
|
def test_create_bad_parameters(self):
|
2015-10-21 18:26:57 +02:00
|
|
|
source_a = factories.HarvestSourceObj()
|
|
|
|
job = factories.HarvestJobObj()
|
2013-09-17 17:49:19 +02:00
|
|
|
|
|
|
|
context = {
|
2015-10-21 18:26:57 +02:00
|
|
|
'model': model,
|
|
|
|
'session': model.Session,
|
2013-09-17 17:49:19 +02:00
|
|
|
'ignore_auth': True,
|
|
|
|
}
|
|
|
|
data_dict = {
|
2015-10-21 18:26:57 +02:00
|
|
|
'job_id': job.id,
|
|
|
|
'source_id': source_a.id,
|
|
|
|
'extras': 1
|
2013-09-17 17:49:19 +02:00
|
|
|
}
|
|
|
|
harvest_object_create = toolkit.get_action('harvest_object_create')
|
2015-10-21 18:26:57 +02:00
|
|
|
self.assertRaises(toolkit.ValidationError, harvest_object_create,
|
|
|
|
context, data_dict)
|
2013-09-17 17:49:19 +02:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
data_dict['extras'] = {'test': 1}
|
2013-09-17 17:49:19 +02:00
|
|
|
|
2015-10-21 18:26:57 +02:00
|
|
|
self.assertRaises(toolkit.ValidationError, harvest_object_create,
|
|
|
|
context, data_dict)
|