2015-10-04 04:13:50 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2017-02-23 20:58:39 +01:00
|
|
|
# Copyright 2015-2017 Mike Fährmann
|
2015-10-04 04:13:50 +02:00
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
|
|
# published by the Free Software Foundation.
|
|
|
|
|
2017-04-20 13:20:41 +02:00
|
|
|
"""Extract images from https://www.deviantart.com/"""
|
2015-10-04 04:13:50 +02:00
|
|
|
|
2017-01-12 21:08:49 +01:00
|
|
|
from .common import Extractor, Message
|
|
|
|
from .. import text, exception
|
2017-07-12 09:47:01 +02:00
|
|
|
from ..cache import cache, memcache
|
2017-05-13 21:42:29 +02:00
|
|
|
import itertools
|
2017-05-13 15:34:20 +02:00
|
|
|
import datetime
|
2017-03-08 16:40:20 +01:00
|
|
|
import time
|
2017-04-03 18:23:13 +02:00
|
|
|
import re
|
2015-10-04 04:13:50 +02:00
|
|
|
|
2017-01-12 21:08:49 +01:00
|
|
|
|
2017-04-03 14:56:47 +02:00
|
|
|
class DeviantartExtractor(Extractor):
|
|
|
|
"""Base class for deviantart extractors"""
|
2015-11-21 04:26:30 +01:00
|
|
|
category = "deviantart"
|
|
|
|
filename_fmt = "{category}_{index}_{title}.{extension}"
|
2017-07-10 18:14:40 +02:00
|
|
|
directory_fmt = ["{category}", "{author[urlname]}"]
|
2015-11-21 04:26:30 +01:00
|
|
|
|
2017-07-10 18:14:40 +02:00
|
|
|
def __init__(self, match=None):
|
2017-01-12 21:08:49 +01:00
|
|
|
Extractor.__init__(self)
|
2017-03-08 16:40:20 +01:00
|
|
|
self.api = DeviantartAPI(self)
|
2017-03-13 21:42:16 +01:00
|
|
|
self.offset = 0
|
|
|
|
|
2017-08-22 20:15:13 +02:00
|
|
|
if match:
|
|
|
|
self.user = match.group(1)
|
|
|
|
self.group = not self.api.user_profile(self.user)
|
|
|
|
if self.group:
|
|
|
|
self.subcategory = "group-" + self.subcategory
|
|
|
|
else:
|
|
|
|
self.user = None
|
|
|
|
self.group = False
|
|
|
|
|
2017-03-13 21:42:16 +01:00
|
|
|
def skip(self, num):
|
|
|
|
self.offset += num
|
|
|
|
return num
|
2015-10-04 04:13:50 +02:00
|
|
|
|
|
|
|
def items(self):
|
|
|
|
yield Message.Version, 1
|
2017-04-03 14:56:47 +02:00
|
|
|
for deviation in self.deviations():
|
2017-07-12 09:47:01 +02:00
|
|
|
if isinstance(deviation, str):
|
|
|
|
yield Message.Queue, deviation
|
|
|
|
continue
|
|
|
|
|
2017-04-03 18:23:13 +02:00
|
|
|
self.prepare(deviation)
|
2017-07-10 18:14:40 +02:00
|
|
|
yield Message.Directory, deviation
|
2017-05-10 16:45:45 +02:00
|
|
|
|
|
|
|
if "content" in deviation:
|
|
|
|
yield self.commit(deviation, deviation["content"])
|
|
|
|
|
|
|
|
if "videos" in deviation:
|
|
|
|
video = max(deviation["videos"],
|
|
|
|
key=lambda x: int(x["quality"][:-1]))
|
|
|
|
yield self.commit(deviation, video)
|
|
|
|
|
|
|
|
if "flash" in deviation:
|
|
|
|
yield self.commit(deviation, deviation["flash"])
|
|
|
|
|
|
|
|
if "excerpt" in deviation:
|
2017-05-13 21:42:29 +02:00
|
|
|
journal = self.api.deviation_content(deviation["deviationid"])
|
|
|
|
yield self.commit_journal(deviation, journal)
|
2017-04-03 14:56:47 +02:00
|
|
|
|
|
|
|
def deviations(self):
|
|
|
|
"""Return an iterable containing all relevant Deviation-objects"""
|
|
|
|
return []
|
|
|
|
|
2017-07-10 18:14:40 +02:00
|
|
|
def prepare(self, deviation):
|
2017-04-03 18:23:13 +02:00
|
|
|
"""Adjust the contents of a Deviation-object"""
|
2017-05-13 21:42:29 +02:00
|
|
|
for key in ("stats", "preview", "is_favourited", "allows_comments"):
|
2017-05-10 16:45:45 +02:00
|
|
|
if key in deviation:
|
|
|
|
del deviation[key]
|
|
|
|
try:
|
2017-08-16 12:13:42 +02:00
|
|
|
deviation["index"] = deviation["url"].rpartition("-")[2]
|
2017-05-10 16:45:45 +02:00
|
|
|
except KeyError:
|
|
|
|
deviation["index"] = 0
|
2017-04-03 18:23:13 +02:00
|
|
|
|
2017-07-10 18:14:40 +02:00
|
|
|
if self.user:
|
|
|
|
deviation["username"] = self.user
|
|
|
|
author = deviation["author"]
|
|
|
|
author["urlname"] = author["username"].lower()
|
|
|
|
deviation["da-category"] = deviation["category"]
|
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
@staticmethod
|
|
|
|
def commit(deviation, target):
|
|
|
|
url = target["src"]
|
|
|
|
deviation["target"] = text.nameext_from_url(url, target.copy())
|
|
|
|
deviation["extension"] = deviation["target"]["extension"]
|
2017-08-16 12:13:42 +02:00
|
|
|
if url.startswith("http:"):
|
|
|
|
url = "https:" + url[5:]
|
2017-05-10 16:45:45 +02:00
|
|
|
return Message.Url, url, deviation
|
2017-04-03 14:56:47 +02:00
|
|
|
|
2017-05-13 21:42:29 +02:00
|
|
|
@staticmethod
|
|
|
|
def commit_journal(deviation, journal):
|
|
|
|
title = text.escape(deviation["title"])
|
|
|
|
url = deviation["url"]
|
|
|
|
thumbs = deviation["thumbs"]
|
2017-05-15 15:58:06 +02:00
|
|
|
html = journal["html"]
|
|
|
|
date = datetime.datetime.utcfromtimestamp(deviation["published_time"])
|
2017-05-13 21:42:29 +02:00
|
|
|
shadow = SHADOW_TEMPLATE.format_map(thumbs[0]) if thumbs else ""
|
2017-05-13 15:34:20 +02:00
|
|
|
|
|
|
|
if "css" in journal:
|
|
|
|
css, cls = journal["css"], "withskin"
|
|
|
|
else:
|
|
|
|
css, cls = "", "journal-green"
|
|
|
|
|
2017-05-15 15:58:06 +02:00
|
|
|
if html.find('<div class="boxtop journaltop">', 0, 250) != -1:
|
|
|
|
needle = '<div class="boxtop journaltop">'
|
|
|
|
header = HEADER_CUSTOM_TEMPLATE.format(
|
|
|
|
title=title, url=url, date=str(date),
|
2017-05-13 21:42:29 +02:00
|
|
|
)
|
2017-05-15 15:58:06 +02:00
|
|
|
else:
|
|
|
|
needle = '<div usr class="gr">'
|
|
|
|
catlist = deviation["category_path"].split("/")
|
|
|
|
categories = " / ".join(
|
2017-08-16 12:13:42 +02:00
|
|
|
('<span class="crumb"><a href="https://www.deviantart.com/{}/"'
|
|
|
|
'><span>{}</span></a></span>').format(cpath, cat.capitalize())
|
2017-05-15 15:58:06 +02:00
|
|
|
for cat, cpath in zip(
|
|
|
|
catlist,
|
|
|
|
itertools.accumulate(catlist, lambda t, c: t + "/" + c)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
header = HEADER_TEMPLATE.format(
|
|
|
|
title=title,
|
|
|
|
url=url,
|
|
|
|
userurl=url[:url.find("/", 8)],
|
|
|
|
username=deviation["author"]["username"],
|
|
|
|
date=str(date),
|
|
|
|
categories=categories,
|
|
|
|
)
|
|
|
|
|
2017-05-13 15:34:20 +02:00
|
|
|
html = JOURNAL_TEMPLATE.format(
|
|
|
|
title=title,
|
2017-05-15 15:58:06 +02:00
|
|
|
html=html.replace(needle, header, 1),
|
2017-05-13 21:42:29 +02:00
|
|
|
shadow=shadow,
|
2017-05-13 15:34:20 +02:00
|
|
|
css=css,
|
|
|
|
cls=cls,
|
|
|
|
)
|
|
|
|
|
|
|
|
deviation["extension"] = "htm"
|
|
|
|
return Message.Url, html, deviation
|
|
|
|
|
2017-07-12 17:05:31 +02:00
|
|
|
@property
|
|
|
|
def flat(self):
|
|
|
|
return self.config("flat", True)
|
|
|
|
|
2017-07-03 21:57:10 +02:00
|
|
|
@staticmethod
|
2017-07-10 18:14:40 +02:00
|
|
|
def _find_folder(folders, name):
|
2017-07-27 20:50:33 +02:00
|
|
|
pattern = r"[^\w]*" + name.replace("-", r"[^\w]+") + r"[^\w]*$"
|
2017-07-03 21:57:10 +02:00
|
|
|
for folder in folders:
|
2017-07-27 20:50:33 +02:00
|
|
|
if re.match(pattern, folder["name"]):
|
2017-07-03 21:57:10 +02:00
|
|
|
return folder
|
|
|
|
raise exception.NotFoundError("folder")
|
|
|
|
|
2017-07-12 17:05:31 +02:00
|
|
|
def _folder_urls(self, folders, category):
|
2017-08-22 20:15:13 +02:00
|
|
|
url = "https://{}.deviantart.com/{}/0/".format(self.user, category)
|
|
|
|
return [url + folder["name"] for folder in folders]
|
2017-07-12 17:05:31 +02:00
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
|
|
|
|
class DeviantartGalleryExtractor(DeviantartExtractor):
|
|
|
|
"""Extractor for all deviations from an artist's gallery"""
|
|
|
|
subcategory = "gallery"
|
2017-07-06 20:40:50 +02:00
|
|
|
pattern = [r"(?:https?://)?([^.]+)\.deviantart\.com"
|
|
|
|
r"(?:/(?:gallery/?(?:\?catpath=/)?)?)?$"]
|
|
|
|
test = [
|
|
|
|
("http://shimoda7.deviantart.com/gallery/", {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "f95b222d939c1e6aa8b9aabe89eaa2d364f06d38",
|
2017-07-10 18:14:40 +02:00
|
|
|
"keyword": "9342c2a7a2bd6eb9f4a6ea539d04d75248ebe05f",
|
2017-07-06 20:40:50 +02:00
|
|
|
}),
|
2017-07-12 09:47:01 +02:00
|
|
|
("https://yakuzafc.deviantart.com/", {
|
|
|
|
"url": "fa6ecb2c3aa78872f762d43f7809b7f0580debc1",
|
|
|
|
}),
|
2017-07-12 17:05:31 +02:00
|
|
|
("http://shimoda7.deviantart.com/gallery/?catpath=/", None),
|
2017-07-06 20:40:50 +02:00
|
|
|
]
|
2015-12-06 21:13:57 +01:00
|
|
|
|
2017-04-03 14:56:47 +02:00
|
|
|
def deviations(self):
|
2017-08-22 20:15:13 +02:00
|
|
|
if self.flat and not self.group:
|
2017-07-12 09:47:01 +02:00
|
|
|
return self.api.gallery_all(self.user, self.offset)
|
|
|
|
else:
|
2017-07-12 17:05:31 +02:00
|
|
|
folders = self.api.gallery_folders(self.user)
|
|
|
|
return self._folder_urls(folders, "gallery")
|
2016-11-06 10:44:50 +01:00
|
|
|
|
|
|
|
|
2017-07-03 21:57:10 +02:00
|
|
|
class DeviantartFolderExtractor(DeviantartExtractor):
|
|
|
|
"""Extractor for deviations inside an artist's gallery folder"""
|
|
|
|
subcategory = "folder"
|
|
|
|
directory_fmt = ["{category}", "{folder[owner]}", "{folder[title]}"]
|
|
|
|
pattern = [r"(?:https?://)?([^.]+)\.deviantart\.com"
|
|
|
|
r"/gallery/(\d+)/([^/?&#]+)"]
|
2017-07-10 18:14:40 +02:00
|
|
|
test = [
|
|
|
|
("http://shimoda7.deviantart.com/gallery/722019/Miscellaneous", {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "1ee23a0bd8f7099d375afe8a29ea1a3bf394ba1e",
|
2017-07-10 18:14:40 +02:00
|
|
|
"keyword": "a0d7093148b9bab8ee0efa6213139efd99f23394",
|
|
|
|
}),
|
|
|
|
("http://majestic-da.deviantart.com/gallery/63419606/CHIBI-KAWAII", {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "1df6f4312f124b0ad9f2a905c8f9e94e89c84370",
|
2017-08-22 20:15:13 +02:00
|
|
|
"keyword": "b651f5d540aaaf7974fa7e181e4cc54151a65e9e",
|
2017-07-10 18:14:40 +02:00
|
|
|
}),
|
|
|
|
]
|
2017-07-03 21:57:10 +02:00
|
|
|
|
|
|
|
def __init__(self, match):
|
2017-08-22 20:15:13 +02:00
|
|
|
DeviantartExtractor.__init__(self, match)
|
2017-07-03 21:57:10 +02:00
|
|
|
self.user, fid, self.fname = match.groups()
|
|
|
|
self.folder = {"owner": self.user, "index": fid}
|
|
|
|
|
|
|
|
def deviations(self):
|
|
|
|
folders = self.api.gallery_folders(self.user)
|
2017-07-10 18:14:40 +02:00
|
|
|
folder = self._find_folder(folders, self.fname)
|
2017-07-03 21:57:10 +02:00
|
|
|
self.folder["title"] = folder["name"]
|
|
|
|
return self.api.gallery(self.user, folder["folderid"], self.offset)
|
|
|
|
|
|
|
|
def prepare(self, deviation):
|
2017-07-10 18:14:40 +02:00
|
|
|
DeviantartExtractor.prepare(self, deviation)
|
2017-07-03 21:57:10 +02:00
|
|
|
deviation["folder"] = self.folder
|
|
|
|
|
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
class DeviantartDeviationExtractor(DeviantartExtractor):
|
|
|
|
"""Extractor for single deviations"""
|
|
|
|
subcategory = "deviation"
|
2017-05-13 15:34:20 +02:00
|
|
|
pattern = [(r"(?:https?://)?([^.]+\.deviantart\.com/"
|
2017-05-10 17:21:33 +02:00
|
|
|
r"(?:art|journal)/[^/?&#]+-\d+)"),
|
2017-07-10 18:14:40 +02:00
|
|
|
(r"(?:https?://)?(sta\.sh/[a-z0-9]+)")]
|
2017-04-03 14:56:47 +02:00
|
|
|
test = [
|
|
|
|
(("http://shimoda7.deviantart.com/art/"
|
|
|
|
"For-the-sake-of-a-memory-10073852"), {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "393dc581ca9e6938dbf0a3db8e9eea6243eb35f4",
|
2017-07-10 18:14:40 +02:00
|
|
|
"keyword": "5f58ecdce9b9ebb51f65d0e24e0f7efe00a74a55",
|
2017-04-03 14:56:47 +02:00
|
|
|
"content": "6a7c74dc823ebbd457bdd9b3c2838a6ee728091e",
|
|
|
|
}),
|
|
|
|
("https://zzz.deviantart.com/art/zzz-1234567890", {
|
|
|
|
"exception": exception.NotFoundError,
|
|
|
|
}),
|
2017-04-17 11:52:16 +02:00
|
|
|
("http://sta.sh/01ijs78ebagf", {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "3a15ed9201e665172b1daece8ef6d42f6a7ad3d5",
|
2017-07-10 18:14:40 +02:00
|
|
|
"keyword": "00246726d49f51ab35ea88d66467067f05b10bc9",
|
2017-04-17 11:52:16 +02:00
|
|
|
}),
|
|
|
|
("http://sta.sh/abcdefghijkl", {
|
|
|
|
"exception": exception.NotFoundError,
|
|
|
|
}),
|
2017-04-03 14:56:47 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, match):
|
|
|
|
DeviantartExtractor.__init__(self)
|
|
|
|
self.url = "https://" + match.group(1)
|
|
|
|
|
|
|
|
def deviations(self):
|
2017-08-05 16:11:46 +02:00
|
|
|
response = self.request(self.url, fatal=False)
|
2017-04-03 14:56:47 +02:00
|
|
|
deviation_id = text.extract(response.text, '//deviation/', '"')[0]
|
|
|
|
if response.status_code != 200 or not deviation_id:
|
|
|
|
raise exception.NotFoundError("image")
|
|
|
|
return (self.api.deviation(deviation_id),)
|
2017-01-12 21:08:49 +01:00
|
|
|
|
|
|
|
|
2017-04-20 13:20:41 +02:00
|
|
|
class DeviantartFavoriteExtractor(DeviantartExtractor):
|
2017-06-28 17:39:07 +02:00
|
|
|
"""Extractor for an artist's favorites"""
|
2017-04-20 13:20:41 +02:00
|
|
|
subcategory = "favorite"
|
2017-07-10 18:14:40 +02:00
|
|
|
directory_fmt = ["{category}", "{username}", "Favourites"]
|
|
|
|
pattern = [r"(?:https?://)?([^.]+)\.deviantart\.com"
|
|
|
|
r"/favourites/?(?:\?catpath=/)?$"]
|
2017-04-03 18:23:13 +02:00
|
|
|
test = [
|
|
|
|
("http://h3813067.deviantart.com/favourites/", {
|
2017-08-16 12:13:42 +02:00
|
|
|
"url": "393dc581ca9e6938dbf0a3db8e9eea6243eb35f4",
|
2017-07-10 18:14:40 +02:00
|
|
|
"keyword": "c7d0a3bacc1e4c5625dda703e25affe047cbbc3f",
|
2017-04-03 18:23:13 +02:00
|
|
|
"content": "6a7c74dc823ebbd457bdd9b3c2838a6ee728091e",
|
|
|
|
}),
|
2017-07-10 18:14:40 +02:00
|
|
|
("http://h3813067.deviantart.com/favourites/?catpath=/", None),
|
2017-04-03 18:23:13 +02:00
|
|
|
]
|
|
|
|
|
2017-07-10 18:14:40 +02:00
|
|
|
def deviations(self):
|
2017-07-12 17:05:31 +02:00
|
|
|
folders = self.api.collections_folders(self.user)
|
|
|
|
if self.flat:
|
|
|
|
return itertools.chain.from_iterable([
|
|
|
|
self.api.collections(self.user, folder["folderid"])
|
|
|
|
for folder in folders
|
|
|
|
])
|
|
|
|
else:
|
|
|
|
return self._folder_urls(folders, "favourites")
|
2017-07-10 18:14:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
class DeviantartCollectionExtractor(DeviantartExtractor):
|
|
|
|
"""Extractor for a single favorite collection"""
|
|
|
|
subcategory = "collection"
|
|
|
|
directory_fmt = ["{category}", "{collection[owner]}",
|
|
|
|
"Favourites", "{collection[title]}"]
|
|
|
|
pattern = [r"(?:https?://)?([^.]+)\.deviantart\.com"
|
|
|
|
r"/favourites/(\d+)/([^/?&#]+)"]
|
|
|
|
test = [("http://rosuuri.deviantart.com/favourites/58951174/Useful", {
|
2017-09-01 16:29:52 +02:00
|
|
|
"url": "f0c12581060aab9699289817b39804d9eb88f675",
|
2017-09-08 17:52:00 +02:00
|
|
|
"keyword": "2778b4abaac240ff6fb1d630d7b04b8e983ef9c4",
|
2017-07-10 18:14:40 +02:00
|
|
|
})]
|
|
|
|
|
2017-04-03 18:23:13 +02:00
|
|
|
def __init__(self, match):
|
2017-08-22 20:15:13 +02:00
|
|
|
DeviantartExtractor.__init__(self, match)
|
2017-07-10 18:14:40 +02:00
|
|
|
self.user, cid, self.cname = match.groups()
|
|
|
|
self.collection = {"owner": self.user, "index": cid}
|
2017-04-03 18:23:13 +02:00
|
|
|
|
|
|
|
def deviations(self):
|
2017-07-03 21:57:10 +02:00
|
|
|
folders = self.api.collections_folders(self.user)
|
2017-07-10 18:14:40 +02:00
|
|
|
folder = self._find_folder(folders, self.cname)
|
2017-07-03 21:57:10 +02:00
|
|
|
self.collection["title"] = folder["name"]
|
|
|
|
return self.api.collections(self.user, folder["folderid"], self.offset)
|
2017-04-03 18:23:13 +02:00
|
|
|
|
|
|
|
def prepare(self, deviation):
|
2017-07-10 18:14:40 +02:00
|
|
|
DeviantartExtractor.prepare(self, deviation)
|
2017-04-03 18:23:13 +02:00
|
|
|
deviation["collection"] = self.collection
|
|
|
|
|
|
|
|
|
2017-05-10 17:21:33 +02:00
|
|
|
class DeviantartJournalExtractor(DeviantartExtractor):
|
2017-06-28 17:39:07 +02:00
|
|
|
"""Extractor for an artist's journals"""
|
2017-05-10 17:21:33 +02:00
|
|
|
subcategory = "journal"
|
2017-07-10 18:14:40 +02:00
|
|
|
directory_fmt = ["{category}", "{username}", "Journal"]
|
2017-07-06 20:40:50 +02:00
|
|
|
pattern = [r"(?:https?://)?([^.]+)\.deviantart\.com"
|
|
|
|
r"/(?:journal|blog)/?(?:\?catpath=/)?$"]
|
|
|
|
test = [
|
2017-08-16 12:13:42 +02:00
|
|
|
("https://angrywhitewanker.deviantart.com/journal/", {
|
|
|
|
"url": "6474f49fbb4d01637ff0762708953252a52dc9c1",
|
|
|
|
"keyword": "5306515383a7ec26b22a2de42045718e6d630f25",
|
2017-07-06 20:40:50 +02:00
|
|
|
}),
|
|
|
|
("http://shimoda7.deviantart.com/journal/?catpath=/", None),
|
|
|
|
]
|
2017-05-10 17:21:33 +02:00
|
|
|
|
|
|
|
def deviations(self):
|
|
|
|
return self.api.browse_user_journals(self.user, self.offset)
|
|
|
|
|
|
|
|
|
2017-01-12 21:08:49 +01:00
|
|
|
class DeviantartAPI():
|
|
|
|
"""Minimal interface for the deviantart API"""
|
2017-03-08 16:40:20 +01:00
|
|
|
def __init__(self, extractor, client_id="5388",
|
2017-01-12 21:08:49 +01:00
|
|
|
client_secret="76b08c69cfb27f26d6161f9ab6d061a1"):
|
2017-03-08 16:40:20 +01:00
|
|
|
self.session = extractor.session
|
|
|
|
self.log = extractor.log
|
2017-09-09 17:31:42 +02:00
|
|
|
self.client_id = extractor.config("client-id", client_id)
|
|
|
|
self.client_secret = extractor.config("client-secret", client_secret)
|
2017-03-08 16:40:20 +01:00
|
|
|
self.delay = 0
|
2017-05-10 16:45:45 +02:00
|
|
|
self.mature = extractor.config("mature", "true")
|
2017-05-06 21:26:27 +02:00
|
|
|
if not isinstance(self.mature, str):
|
|
|
|
self.mature = "true" if self.mature else "false"
|
2017-01-12 21:08:49 +01:00
|
|
|
|
2017-05-10 17:21:33 +02:00
|
|
|
def browse_user_journals(self, username, offset=0):
|
|
|
|
"""Yield all journal entries of a specific user"""
|
|
|
|
endpoint = "browse/user/journals"
|
2017-07-06 20:40:50 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 50,
|
2017-05-10 17:21:33 +02:00
|
|
|
"mature_content": self.mature, "featured": "false"}
|
|
|
|
return self._pagination(endpoint, params)
|
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
def collections(self, username, folder_id, offset=0):
|
|
|
|
"""Yield all Deviation-objects contained in a collection folder"""
|
|
|
|
endpoint = "collections/" + folder_id
|
2017-07-06 20:40:50 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 24,
|
2017-05-06 21:26:27 +02:00
|
|
|
"mature_content": self.mature}
|
2017-04-03 14:56:47 +02:00
|
|
|
return self._pagination(endpoint, params)
|
|
|
|
|
2017-07-12 09:47:01 +02:00
|
|
|
@memcache(keyarg=1)
|
2017-04-03 14:56:47 +02:00
|
|
|
def collections_folders(self, username, offset=0):
|
|
|
|
"""Yield all collection folders of a specific user"""
|
|
|
|
endpoint = "collections/folders"
|
2017-07-03 21:57:10 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 50,
|
2017-05-06 21:26:27 +02:00
|
|
|
"mature_content": self.mature}
|
2017-07-12 09:47:01 +02:00
|
|
|
return self._pagination_list(endpoint, params)
|
2017-04-03 14:56:47 +02:00
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
def deviation(self, deviation_id):
|
|
|
|
"""Query and return info about a single Deviation"""
|
|
|
|
endpoint = "deviation/" + deviation_id
|
|
|
|
return self._call(endpoint)
|
|
|
|
|
|
|
|
def deviation_content(self, deviation_id):
|
2017-05-13 15:34:20 +02:00
|
|
|
"""Get extended content of a single Deviation"""
|
2017-05-10 16:45:45 +02:00
|
|
|
endpoint = "deviation/content"
|
|
|
|
params = {"deviationid": deviation_id}
|
|
|
|
return self._call(endpoint, params)
|
|
|
|
|
2017-07-03 21:57:10 +02:00
|
|
|
def gallery(self, username, folder_id="", offset=0):
|
|
|
|
"""Yield all Deviation-objects contained in a gallery folder"""
|
|
|
|
endpoint = "gallery/" + folder_id
|
2017-07-06 20:40:50 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 24,
|
2017-07-03 21:57:10 +02:00
|
|
|
"mature_content": self.mature, "mode": "newest"}
|
|
|
|
return self._pagination(endpoint, params)
|
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
def gallery_all(self, username, offset=0):
|
|
|
|
"""Yield all Deviation-objects of a specific user"""
|
|
|
|
endpoint = "gallery/all"
|
2017-07-06 20:40:50 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 24,
|
2017-05-06 21:26:27 +02:00
|
|
|
"mature_content": self.mature}
|
2017-04-03 14:56:47 +02:00
|
|
|
return self._pagination(endpoint, params)
|
2017-01-12 21:08:49 +01:00
|
|
|
|
2017-07-12 09:47:01 +02:00
|
|
|
@memcache(keyarg=1)
|
2017-07-03 21:57:10 +02:00
|
|
|
def gallery_folders(self, username, offset=0):
|
|
|
|
"""Yield all gallery folders of a specific user"""
|
|
|
|
endpoint = "gallery/folders"
|
2017-07-06 20:40:50 +02:00
|
|
|
params = {"username": username, "offset": offset, "limit": 50,
|
2017-07-03 21:57:10 +02:00
|
|
|
"mature_content": self.mature}
|
2017-07-12 09:47:01 +02:00
|
|
|
return self._pagination_list(endpoint, params)
|
|
|
|
|
2017-08-22 20:15:13 +02:00
|
|
|
@memcache(keyarg=1)
|
2017-07-12 09:47:01 +02:00
|
|
|
def user_profile(self, username):
|
|
|
|
"""Get user profile information"""
|
|
|
|
endpoint = "user/profile/" + username
|
|
|
|
return self._call(endpoint, expect_error=True)
|
2017-07-03 21:57:10 +02:00
|
|
|
|
2017-01-12 21:08:49 +01:00
|
|
|
def authenticate(self):
|
2017-04-03 14:56:47 +02:00
|
|
|
"""Authenticate the application by requesting an access token"""
|
|
|
|
access_token = self._authenticate_impl(
|
2017-01-12 21:08:49 +01:00
|
|
|
self.client_id, self.client_secret
|
|
|
|
)
|
2017-04-03 14:56:47 +02:00
|
|
|
self.session.headers["Authorization"] = access_token
|
2017-01-12 21:08:49 +01:00
|
|
|
|
2017-07-03 21:57:10 +02:00
|
|
|
@cache(maxage=3590, keyarg=1)
|
2017-01-12 21:08:49 +01:00
|
|
|
def _authenticate_impl(self, client_id, client_secret):
|
2017-03-08 16:40:20 +01:00
|
|
|
"""Actual authenticate implementation"""
|
2017-01-12 21:08:49 +01:00
|
|
|
url = "https://www.deviantart.com/oauth2/token"
|
|
|
|
data = {
|
|
|
|
"grant_type": "client_credentials",
|
|
|
|
"client_id": client_id,
|
|
|
|
"client_secret": client_secret,
|
|
|
|
}
|
|
|
|
response = self.session.post(url, data=data)
|
|
|
|
if response.status_code != 200:
|
2017-03-08 16:40:20 +01:00
|
|
|
raise exception.AuthenticationError()
|
2017-01-12 21:08:49 +01:00
|
|
|
return "Bearer " + response.json()["access_token"]
|
2017-03-08 16:40:20 +01:00
|
|
|
|
2017-07-12 09:47:01 +02:00
|
|
|
def _call(self, endpoint, params=None, expect_error=False):
|
2017-03-08 16:40:20 +01:00
|
|
|
"""Call an API endpoint"""
|
2017-04-03 14:56:47 +02:00
|
|
|
url = "https://www.deviantart.com/api/v1/oauth2/" + endpoint
|
2017-03-13 21:42:16 +01:00
|
|
|
tries = 1
|
2017-03-08 16:40:20 +01:00
|
|
|
while True:
|
|
|
|
if self.delay:
|
|
|
|
time.sleep(self.delay)
|
|
|
|
|
2017-03-13 21:42:16 +01:00
|
|
|
self.authenticate()
|
2017-03-08 16:40:20 +01:00
|
|
|
response = self.session.get(url, params=params)
|
|
|
|
|
|
|
|
if response.status_code == 200:
|
|
|
|
break
|
|
|
|
elif response.status_code == 429:
|
|
|
|
self.delay += 1
|
|
|
|
self.log.debug("rate limit (delay: %d)", self.delay)
|
|
|
|
else:
|
2017-07-12 09:47:01 +02:00
|
|
|
if expect_error:
|
|
|
|
return None
|
2017-03-08 16:40:20 +01:00
|
|
|
self.delay = 1
|
2017-03-13 21:42:16 +01:00
|
|
|
self.log.debug("http status code %d (%d/3)",
|
|
|
|
response.status_code, tries)
|
2017-03-08 16:40:20 +01:00
|
|
|
tries += 1
|
2017-03-13 21:42:16 +01:00
|
|
|
if tries > 3:
|
2017-03-08 16:40:20 +01:00
|
|
|
raise Exception(response.text)
|
|
|
|
try:
|
|
|
|
return response.json()
|
|
|
|
except ValueError:
|
|
|
|
return {}
|
2017-04-03 14:56:47 +02:00
|
|
|
|
|
|
|
def _pagination(self, endpoint, params=None):
|
|
|
|
while True:
|
|
|
|
data = self._call(endpoint, params)
|
|
|
|
if "results" in data:
|
|
|
|
yield from data["results"]
|
|
|
|
if not data["has_more"]:
|
|
|
|
return
|
|
|
|
params["offset"] = data["next_offset"]
|
|
|
|
else:
|
|
|
|
self.log.error("Unexpected API response: %s", data)
|
|
|
|
return
|
2017-05-10 16:45:45 +02:00
|
|
|
|
2017-07-12 09:47:01 +02:00
|
|
|
def _pagination_list(self, endpoint, params=None):
|
|
|
|
result = []
|
|
|
|
result.extend(self._pagination(endpoint, params))
|
|
|
|
return result
|
|
|
|
|
2017-05-10 16:45:45 +02:00
|
|
|
|
2017-05-13 21:42:29 +02:00
|
|
|
SHADOW_TEMPLATE = """
|
|
|
|
<span class="shadow">
|
|
|
|
<img src="{src}" class="smshadow" width="{width}" height="{height}">
|
|
|
|
</span>
|
|
|
|
<br><br>
|
|
|
|
"""
|
|
|
|
|
2017-05-13 15:34:20 +02:00
|
|
|
HEADER_TEMPLATE = """<div usr class="gr">
|
|
|
|
<div class="metadata">
|
|
|
|
<h2><a href="{url}">{title}</a></h2>
|
|
|
|
<ul>
|
|
|
|
<li class="author">
|
|
|
|
by <span class="name"><span class="username-with-symbol u">
|
|
|
|
<a class="u regular username" href="{userurl}">{username}</a>\
|
|
|
|
<span class="user-symbol regular"></span></span></span>,
|
|
|
|
<span>{date}</span>
|
|
|
|
</li>
|
|
|
|
<li class="category">
|
|
|
|
{categories}
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
"""
|
|
|
|
|
2017-05-15 15:58:06 +02:00
|
|
|
HEADER_CUSTOM_TEMPLATE = """<div class='boxtop journaltop'>
|
|
|
|
<h2>
|
2017-08-16 12:13:42 +02:00
|
|
|
<img src="https://st.deviantart.net/minish/gruzecontrol/icons/journal.gif\
|
2017-05-19 19:22:39 +02:00
|
|
|
?2" style="vertical-align:middle" alt=""/>
|
2017-05-15 15:58:06 +02:00
|
|
|
<a href="{url}">{title}</a>
|
|
|
|
</h2>
|
|
|
|
Journal Entry: <span>{date}</span>
|
|
|
|
"""
|
|
|
|
|
2017-05-12 14:10:25 +02:00
|
|
|
JOURNAL_TEMPLATE = """text:<!DOCTYPE html>
|
2017-05-10 16:45:45 +02:00
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta charset="utf-8">
|
|
|
|
<title>{title}</title>
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/deviantart-network_lc.css?3843780832">
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/group_secrets_lc.css?3250492874">
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/v6core_lc.css?4246581581">
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/sidebar_lc.css?1490570941">
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/writer_lc.css?3090682151">
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
css/v6loggedin_lc.css?3001430805">
|
|
|
|
<style>{css}</style>
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
roses/cssmin/core.css?1488405371919" >
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
roses/cssmin/peeky.css?1487067424177" >
|
2017-08-16 12:13:42 +02:00
|
|
|
<link rel="stylesheet" href="https://st.deviantart.net/\
|
2017-05-10 16:45:45 +02:00
|
|
|
roses/cssmin/desktop.css?1491362542749" >
|
|
|
|
</head>
|
|
|
|
<body id="deviantART-v7" class="bubble no-apps loggedout w960 deviantart">
|
|
|
|
<div id="output">
|
|
|
|
<div class="dev-page-container bubbleview">
|
|
|
|
<div class="dev-page-view view-mode-normal">
|
|
|
|
<div class="dev-view-main-content">
|
|
|
|
<div class="dev-view-deviation">
|
2017-05-13 21:42:29 +02:00
|
|
|
{shadow}
|
2017-05-10 16:45:45 +02:00
|
|
|
<div class="journal-wrapper tt-a">
|
|
|
|
<div class="journal-wrapper2">
|
2017-05-13 15:34:20 +02:00
|
|
|
<div class="journal {cls} journalcontrol">
|
2017-05-10 16:45:45 +02:00
|
|
|
{html}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
"""
|