import logging import re import uuid from sqlalchemy import exists from sqlalchemy.sql import update, bindparam from pylons import config from ckan import plugins as p from ckan import model from ckan.model import Session, Package, PACKAGE_NAME_MAX_LENGTH from ckan.logic.schema import default_create_package_schema from ckan.lib.navl.validators import ignore_missing, ignore from ckan.lib.munge import munge_title_to_name, substitute_ascii_equivalents from ckanext.harvest.model import (HarvestObject, HarvestGatherError, HarvestObjectError, HarvestJob) from ckan.plugins.core import SingletonPlugin, implements from ckanext.harvest.interfaces import IHarvester if p.toolkit.check_ckan_version(min_version='2.3'): from ckan.lib.munge import munge_tag else: # Fallback munge_tag for older ckan versions which don't have a decent # munger def _munge_to_length(string, min_length, max_length): '''Pad/truncates a string''' if len(string) < min_length: string += '_' * (min_length - len(string)) if len(string) > max_length: string = string[:max_length] return string def munge_tag(tag): tag = substitute_ascii_equivalents(tag) tag = tag.lower().strip() tag = re.sub(r'[^a-zA-Z0-9\- ]', '', tag).replace(' ', '-') tag = _munge_to_length(tag, model.MIN_TAG_LENGTH, model.MAX_TAG_LENGTH) return tag log = logging.getLogger(__name__) class HarvesterBase(SingletonPlugin): ''' Generic base class for harvesters, providing a number of useful functions. A harvester doesn't have to derive from this - it could just have: implements(IHarvester) ''' implements(IHarvester) config = None _user_name = None @classmethod def _gen_new_name(cls, title, existing_name=None, append_type=None): ''' Returns a 'name' for the dataset (URL friendly), based on the title. If the ideal name is already used, it will append a number to it to ensure it is unique. If generating a new name because the title of the dataset has changed, specify the existing name, in case the name doesn't need to change after all. :param existing_name: the current name of the dataset - only specify this if the dataset exists :type existing_name: string :param append_type: the type of characters to add to make it unique - either 'number-sequence' or 'random-hex'. :type append_type: string ''' # If append_type was given, use it. Otherwise, use the configured default. # If nothing was given and no defaults were set, use 'number-sequence'. if append_type: append_type_param = append_type else: append_type_param = config.get('ckanext.harvest.default_dataset_name_append', 'number-sequence') ideal_name = munge_title_to_name(title) ideal_name = re.sub('-+', '-', ideal_name) # collapse multiple dashes return cls._ensure_name_is_unique(ideal_name, existing_name=existing_name, append_type=append_type_param) @staticmethod def _ensure_name_is_unique(ideal_name, existing_name=None, append_type='number-sequence'): ''' Returns a dataset name based on the ideal_name, only it will be guaranteed to be different than all the other datasets, by adding a number on the end if necessary. If generating a new name because the title of the dataset has changed, specify the existing name, in case the name doesn't need to change after all. The maximum dataset name length is taken account of. :param ideal_name: the desired name for the dataset, if its not already been taken (usually derived by munging the dataset title) :type ideal_name: string :param existing_name: the current name of the dataset - only specify this if the dataset exists :type existing_name: string :param append_type: the type of characters to add to make it unique - either 'number-sequence' or 'random-hex'. :type append_type: string ''' ideal_name = ideal_name[:PACKAGE_NAME_MAX_LENGTH] if existing_name == ideal_name: return ideal_name if append_type == 'number-sequence': MAX_NUMBER_APPENDED = 999 APPEND_MAX_CHARS = len(str(MAX_NUMBER_APPENDED)) elif append_type == 'random-hex': APPEND_MAX_CHARS = 5 # 16^5 = 1 million combinations else: raise NotImplementedError('append_type cannot be %s' % append_type) # Find out which package names have been taken. Restrict it to names # derived from the ideal name plus and numbers added like_q = u'%s%%' % \ ideal_name[:PACKAGE_NAME_MAX_LENGTH-APPEND_MAX_CHARS] name_results = Session.query(Package.name)\ .filter(Package.name.ilike(like_q))\ .all() taken = set([name_result[0] for name_result in name_results]) if existing_name and existing_name in taken: taken.remove(existing_name) if ideal_name not in taken: # great, the ideal name is available return ideal_name elif existing_name and existing_name.startswith(ideal_name): # the ideal name is not available, but its an existing dataset with # a name based on the ideal one, so there's no point changing it to # a different number return existing_name elif append_type == 'number-sequence': # find the next available number counter = 1 while counter <= MAX_NUMBER_APPENDED: candidate_name = \ ideal_name[:PACKAGE_NAME_MAX_LENGTH-len(str(counter))] + \ str(counter) if candidate_name not in taken: return candidate_name counter = counter + 1 return None elif append_type == 'random-hex': return ideal_name[:PACKAGE_NAME_MAX_LENGTH-APPEND_MAX_CHARS] + \ str(uuid.uuid4())[:APPEND_MAX_CHARS] _save_gather_error = HarvestGatherError.create _save_object_error = HarvestObjectError.create def _get_user_name(self): ''' Returns the name of the user that will perform the harvesting actions (deleting, updating and creating datasets) By default this will be the old 'harvest' user to maintain compatibility. If not present, the internal site admin user will be used. This is the recommended setting, but if necessary it can be overridden with the `ckanext.harvest.user_name` config option: ckanext.harvest.user_name = harvest ''' if self._user_name: return self._user_name config_user_name = config.get('ckanext.harvest.user_name') if config_user_name: self._user_name = config_user_name return self._user_name context = {'model': model, 'ignore_auth': True, } # Check if 'harvest' user exists and if is a sysadmin try: user_harvest = p.toolkit.get_action('user_show')( context, {'id': 'harvest'}) if user_harvest['sysadmin']: self._user_name = 'harvest' return self._user_name except p.toolkit.ObjectNotFound: pass context['defer_commit'] = True # See ckan/ckan#1714 self._site_user = p.toolkit.get_action('get_site_user')(context, {}) self._user_name = self._site_user['name'] return self._user_name def _create_harvest_objects(self, remote_ids, harvest_job): ''' Given a list of remote ids and a Harvest Job, create as many Harvest Objects and return a list of their ids to be passed to the fetch stage. TODO: Not sure it is worth keeping this function ''' try: object_ids = [] if len(remote_ids): for remote_id in remote_ids: # Create a new HarvestObject for this identifier obj = HarvestObject(guid = remote_id, job = harvest_job) obj.save() object_ids.append(obj.id) return object_ids else: self._save_gather_error('No remote datasets could be identified', harvest_job) except Exception, e: self._save_gather_error('%r' % e.message, harvest_job) def _create_or_update_package(self, package_dict, harvest_object, package_dict_form='rest'): ''' Creates a new package or updates an existing one according to the package dictionary provided. The package dictionary can be in one of two forms: 1. 'rest' - as seen on the RESTful API: http://datahub.io/api/rest/dataset/1996_population_census_data_canada This is the legacy form. It is the default to provide backward compatibility. * 'extras' is a dict e.g. {'theme': 'health', 'sub-theme': 'cancer'} * 'tags' is a list of strings e.g. ['large-river', 'flood'] 2. 'package_show' form, as provided by the Action API (CKAN v2.0+): http://datahub.io/api/action/package_show?id=1996_population_census_data_canada * 'extras' is a list of dicts e.g. [{'key': 'theme', 'value': 'health'}, {'key': 'sub-theme', 'value': 'cancer'}] * 'tags' is a list of dicts e.g. [{'name': 'large-river'}, {'name': 'flood'}] Note that the package_dict must contain an id, which will be used to check if the package needs to be created or updated (use the remote dataset id). If the remote server provides the modification date of the remote package, add it to package_dict['metadata_modified']. :returns: The same as what import_stage should return. i.e. True if the create or update occurred ok, 'unchanged' if it didn't need updating or False if there were errors. TODO: Not sure it is worth keeping this function. If useful it should use the output of package_show logic function (maybe keeping support for rest api based dicts ''' assert package_dict_form in ('rest', 'package_show') try: # Change default schema schema = default_create_package_schema() schema['id'] = [ignore_missing, unicode] schema['__junk'] = [ignore] # Check API version if self.config: try: api_version = int(self.config.get('api_version', 2)) except ValueError: raise ValueError('api_version must be an integer') else: api_version = 2 user_name = self._get_user_name() context = { 'model': model, 'session': Session, 'user': user_name, 'api_version': api_version, 'schema': schema, 'ignore_auth': True, } if self.config and self.config.get('clean_tags', False): tags = package_dict.get('tags', []) package_dict['tags'] = self._clean_tags(tags) # Check if package exists try: # _find_existing_package can be overridden if necessary existing_package_dict = self._find_existing_package(package_dict) # In case name has been modified when first importing. See issue #101. package_dict['name'] = existing_package_dict['name'] # Check modified date if not 'metadata_modified' in package_dict or \ package_dict['metadata_modified'] > existing_package_dict.get('metadata_modified'): log.info('Package with GUID %s exists and needs to be updated' % harvest_object.guid) # Update package context.update({'id':package_dict['id']}) package_dict.setdefault('name', existing_package_dict['name']) new_package = p.toolkit.get_action( 'package_update' if package_dict_form == 'package_show' else 'package_update_rest')(context, package_dict) else: log.info('No changes to package with GUID %s, skipping...' % harvest_object.guid) # NB harvest_object.current/package_id are not set return 'unchanged' # Flag the other objects linking to this package as not current anymore from ckanext.harvest.model import harvest_object_table conn = Session.connection() u = update(harvest_object_table) \ .where(harvest_object_table.c.package_id==bindparam('b_package_id')) \ .values(current=False) conn.execute(u, b_package_id=new_package['id']) # Flag this as the current harvest object harvest_object.package_id = new_package['id'] harvest_object.current = True harvest_object.save() except p.toolkit.ObjectNotFound: # Package needs to be created # Get rid of auth audit on the context otherwise we'll get an # exception context.pop('__auth_audit', None) # Set name for new package to prevent name conflict, see issue #117 if package_dict.get('name', None): package_dict['name'] = self._gen_new_name(package_dict['name']) else: package_dict['name'] = self._gen_new_name(package_dict['title']) log.info('Package with GUID %s does not exist, let\'s create it' % harvest_object.guid) harvest_object.current = True harvest_object.package_id = package_dict['id'] # Defer constraints and flush so the dataset can be indexed with # the harvest object id (on the after_show hook from the harvester # plugin) harvest_object.add() model.Session.execute('SET CONSTRAINTS harvest_object_package_id_fkey DEFERRED') model.Session.flush() new_package = p.toolkit.get_action( 'package_create' if package_dict_form == 'package_show' else 'package_create_rest')(context, package_dict) Session.commit() return True except p.toolkit.ValidationError, e: log.exception(e) self._save_object_error('Invalid package with GUID %s: %r'%(harvest_object.guid,e.error_dict),harvest_object,'Import') except Exception, e: log.exception(e) self._save_object_error('%r'%e,harvest_object,'Import') return None def _find_existing_package(self, package_dict): data_dict = {'id': package_dict['id']} package_show_context = {'model': model, 'session': Session, 'ignore_auth': True} return p.toolkit.get_action('package_show')( package_show_context, data_dict) def _clean_tags(self, tags): try: def _update_tag(tag_dict, key, newvalue): # update the dict and return it tag_dict[key] = newvalue return tag_dict # assume it's in the package_show form tags = [_update_tag(t, 'name', munge_tag(t['name'])) for t in tags if munge_tag(t['name']) != ''] except TypeError: # a TypeError is raised if `t` above is a string # REST format: 'tags' is a list of strings tags = [munge_tag(t) for t in tags if munge_tag(t) != ''] tags = list(set(tags)) return tags return tags @classmethod def last_error_free_job(cls, harvest_job): # TODO weed out cancelled jobs somehow. # look for jobs with no gather errors jobs = \ model.Session.query(HarvestJob) \ .filter(HarvestJob.source == harvest_job.source) \ .filter(HarvestJob.gather_started != None) \ .filter(HarvestJob.status == 'Finished') \ .filter(HarvestJob.id != harvest_job.id) \ .filter( ~exists().where( HarvestGatherError.harvest_job_id == HarvestJob.id)) \ .order_by(HarvestJob.gather_started.desc()) # now check them until we find one with no fetch/import errors # (looping rather than doing sql, in case there are lots of objects # and lots of jobs) for job in jobs: for obj in job.objects: if obj.current is False and \ obj.report_status != 'not modified': # unsuccessful, so go onto the next job break else: return job