fimfarchive/fimfarchive/mappers.py
2017-10-25 22:08:24 +02:00

102 lines
2.2 KiB
Python

"""
Mappers for Fimfarchive.
"""
#
# Fimfarchive, preserves stories from Fimfiction.
# Copyright (C) 2015 Joakim Soderlund
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
import os
import arrow
from fimfarchive.exceptions import InvalidStoryError
__all__ = (
'Mapper',
'StaticMapper',
'StoryDateMapper',
'StoryPathMapper',
)
class Mapper:
"""
Callable which maps something to something else.
"""
def __call__(self, *args, **kwargs):
raise NotImplementedError()
class StaticMapper(Mapper):
"""
Returns the supplied value for any call.
"""
def __init__(self, value=None):
self.value = value
def __call__(self, *args, **kwargs):
return self.value
class StoryDateMapper(Mapper):
"""
Returns the latest timestamp in a story, or None.
"""
def __call__(self, story):
try:
meta = getattr(story, 'meta', None)
except InvalidStoryError:
return None
if not meta:
return None
dates = {
meta.get('date_modified'),
}
for chapter in meta.get('chapters') or tuple():
dates.add(chapter.get('date_modified'))
dates.discard(None)
if dates:
return max(arrow.get(date) for date in dates)
else:
return None
class StoryPathMapper(Mapper):
"""
Returns a key-based file path for a story.
"""
def __init__(self, directory):
self.directory = directory
def __call__(self, story):
directory = str(self.directory)
key = str(story.key)
return os.path.join(directory, key)