mirror of
https://github.com/JockeTF/fimfarchive.git
synced 2024-11-22 05:17:59 +01:00
Add alpha to beta converter
This commit is contained in:
parent
b590c60c3b
commit
a4c1fe5a2d
5 changed files with 783 additions and 0 deletions
|
@ -23,8 +23,10 @@ Converter module.
|
||||||
|
|
||||||
|
|
||||||
from .base import Converter
|
from .base import Converter
|
||||||
|
from .alpha_beta import AlphaBetaConverter
|
||||||
|
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
'Converter',
|
'Converter',
|
||||||
|
'AlphaBetaConverter',
|
||||||
)
|
)
|
||||||
|
|
468
fimfarchive/converters/alpha_beta.py
Normal file
468
fimfarchive/converters/alpha_beta.py
Normal file
|
@ -0,0 +1,468 @@
|
||||||
|
"""
|
||||||
|
Alpha to beta converter for story meta.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# 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/>.
|
||||||
|
#
|
||||||
|
|
||||||
|
|
||||||
|
from copy import deepcopy
|
||||||
|
from typing import Any, Dict, Iterable, Iterator, List, Optional, Tuple
|
||||||
|
from urllib.parse import quote_plus as urlquote
|
||||||
|
|
||||||
|
import arrow
|
||||||
|
import bbcode
|
||||||
|
from jmespath import compile as jmes
|
||||||
|
from jmespath.parser import ParsedResult
|
||||||
|
|
||||||
|
from fimfarchive.flavors import MetaFormat
|
||||||
|
from fimfarchive.stories import Story
|
||||||
|
|
||||||
|
from .base import Converter
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = (
|
||||||
|
'AlphaBetaConverter',
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
HOST = 'https://www.fimfiction.net'
|
||||||
|
EPOCH = arrow.get(0).isoformat()
|
||||||
|
|
||||||
|
|
||||||
|
TAGS = {
|
||||||
|
'2nd Person': {
|
||||||
|
'id': 225,
|
||||||
|
'name': 'Second Person',
|
||||||
|
'old_id': 'g:second_person',
|
||||||
|
'type': 'content',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/second-person',
|
||||||
|
},
|
||||||
|
'Adventure': {
|
||||||
|
'id': 226,
|
||||||
|
'name': 'Adventure',
|
||||||
|
'old_id': 'g:adventure',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/adventure',
|
||||||
|
},
|
||||||
|
'Alternate Universe': {
|
||||||
|
'id': 240,
|
||||||
|
'name': 'Alternate Universe',
|
||||||
|
'old_id': 'g:alternate_universe',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/alternate-universe',
|
||||||
|
},
|
||||||
|
'Anthro': {
|
||||||
|
'id': 227,
|
||||||
|
'name': 'Anthro',
|
||||||
|
'old_id': 'g:anthro',
|
||||||
|
'type': 'content',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/anthro',
|
||||||
|
},
|
||||||
|
'Comedy': {
|
||||||
|
'id': 228,
|
||||||
|
'name': 'Comedy',
|
||||||
|
'old_id': 'g:comedy',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/comedy',
|
||||||
|
},
|
||||||
|
'Crossover': {
|
||||||
|
'id': 229,
|
||||||
|
'name': 'Crossover',
|
||||||
|
'old_id': 'g:crossover',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/crossover',
|
||||||
|
},
|
||||||
|
'Dark': {
|
||||||
|
'id': 122,
|
||||||
|
'name': 'Dark',
|
||||||
|
'old_id': 'g:dark',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/dark',
|
||||||
|
},
|
||||||
|
'Drama': {
|
||||||
|
'id': 230,
|
||||||
|
'name': 'Drama',
|
||||||
|
'old_id': 'g:drama',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/drama',
|
||||||
|
},
|
||||||
|
'Equestria Girls': {
|
||||||
|
'id': 123,
|
||||||
|
'name': 'Equestria Girls',
|
||||||
|
'old_id': 'g:equestria_girls',
|
||||||
|
'type': 'series',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/equestria-girls',
|
||||||
|
},
|
||||||
|
'Horror': {
|
||||||
|
'id': 231,
|
||||||
|
'name': 'Horror',
|
||||||
|
'old_id': 'g:horror',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/horror',
|
||||||
|
},
|
||||||
|
'Human': {
|
||||||
|
'id': 232,
|
||||||
|
'name': 'Human',
|
||||||
|
'old_id': 'g:human',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/human',
|
||||||
|
},
|
||||||
|
'Mystery': {
|
||||||
|
'id': 233,
|
||||||
|
'name': 'Mystery',
|
||||||
|
'old_id': 'g:mystery',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/mystery',
|
||||||
|
},
|
||||||
|
'Random': {
|
||||||
|
'id': 234,
|
||||||
|
'name': 'Random',
|
||||||
|
'old_id': 'g:random',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/random',
|
||||||
|
},
|
||||||
|
'Romance': {
|
||||||
|
'id': 120,
|
||||||
|
'name': 'Romance',
|
||||||
|
'old_id': 'g:romance',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/romance',
|
||||||
|
},
|
||||||
|
'Sad': {
|
||||||
|
'id': 235,
|
||||||
|
'name': 'Sad',
|
||||||
|
'old_id': 'g:sad',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/sad',
|
||||||
|
},
|
||||||
|
'Sci-Fi': {
|
||||||
|
'id': 236,
|
||||||
|
'name': 'Science Fiction',
|
||||||
|
'old_id': 'g:scifi',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/scifi',
|
||||||
|
},
|
||||||
|
'Slice of Life': {
|
||||||
|
'id': 237,
|
||||||
|
'name': 'Slice of Life',
|
||||||
|
'old_id': 'g:slice_of_life',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/slice-of-life',
|
||||||
|
},
|
||||||
|
'Thriller': {
|
||||||
|
'id': 238,
|
||||||
|
'name': 'Thriller',
|
||||||
|
'old_id': 'g:thriller',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/thriller',
|
||||||
|
},
|
||||||
|
'Tragedy': {
|
||||||
|
'id': 239,
|
||||||
|
'name': 'Tragedy',
|
||||||
|
'old_id': 'g:tragedy',
|
||||||
|
'type': 'genre',
|
||||||
|
'url': 'https://www.fimfiction.net/tag/tragedy',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class Handler(Iterable[Tuple[str, Any]]):
|
||||||
|
"""
|
||||||
|
Maps story meta to another style.
|
||||||
|
"""
|
||||||
|
attrs: Iterable[str] = tuple()
|
||||||
|
static: Dict[str, Any] = dict()
|
||||||
|
paths: Dict[str, ParsedResult] = dict()
|
||||||
|
|
||||||
|
def __init__(self, meta: Dict[str, Any]) -> None:
|
||||||
|
"""
|
||||||
|
Constructor.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
meta: The story meta to map.
|
||||||
|
"""
|
||||||
|
self.meta = meta
|
||||||
|
|
||||||
|
def __getattr__(self, key: str) -> Any:
|
||||||
|
"""
|
||||||
|
Returns values from indirect sources.
|
||||||
|
"""
|
||||||
|
if key in self.static:
|
||||||
|
return self.static[key]
|
||||||
|
|
||||||
|
if key in self.paths:
|
||||||
|
meta = self.meta
|
||||||
|
path = self.paths[key]
|
||||||
|
return path.search(meta)
|
||||||
|
|
||||||
|
return self.meta.get(key)
|
||||||
|
|
||||||
|
def __iter__(self) -> Iterator[Tuple[str, Any]]:
|
||||||
|
"""
|
||||||
|
Yields all story meta items.
|
||||||
|
"""
|
||||||
|
for attr in self.attrs:
|
||||||
|
value = getattr(self, attr)
|
||||||
|
yield attr, value
|
||||||
|
|
||||||
|
|
||||||
|
class ArchiveHandler(Handler):
|
||||||
|
"""
|
||||||
|
Maps an archive meta dict from root.
|
||||||
|
"""
|
||||||
|
attrs = (
|
||||||
|
'date_checked',
|
||||||
|
'date_created',
|
||||||
|
'date_fetched',
|
||||||
|
'date_updated',
|
||||||
|
'path',
|
||||||
|
)
|
||||||
|
|
||||||
|
paths = {
|
||||||
|
'date_checked': jmes('archive.date_checked'),
|
||||||
|
'date_created': jmes('archive.date_created'),
|
||||||
|
'date_fetched': jmes('archive.date_fetched'),
|
||||||
|
'date_updated': jmes('archive.date_updated'),
|
||||||
|
'path': jmes('archive.path || path'),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class AuthorHandler(Handler):
|
||||||
|
"""
|
||||||
|
Maps an author meta dict.
|
||||||
|
"""
|
||||||
|
attrs = (
|
||||||
|
'avatar',
|
||||||
|
'bio_html',
|
||||||
|
'date_joined',
|
||||||
|
'id',
|
||||||
|
'name',
|
||||||
|
'num_blog_posts',
|
||||||
|
'num_followers',
|
||||||
|
'num_stories',
|
||||||
|
'url',
|
||||||
|
)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def url(self):
|
||||||
|
uid = int(self.id)
|
||||||
|
name = urlquote(str(self.name))
|
||||||
|
return f'{HOST}/user/{uid}/{name}'
|
||||||
|
|
||||||
|
|
||||||
|
class ChapterHandler(Handler):
|
||||||
|
"""
|
||||||
|
Maps a chapter meta dict.
|
||||||
|
"""
|
||||||
|
attrs = (
|
||||||
|
'chapter_number',
|
||||||
|
'date_modified',
|
||||||
|
'date_published',
|
||||||
|
'id',
|
||||||
|
'num_views',
|
||||||
|
'num_words',
|
||||||
|
'published',
|
||||||
|
'title',
|
||||||
|
'url',
|
||||||
|
)
|
||||||
|
|
||||||
|
static = {
|
||||||
|
'published': True,
|
||||||
|
}
|
||||||
|
|
||||||
|
paths = {
|
||||||
|
'url': jmes('link'),
|
||||||
|
'num_views': jmes('views'),
|
||||||
|
'num_words': jmes('words'),
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self, meta: Dict[str, Any], index: int) -> None:
|
||||||
|
"""
|
||||||
|
Constructor.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
meta: The chapter meta to map.
|
||||||
|
index: The current chapter index.
|
||||||
|
"""
|
||||||
|
self.meta = meta
|
||||||
|
self.chapter_number = int(index) + 1
|
||||||
|
|
||||||
|
@property
|
||||||
|
def date_modified(self) -> Optional[str]:
|
||||||
|
timestamp = self.meta.get('date_modified')
|
||||||
|
|
||||||
|
if timestamp is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
return arrow.get(timestamp).isoformat()
|
||||||
|
|
||||||
|
|
||||||
|
class RootHandler(Handler):
|
||||||
|
"""
|
||||||
|
Maps a root meta dict.
|
||||||
|
"""
|
||||||
|
attrs = (
|
||||||
|
'archive',
|
||||||
|
'author',
|
||||||
|
'chapters',
|
||||||
|
'color',
|
||||||
|
'completion_status',
|
||||||
|
'content_rating',
|
||||||
|
'cover_image',
|
||||||
|
'date_modified',
|
||||||
|
'date_published',
|
||||||
|
'date_updated',
|
||||||
|
'description_html',
|
||||||
|
'id',
|
||||||
|
'num_chapters',
|
||||||
|
'num_comments',
|
||||||
|
'num_dislikes',
|
||||||
|
'num_likes',
|
||||||
|
'num_views',
|
||||||
|
'num_words',
|
||||||
|
'prequel',
|
||||||
|
'published',
|
||||||
|
'rating',
|
||||||
|
'short_description',
|
||||||
|
'status',
|
||||||
|
'submitted',
|
||||||
|
'tags',
|
||||||
|
'title',
|
||||||
|
'total_num_views',
|
||||||
|
'url',
|
||||||
|
)
|
||||||
|
|
||||||
|
static = {
|
||||||
|
'date_modified': EPOCH,
|
||||||
|
'published': True,
|
||||||
|
'status': 'visible',
|
||||||
|
'submitted': True,
|
||||||
|
}
|
||||||
|
|
||||||
|
paths = {
|
||||||
|
'num_chapters': jmes('chapter_count'),
|
||||||
|
'num_comments': jmes('comments'),
|
||||||
|
'num_dislikes': jmes('dislikes'),
|
||||||
|
'num_likes': jmes('likes'),
|
||||||
|
'num_views': jmes('views'),
|
||||||
|
'num_words': jmes('words'),
|
||||||
|
'total_num_views': jmes('total_views'),
|
||||||
|
}
|
||||||
|
|
||||||
|
@property
|
||||||
|
def archive(self) -> Dict[str, Any]:
|
||||||
|
handler = ArchiveHandler(self.meta)
|
||||||
|
return dict(iter(handler))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def author(self) -> Dict[str, Any]:
|
||||||
|
author = self.meta.get('author') or dict()
|
||||||
|
handler = AuthorHandler(author)
|
||||||
|
return dict(iter(handler))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def chapters(self) -> List[Dict[str, Any]]:
|
||||||
|
items = enumerate(self.meta.get('chapters') or list())
|
||||||
|
handlers = (ChapterHandler(c, i) for i, c in items)
|
||||||
|
return [dict(iter(handler)) for handler in handlers]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def completion_status(self) -> Optional[str]:
|
||||||
|
status = self.meta.get('status')
|
||||||
|
return status and status.strip().lower()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def content_rating(self) -> Optional[str]:
|
||||||
|
rating = self.meta.get('content_rating_text')
|
||||||
|
return rating and rating.strip().lower()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def cover_image(self) -> Dict[str, Any]:
|
||||||
|
image = self.meta.get('image')
|
||||||
|
|
||||||
|
if image is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
base = image.rsplit("-", 1)[0]
|
||||||
|
assert base.startswith('http')
|
||||||
|
|
||||||
|
return {
|
||||||
|
'full': f'{base}-full',
|
||||||
|
'large': f'{base}-large',
|
||||||
|
'medium': f'{base}-medium',
|
||||||
|
'thumbnail': f'{base}-tiny',
|
||||||
|
}
|
||||||
|
|
||||||
|
@property
|
||||||
|
def date_updated(self) -> Optional[str]:
|
||||||
|
timestamp = self.meta.get('date_modified')
|
||||||
|
|
||||||
|
if timestamp is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
return arrow.get(timestamp).isoformat()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def description_html(self) -> Optional[str]:
|
||||||
|
desc = self.meta.get('description')
|
||||||
|
|
||||||
|
if desc is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
html = bbcode.render_html(desc)
|
||||||
|
return f'<p>{html.strip()}</p>'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def rating(self) -> int:
|
||||||
|
likes = self.num_likes
|
||||||
|
dislikes = self.num_dislikes
|
||||||
|
|
||||||
|
if None in (likes, dislikes):
|
||||||
|
return None
|
||||||
|
|
||||||
|
try:
|
||||||
|
return round(likes / (likes + dislikes) * 100)
|
||||||
|
except ZeroDivisionError:
|
||||||
|
return 50
|
||||||
|
|
||||||
|
@property
|
||||||
|
def tags(self) -> List[Dict[str, Any]]:
|
||||||
|
cats = self.meta.get('categories') or dict()
|
||||||
|
tags = [TAGS[k] for k, v in cats.items() if v]
|
||||||
|
return deepcopy(tags)
|
||||||
|
|
||||||
|
|
||||||
|
class AlphaBetaConverter(Converter):
|
||||||
|
"""
|
||||||
|
Converts story meta from alpha to beta format.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __call__(self, story: Story) -> Story:
|
||||||
|
if MetaFormat.ALPHA not in story.flavors:
|
||||||
|
raise ValueError(f"Missing flavor: {MetaFormat.ALPHA}")
|
||||||
|
|
||||||
|
handler = RootHandler(story.meta)
|
||||||
|
meta = dict(iter(handler))
|
||||||
|
|
||||||
|
flavors = set(story.flavors)
|
||||||
|
flavors.remove(MetaFormat.ALPHA)
|
||||||
|
flavors.add(MetaFormat.BETA)
|
||||||
|
|
||||||
|
return story.merge(meta=meta, flavors=flavors)
|
|
@ -1,4 +1,5 @@
|
||||||
arrow
|
arrow
|
||||||
|
bbcode
|
||||||
blinker
|
blinker
|
||||||
jmespath
|
jmespath
|
||||||
requests
|
requests
|
||||||
|
|
164
tests/converters/test_alpha_beta.json
Normal file
164
tests/converters/test_alpha_beta.json
Normal file
|
@ -0,0 +1,164 @@
|
||||||
|
{
|
||||||
|
"pairs": [
|
||||||
|
{
|
||||||
|
"alpha": {
|
||||||
|
"author": {
|
||||||
|
"id": 18,
|
||||||
|
"name": "Sethisto"
|
||||||
|
},
|
||||||
|
"categories": {
|
||||||
|
"2nd Person": false,
|
||||||
|
"Adventure": false,
|
||||||
|
"Alternate Universe": false,
|
||||||
|
"Anthro": false,
|
||||||
|
"Comedy": false,
|
||||||
|
"Crossover": false,
|
||||||
|
"Dark": false,
|
||||||
|
"Drama": false,
|
||||||
|
"Equestria Girls": false,
|
||||||
|
"Horror": false,
|
||||||
|
"Human": false,
|
||||||
|
"Mystery": false,
|
||||||
|
"Random": true,
|
||||||
|
"Romance": true,
|
||||||
|
"Sad": false,
|
||||||
|
"Sci-Fi": false,
|
||||||
|
"Slice of Life": false,
|
||||||
|
"Thriller": false,
|
||||||
|
"Tragedy": false
|
||||||
|
},
|
||||||
|
"chapter_count": 1,
|
||||||
|
"chapters": [
|
||||||
|
{
|
||||||
|
"date_modified": 1390908352,
|
||||||
|
"id": 10,
|
||||||
|
"link": "https://www.fimfiction.net/story/9/1/the-greatest-equine-who-has-ever-lived/chapter-1",
|
||||||
|
"title": "Chapter 1",
|
||||||
|
"views": 9943,
|
||||||
|
"words": 321
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"comments": 223,
|
||||||
|
"content_rating": 0,
|
||||||
|
"content_rating_text": "Everyone",
|
||||||
|
"date_modified": 1309035953,
|
||||||
|
"description": "REDACTED",
|
||||||
|
"dislikes": 51,
|
||||||
|
"full_image": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-full",
|
||||||
|
"id": 9,
|
||||||
|
"image": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-medium",
|
||||||
|
"likes": 365,
|
||||||
|
"short_description": "",
|
||||||
|
"status": "Incomplete",
|
||||||
|
"title": "The Greatest Equine Who has Ever Lived!",
|
||||||
|
"total_views": 9943,
|
||||||
|
"url": "https://www.fimfiction.net/story/9/the-greatest-equine-who-has-ever-lived",
|
||||||
|
"views": 9943,
|
||||||
|
"words": 321
|
||||||
|
},
|
||||||
|
"beta": {
|
||||||
|
"author": {
|
||||||
|
"avatar": {
|
||||||
|
"128": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-128",
|
||||||
|
"16": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-16",
|
||||||
|
"192": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-192",
|
||||||
|
"256": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-256",
|
||||||
|
"32": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-32",
|
||||||
|
"384": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-384",
|
||||||
|
"48": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-48",
|
||||||
|
"512": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-512",
|
||||||
|
"64": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-64",
|
||||||
|
"96": "https://cdn-img.fimfiction.net/user/t74v-1431818459-18-96"
|
||||||
|
},
|
||||||
|
"bio_html": "",
|
||||||
|
"date_joined": "2011-06-25T16:53:48-04:00",
|
||||||
|
"id": 18,
|
||||||
|
"name": "Sethisto",
|
||||||
|
"num_blog_posts": 0,
|
||||||
|
"num_followers": 137,
|
||||||
|
"num_stories": 1,
|
||||||
|
"url": "https://www.fimfiction.net/user/18/Sethisto"
|
||||||
|
},
|
||||||
|
"chapters": [
|
||||||
|
{
|
||||||
|
"chapter_number": 1,
|
||||||
|
"date_modified": "2014-01-28T06:25:52-05:00",
|
||||||
|
"date_published": "2011-07-08T14:04:11-04:00",
|
||||||
|
"id": 10,
|
||||||
|
"num_views": 9943,
|
||||||
|
"num_words": 321,
|
||||||
|
"published": true,
|
||||||
|
"title": "Chapter 1",
|
||||||
|
"url": "https://www.fimfiction.net/story/9/1/the-greatest-equine-who-has-ever-lived/chapter-1"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"color": {
|
||||||
|
"hex": "3e3e7e",
|
||||||
|
"rgb": [
|
||||||
|
62,
|
||||||
|
62,
|
||||||
|
126
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"completion_status": "incomplete",
|
||||||
|
"content_rating": "everyone",
|
||||||
|
"cover_image": {
|
||||||
|
"full": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-full",
|
||||||
|
"large": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-large",
|
||||||
|
"medium": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-medium",
|
||||||
|
"thumbnail": "https://cdn-img.fimfiction.net/story/vr3n-1432418803-9-tiny"
|
||||||
|
},
|
||||||
|
"date_modified": "1969-12-31T19:00:00-05:00",
|
||||||
|
"date_published": "2011-07-08T14:04:11-04:00",
|
||||||
|
"date_updated": "2011-06-25T17:05:53-04:00",
|
||||||
|
"description_html": "<p>REDACTED</p>",
|
||||||
|
"id": 9,
|
||||||
|
"num_chapters": 1,
|
||||||
|
"num_comments": 223,
|
||||||
|
"num_dislikes": 51,
|
||||||
|
"num_likes": 365,
|
||||||
|
"num_views": 9943,
|
||||||
|
"num_words": 321,
|
||||||
|
"prequel": null,
|
||||||
|
"published": true,
|
||||||
|
"rating": 88,
|
||||||
|
"short_description": "",
|
||||||
|
"status": "visible",
|
||||||
|
"submitted": true,
|
||||||
|
"tags": [
|
||||||
|
{
|
||||||
|
"id": 20,
|
||||||
|
"name": "Trixie",
|
||||||
|
"old_id": "c:21",
|
||||||
|
"type": "character",
|
||||||
|
"url": "https://www.fimfiction.net/tag/trixie"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 6,
|
||||||
|
"name": "Twilight Sparkle",
|
||||||
|
"old_id": "c:7",
|
||||||
|
"type": "character",
|
||||||
|
"url": "https://www.fimfiction.net/tag/twilight-sparkle"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 234,
|
||||||
|
"name": "Random",
|
||||||
|
"old_id": "g:random",
|
||||||
|
"type": "genre",
|
||||||
|
"url": "https://www.fimfiction.net/tag/random"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 120,
|
||||||
|
"name": "Romance",
|
||||||
|
"old_id": "g:romance",
|
||||||
|
"type": "genre",
|
||||||
|
"url": "https://www.fimfiction.net/tag/romance"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"title": "The Greatest Equine Who has Ever Lived!",
|
||||||
|
"total_num_views": 9943,
|
||||||
|
"url": "https://www.fimfiction.net/story/9/the-greatest-equine-who-has-ever-lived"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
148
tests/converters/test_alpha_beta.py
Normal file
148
tests/converters/test_alpha_beta.py
Normal file
|
@ -0,0 +1,148 @@
|
||||||
|
"""
|
||||||
|
Alpha to beta converter tests.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# 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 json
|
||||||
|
from copy import deepcopy
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
import arrow
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from fimfarchive.converters import AlphaBetaConverter
|
||||||
|
from fimfarchive.flavors import MetaFormat
|
||||||
|
|
||||||
|
|
||||||
|
def to_null(data: Dict[str, Any], *keys: str) -> None:
|
||||||
|
"""
|
||||||
|
Nulls the requested keys.
|
||||||
|
"""
|
||||||
|
for key in keys:
|
||||||
|
data[key] = None
|
||||||
|
|
||||||
|
|
||||||
|
def to_utc(data: Dict[str, Any], *keys: str) -> None:
|
||||||
|
"""
|
||||||
|
Converts the requested keys to UTC time strings.
|
||||||
|
"""
|
||||||
|
for key in keys:
|
||||||
|
value = data.get(key)
|
||||||
|
|
||||||
|
if value is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
time = arrow.get(value).to('utc')
|
||||||
|
data[key] = time.isoformat()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope='module')
|
||||||
|
def data():
|
||||||
|
"""
|
||||||
|
Returns test data from JSON.
|
||||||
|
"""
|
||||||
|
path = f'{__file__[:-3]}.json'
|
||||||
|
|
||||||
|
with open(path, 'rt') as fobj:
|
||||||
|
return json.load(fobj)
|
||||||
|
|
||||||
|
|
||||||
|
class TestAlphaBetaConverter:
|
||||||
|
"""
|
||||||
|
AlphaBetaConverter tests.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def converter(self):
|
||||||
|
"""
|
||||||
|
Returns an alpha beta converter instance.
|
||||||
|
"""
|
||||||
|
return AlphaBetaConverter()
|
||||||
|
|
||||||
|
@pytest.fixture(params=range(1))
|
||||||
|
def pair(self, request, data):
|
||||||
|
"""
|
||||||
|
Returns meta test data pairs.
|
||||||
|
"""
|
||||||
|
return data['pairs'][request.param]
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def alpha(self, pair):
|
||||||
|
"""
|
||||||
|
Returns meta in alpha format.
|
||||||
|
"""
|
||||||
|
return deepcopy(pair['alpha'])
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def beta(self, pair):
|
||||||
|
"""
|
||||||
|
Returns meta in beta format.
|
||||||
|
"""
|
||||||
|
return deepcopy(pair['beta'])
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def expected(self, beta):
|
||||||
|
"""
|
||||||
|
Returns the expected meta result.
|
||||||
|
"""
|
||||||
|
data = deepcopy(beta)
|
||||||
|
|
||||||
|
data['archive'] = {
|
||||||
|
'date_checked': None,
|
||||||
|
'date_created': None,
|
||||||
|
'date_fetched': None,
|
||||||
|
'date_updated': None,
|
||||||
|
'path': None,
|
||||||
|
}
|
||||||
|
|
||||||
|
to_null(data, 'color', 'date_published')
|
||||||
|
to_utc(data, 'date_modified', 'date_updated')
|
||||||
|
|
||||||
|
to_null(data['author'], *(
|
||||||
|
'avatar',
|
||||||
|
'bio_html',
|
||||||
|
'date_joined',
|
||||||
|
'num_blog_posts',
|
||||||
|
'num_followers',
|
||||||
|
'num_stories',
|
||||||
|
))
|
||||||
|
|
||||||
|
for chapter in data['chapters']:
|
||||||
|
to_null(chapter, 'date_published')
|
||||||
|
to_utc(chapter, 'date_modified')
|
||||||
|
|
||||||
|
data['tags'] = [
|
||||||
|
tag for tag in data['tags']
|
||||||
|
if tag['type'] in {'content', 'genre', 'series'}
|
||||||
|
]
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
def test_conversion(self, converter, story, expected, alpha):
|
||||||
|
"""
|
||||||
|
Tests conversion of story meta from alpha to beta format.
|
||||||
|
"""
|
||||||
|
story = story.merge(flavors=[MetaFormat.ALPHA], meta=alpha)
|
||||||
|
converted = converter(story)
|
||||||
|
|
||||||
|
assert MetaFormat.BETA in converted.flavors
|
||||||
|
assert expected == converted.meta
|
Loading…
Reference in a new issue