2012-05-29 16:21:34 +02:00
|
|
|
import logging
|
2013-01-17 16:43:45 +01:00
|
|
|
from sqlalchemy import or_, func
|
2012-03-02 17:49:39 +01:00
|
|
|
from ckan.model import User
|
2012-10-29 18:15:02 +01:00
|
|
|
import datetime
|
2012-03-02 17:49:39 +01:00
|
|
|
|
2012-12-11 13:49:05 +01:00
|
|
|
from ckan import logic
|
2012-02-29 16:20:35 +01:00
|
|
|
from ckan.plugins import PluginImplementations
|
|
|
|
from ckanext.harvest.interfaces import IHarvester
|
|
|
|
|
2013-01-17 16:43:45 +01:00
|
|
|
import ckan.plugins as p
|
2013-02-28 13:17:15 +01:00
|
|
|
from ckan.logic import NotFound, check_access, side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-12-12 12:45:13 +01:00
|
|
|
from ckanext.harvest import model as harvest_model
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
from ckanext.harvest.model import (HarvestSource, HarvestJob, HarvestObject)
|
|
|
|
from ckanext.harvest.logic.dictization import (harvest_source_dictize,
|
|
|
|
harvest_job_dictize,
|
|
|
|
harvest_object_dictize)
|
2012-12-11 13:49:05 +01:00
|
|
|
from ckanext.harvest.logic.schema import harvest_source_db_to_form_schema
|
2012-05-29 16:21:34 +02:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_source_show(context,data_dict):
|
2012-12-11 13:49:05 +01:00
|
|
|
'''
|
|
|
|
Returns the metadata of a harvest source
|
|
|
|
|
|
|
|
This method just proxies the request to package_show. All auth checks and
|
|
|
|
validation will be done there.
|
2012-03-01 13:02:16 +01:00
|
|
|
|
2012-12-11 13:49:05 +01:00
|
|
|
:param id: the id or name of the harvest source
|
|
|
|
:type id: string
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-12-11 13:49:05 +01:00
|
|
|
:returns: harvest source metadata
|
|
|
|
:rtype: dictionary
|
|
|
|
'''
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-12-11 13:49:05 +01:00
|
|
|
source_dict = logic.get_action('package_show')(context, data_dict)
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-12-13 19:00:07 +01:00
|
|
|
# For compatibility with old code, add the active field
|
|
|
|
# based on the package state
|
|
|
|
source_dict['active'] = (source_dict['state'] == 'active')
|
|
|
|
|
2012-12-11 13:49:05 +01:00
|
|
|
return source_dict
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2013-01-17 16:43:45 +01:00
|
|
|
def harvest_source_show_status(context, data_dict):
|
2012-12-12 12:45:13 +01:00
|
|
|
'''
|
|
|
|
Returns a status report for a harvest source
|
|
|
|
|
|
|
|
Given a particular source, returns a dictionary containing information
|
|
|
|
about the source jobs, datasets created, errors, etc.
|
|
|
|
Note that this information is already included on the output of
|
|
|
|
harvest_source_show, under the 'status' field.
|
|
|
|
|
|
|
|
:param id: the id or name of the harvest source
|
|
|
|
:type id: string
|
|
|
|
|
|
|
|
:rtype: dictionary
|
|
|
|
'''
|
|
|
|
model = context.get('model')
|
|
|
|
|
|
|
|
source = harvest_model.HarvestSource.get(data_dict['id'])
|
|
|
|
if not source:
|
2013-02-28 13:17:15 +01:00
|
|
|
raise p.toolkit.ObjectNotFound('Harvest source {0} does not exist'.format(data_dict['id']))
|
2012-12-12 12:45:13 +01:00
|
|
|
|
|
|
|
out = {
|
|
|
|
'job_count': 0,
|
2013-01-22 14:13:24 +01:00
|
|
|
'last_job': None,
|
2013-01-17 16:43:45 +01:00
|
|
|
'total_datasets': 0,
|
2012-12-12 12:45:13 +01:00
|
|
|
}
|
|
|
|
|
2013-01-17 16:43:45 +01:00
|
|
|
jobs = harvest_model.HarvestJob.filter(source=source).all()
|
|
|
|
|
2012-12-12 12:45:13 +01:00
|
|
|
job_count = len(jobs)
|
|
|
|
if job_count == 0:
|
|
|
|
return out
|
2013-01-17 16:43:45 +01:00
|
|
|
|
|
|
|
out['job_count'] = job_count
|
2012-12-12 12:45:13 +01:00
|
|
|
|
2013-02-25 16:32:39 +01:00
|
|
|
# Get the most recent job
|
|
|
|
last_job = harvest_model.HarvestJob.filter(source=source) \
|
2012-12-12 12:45:13 +01:00
|
|
|
.order_by(harvest_model.HarvestJob.created.desc()).first()
|
|
|
|
|
2013-01-17 16:43:45 +01:00
|
|
|
if not last_job:
|
|
|
|
return out
|
|
|
|
|
2013-01-22 14:13:24 +01:00
|
|
|
out['last_job'] = harvest_job_dictize(last_job, context)
|
2013-01-17 16:43:45 +01:00
|
|
|
|
|
|
|
# Overall statistics
|
|
|
|
packages = model.Session.query(model.Package) \
|
|
|
|
.join(harvest_model.HarvestObject) \
|
|
|
|
.filter(harvest_model.HarvestObject.harvest_source_id==source.id) \
|
|
|
|
.filter(harvest_model.HarvestObject.current==True) \
|
|
|
|
.filter(model.Package.state==u'active')
|
|
|
|
|
|
|
|
out['total_datasets'] = packages.count()
|
2012-12-12 12:45:13 +01:00
|
|
|
|
|
|
|
return out
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_source_list(context, data_dict):
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
check_access('harvest_source_list',context,data_dict)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
model = context['model']
|
2012-03-01 13:02:16 +01:00
|
|
|
session = context['session']
|
2012-03-02 17:49:39 +01:00
|
|
|
user = context.get('user','')
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-03-02 17:49:39 +01:00
|
|
|
sources = _get_sources_for_user(context, data_dict)
|
2012-02-29 11:59:02 +01:00
|
|
|
|
|
|
|
context.update({'detailed':False})
|
|
|
|
return [harvest_source_dictize(source, context) for source in sources]
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-06-01 18:03:40 +02:00
|
|
|
def harvest_source_for_a_dataset(context, data_dict):
|
|
|
|
'''For a given dataset, return the harvest source that
|
|
|
|
created or last updated it, otherwise NotFound.'''
|
|
|
|
|
|
|
|
model = context['model']
|
|
|
|
session = context['session']
|
|
|
|
|
|
|
|
dataset_id = data_dict.get('id')
|
|
|
|
|
|
|
|
query = session.query(HarvestSource)\
|
|
|
|
.join(HarvestObject)\
|
|
|
|
.filter_by(package_id=dataset_id)\
|
|
|
|
.order_by(HarvestObject.gathered.desc())
|
|
|
|
source = query.first() # newest
|
|
|
|
|
|
|
|
if not source:
|
|
|
|
raise NotFound
|
|
|
|
|
|
|
|
return harvest_source_dictize(source,context)
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_job_show(context,data_dict):
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
check_access('harvest_job_show',context,data_dict)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
id = data_dict.get('id')
|
|
|
|
attr = data_dict.get('attr',None)
|
|
|
|
|
|
|
|
job = HarvestJob.get(id,attr=attr)
|
|
|
|
if not job:
|
|
|
|
raise NotFound
|
|
|
|
|
|
|
|
return harvest_job_dictize(job,context)
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2013-01-23 19:04:19 +01:00
|
|
|
def harvest_job_report(context, data_dict):
|
|
|
|
|
|
|
|
check_access('harvest_job_show', context, data_dict)
|
|
|
|
|
|
|
|
model = context['model']
|
|
|
|
id = data_dict.get('id')
|
|
|
|
|
|
|
|
job = HarvestJob.get(id)
|
|
|
|
if not job:
|
|
|
|
raise NotFound
|
|
|
|
|
2013-02-25 18:17:08 +01:00
|
|
|
report = {
|
|
|
|
'gather_errors': [],
|
|
|
|
'object_errors': []
|
|
|
|
}
|
|
|
|
|
|
|
|
# Gather errors
|
|
|
|
q = model.Session.query(harvest_model.HarvestGatherError) \
|
|
|
|
.join(harvest_model.HarvestJob) \
|
|
|
|
.filter(harvest_model.HarvestGatherError.harvest_job_id==job.id) \
|
|
|
|
.order_by(harvest_model.HarvestGatherError.created.desc())
|
|
|
|
|
|
|
|
for error in q.all():
|
|
|
|
report['gather_errors'].append({
|
|
|
|
'message': error.message
|
|
|
|
})
|
|
|
|
|
|
|
|
# Object errors
|
|
|
|
|
2013-01-24 19:35:43 +01:00
|
|
|
# Check if the harvester for this job's source has a method for returning
|
|
|
|
# the URL to the original document
|
|
|
|
original_url_builder = None
|
|
|
|
for harvester in PluginImplementations(IHarvester):
|
|
|
|
if harvester.info()['name'] == job.source.type:
|
|
|
|
if hasattr(harvester, 'get_original_url'):
|
|
|
|
original_url_builder = harvester.get_original_url
|
|
|
|
|
|
|
|
q = model.Session.query(harvest_model.HarvestObjectError, harvest_model.HarvestObject.guid) \
|
2013-01-23 19:04:19 +01:00
|
|
|
.join(harvest_model.HarvestObject) \
|
|
|
|
.filter(harvest_model.HarvestObject.harvest_job_id==job.id) \
|
|
|
|
.order_by(harvest_model.HarvestObjectError.harvest_object_id)
|
|
|
|
|
2013-01-24 19:35:43 +01:00
|
|
|
for error, guid in q.all():
|
2013-02-25 18:17:08 +01:00
|
|
|
if not error.harvest_object_id in report['object_errors']:
|
|
|
|
report['object_errors'][error.harvest_object_id] = {
|
2013-01-24 19:35:43 +01:00
|
|
|
'guid': guid,
|
|
|
|
'errors': []
|
|
|
|
}
|
|
|
|
if original_url_builder:
|
|
|
|
url = original_url_builder(error.harvest_object_id)
|
|
|
|
if url:
|
2013-02-25 18:17:08 +01:00
|
|
|
report['object_errors'][error.harvest_object_id]['original_url'] = url
|
2013-01-24 19:35:43 +01:00
|
|
|
|
2013-02-25 18:17:08 +01:00
|
|
|
report['object_errors'][error.harvest_object_id]['errors'].append({
|
2013-01-23 19:04:19 +01:00
|
|
|
'message': error.message,
|
|
|
|
'line': error.line,
|
|
|
|
'type': error.stage
|
|
|
|
})
|
|
|
|
|
2013-01-24 19:35:43 +01:00
|
|
|
return report
|
2013-01-22 14:13:24 +01:00
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_job_list(context,data_dict):
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
check_access('harvest_job_list',context,data_dict)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
model = context['model']
|
2012-03-01 13:02:16 +01:00
|
|
|
session = context['session']
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2012-02-29 16:20:35 +01:00
|
|
|
source_id = data_dict.get('source_id',False)
|
2012-02-29 11:59:02 +01:00
|
|
|
status = data_dict.get('status',False)
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
query = session.query(HarvestJob)
|
2012-02-29 16:20:35 +01:00
|
|
|
|
|
|
|
if source_id:
|
|
|
|
query = query.filter(HarvestJob.source_id==source_id)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
if status:
|
2012-02-29 16:20:35 +01:00
|
|
|
query = query.filter(HarvestJob.status==status)
|
|
|
|
|
2013-02-04 19:20:58 +01:00
|
|
|
query = query.order_by(HarvestJob.created.desc())
|
|
|
|
|
2012-02-29 16:20:35 +01:00
|
|
|
jobs = query.all()
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2013-02-04 19:28:45 +01:00
|
|
|
context['return_error_summary'] = False
|
|
|
|
return [harvest_job_dictize(job, context) for job in jobs]
|
2012-02-29 11:59:02 +01:00
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_object_show(context,data_dict):
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
check_access('harvest_object_show',context,data_dict)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
id = data_dict.get('id')
|
|
|
|
attr = data_dict.get('attr',None)
|
|
|
|
obj = HarvestObject.get(id,attr=attr)
|
|
|
|
if not obj:
|
|
|
|
raise NotFound
|
|
|
|
|
|
|
|
return harvest_object_dictize(obj,context)
|
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-02-29 11:59:02 +01:00
|
|
|
def harvest_object_list(context,data_dict):
|
|
|
|
|
2012-03-01 13:02:16 +01:00
|
|
|
check_access('harvest_object_list',context,data_dict)
|
|
|
|
|
2012-02-29 11:59:02 +01:00
|
|
|
model = context['model']
|
2012-03-01 13:02:16 +01:00
|
|
|
session = context['session']
|
2012-02-29 11:59:02 +01:00
|
|
|
|
|
|
|
only_current = data_dict.get('only_current',True)
|
2012-03-02 17:49:39 +01:00
|
|
|
source_id = data_dict.get('source_id',False)
|
|
|
|
|
|
|
|
query = session.query(HarvestObject)
|
|
|
|
|
|
|
|
if source_id:
|
|
|
|
query = query.filter(HarvestObject.source_id==source_id)
|
2012-02-29 11:59:02 +01:00
|
|
|
|
|
|
|
if only_current:
|
2012-03-02 17:49:39 +01:00
|
|
|
query = query.filter(HarvestObject.current==True)
|
|
|
|
|
|
|
|
objects = query.all()
|
2012-02-29 11:59:02 +01:00
|
|
|
|
|
|
|
return [getattr(obj,'id') for obj in objects]
|
2012-02-29 16:20:35 +01:00
|
|
|
|
2013-02-28 13:17:15 +01:00
|
|
|
@side_effect_free
|
2012-03-01 13:02:16 +01:00
|
|
|
def harvesters_info_show(context,data_dict):
|
|
|
|
|
|
|
|
check_access('harvesters_info_show',context,data_dict)
|
|
|
|
|
2012-02-29 16:20:35 +01:00
|
|
|
available_harvesters = []
|
|
|
|
for harvester in PluginImplementations(IHarvester):
|
|
|
|
info = harvester.info()
|
|
|
|
if not info or 'name' not in info:
|
|
|
|
log.error('Harvester %r does not provide the harvester name in the info response' % str(harvester))
|
|
|
|
continue
|
|
|
|
info['show_config'] = (info.get('form_config_interface','') == 'Text')
|
|
|
|
available_harvesters.append(info)
|
|
|
|
|
|
|
|
return available_harvesters
|
2012-03-02 17:49:39 +01:00
|
|
|
|
|
|
|
def _get_sources_for_user(context,data_dict):
|
|
|
|
|
|
|
|
model = context['model']
|
|
|
|
session = context['session']
|
|
|
|
user = context.get('user','')
|
|
|
|
|
|
|
|
only_active = data_dict.get('only_active',False)
|
2012-10-29 18:15:02 +01:00
|
|
|
only_to_run = data_dict.get('only_to_run',False)
|
2012-03-02 17:49:39 +01:00
|
|
|
|
|
|
|
query = session.query(HarvestSource) \
|
|
|
|
.order_by(HarvestSource.created.desc())
|
|
|
|
|
|
|
|
if only_active:
|
|
|
|
query = query.filter(HarvestSource.active==True) \
|
|
|
|
|
2012-10-29 18:15:02 +01:00
|
|
|
if only_to_run:
|
2012-11-05 14:17:32 +01:00
|
|
|
query = query.filter(HarvestSource.frequency!='MANUAL')
|
2012-10-29 18:15:02 +01:00
|
|
|
query = query.filter(or_(HarvestSource.next_run<=datetime.datetime.utcnow(),
|
|
|
|
HarvestSource.next_run==None)
|
|
|
|
)
|
|
|
|
|
2012-12-18 00:46:43 +01:00
|
|
|
user_obj = User.get(user)
|
2012-03-02 17:49:39 +01:00
|
|
|
# Sysadmins will get all sources
|
2012-12-24 23:34:37 +01:00
|
|
|
if user_obj and not user_obj.sysadmin:
|
2012-03-02 17:49:39 +01:00
|
|
|
# This only applies to a non sysadmin user when using the
|
|
|
|
# publisher auth profile. When using the default profile,
|
|
|
|
# normal users will never arrive at this point, but even if they
|
|
|
|
# do, they will get an empty list.
|
|
|
|
|
|
|
|
publisher_filters = []
|
2012-05-29 16:21:34 +02:00
|
|
|
publishers_for_the_user = user_obj.get_groups(u'publisher')
|
|
|
|
for publisher_id in [g.id for g in publishers_for_the_user]:
|
2012-03-02 17:49:39 +01:00
|
|
|
publisher_filters.append(HarvestSource.publisher_id==publisher_id)
|
|
|
|
|
|
|
|
if len(publisher_filters):
|
|
|
|
query = query.filter(or_(*publisher_filters))
|
|
|
|
else:
|
|
|
|
# This user does not belong to a publisher yet, no sources for him/her
|
|
|
|
return []
|
|
|
|
|
2012-05-29 16:21:34 +02:00
|
|
|
log.debug('User %s with publishers %r has Harvest Sources: %r',
|
|
|
|
user, publishers_for_the_user, [(hs.id, hs.url) for hs in query])
|
|
|
|
|
2012-03-02 17:49:39 +01:00
|
|
|
sources = query.all()
|
|
|
|
|
|
|
|
return sources
|
2012-06-01 18:03:40 +02:00
|
|
|
|