# -*- coding: utf-8 -*- """ This plugin enables a kind of permalink which can be used to refer to a piece of content which is resistant to the file being moved or renamed. """ import logging import itertools import os import os.path from pelican import signals from pelican.generators import Generator from pelican.utils import mkdir_p from pelican.utils import clean_output_dir logger = logging.getLogger(__name__) def article_url(content): ''' Get the URL for an item of content ''' return '{content.settings[SITEURL]}/{content.url}'.format( content=content).encode('utf-8') REDIRECT_STRING = ''' Page Redirection to {title} If you are not redirected automatically, follow the link to {title} ''' class PermalinkGenerator(Generator): ''' Generate a redirect page for every item of content with a permalink_id metadata ''' def generate_context(self): ''' Setup context ''' self.permalink_output_path = os.path.join( self.output_path, self.settings['PERMALINK_PATH']) self.permalink_id_metadata_key = self.settings['PERMALINK_ID_METADATA_KEY'] def generate_output(self, writer=None): ''' Generate redirect files ''' logger.info( 'Generating permalink files in %r', self.permalink_output_path) clean_output_dir(self.permalink_output_path, []) mkdir_p(self.permalink_output_path) for content in itertools.chain( self.context['articles'], self.context['pages']): for permalink_id in content.get_permalink_ids_iter(): permalink_path = os.path.join( self.permalink_output_path, permalink_id) + '.html' redirect_string = REDIRECT_STRING.format( url=article_url(content), title=content.title) open(permalink_path, 'w').write(redirect_string) def get_permalink_ids_iter(self): ''' Method to get permalink ids from content. To be bound to the class last thing ''' permalink_id_key = self.settings['PERMALINK_ID_METADATA_KEY'] permalink_ids_raw = self.metadata.get(permalink_id_key, '') for permalink_id in permalink_ids_raw.split(','): if permalink_id: yield permalink_id.strip() def get_permalink_ids(self): ''' Method to get permalink ids from content. To be bound to the class last thing ''' return list(self.get_permalink_ids_iter()) def get_permalink_path(self): """Get just path component of permalink.""" try: first_permalink_id = self.get_permalink_ids_iter().next() except StopIteration: return None return '/{settings[PERMALINK_PATH]}/{first_permalink}'.format( settings=self.settings, first_permalink=first_permalink_id) def get_permalink_url(self): ''' Get a permalink URL ''' return "/".join((self.settings['SITEURL'], self.get_permalink_path())) PERMALINK_METHODS = ( get_permalink_ids_iter, get_permalink_ids, get_permalink_url, get_permalink_path, ) def add_permalink_methods(content_inst): ''' Add permalink methods to object ''' for permalink_method in PERMALINK_METHODS: setattr( content_inst, permalink_method.__name__, permalink_method.__get__(content_inst, content_inst.__class__)) def add_permalink_option_defaults(pelicon_inst): ''' Add perlican defaults ''' pelicon_inst.settings.setdefault('PERMALINK_PATH', 'permalinks') pelicon_inst.settings.setdefault('PERMALINK_ID_METADATA_KEY', 'permalink_id') def get_generators(_pelican_object): return PermalinkGenerator def register(): signals.get_generators.connect(get_generators) signals.content_object_init.connect(add_permalink_methods) signals.initialized.connect(add_permalink_option_defaults)