You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
gallery-dl/gallery_dl/extractor/deviantart.py

164 lines
6.1 KiB

# -*- coding: utf-8 -*-
# Copyright 2015 Mike Fährmann
#
# 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.
"""Extract images from http://www.deviantart.com/"""
from .common import Extractor, Message
from .. import text, exception
from ..cache import cache
class DeviantartUserExtractor(Extractor):
"""Extractor for all works from an artist on deviantart.com"""
category = "deviantart"
subcategory = "user"
directory_fmt = ["{category}", "{username}"]
filename_fmt = "{category}_{index}_{title}.{extension}"
pattern = [r"(?:https?://)?([^\.]+)\.deviantart\.com(?:/gallery)?/?$"]
test = [("http://shimoda7.deviantart.com/gallery/", {
"url": "63bfa8efba199e27181943c9060f6770f91a8441",
"keyword": "4ffe227a50f373faf643d7e5ae89a04859af8d19",
})]
def __init__(self, match):
Extractor.__init__(self)
self.api = DeviantartAPI(self.session)
self.user = match.group(1)
def items(self):
first = True
yield Message.Version, 1
for deviation in self.api.gallery_all(self.user):
del deviation["stats"]
if first:
yield Message.Directory, deviation["author"]
first = False
deviation["index"] = deviation["url"].rsplit("-", maxsplit=1)[-1]
yield Message.Url, deviation["content"]["src"], deviation
class DeviantartImageExtractor(Extractor):
"""Extractor for single images from deviantart.com"""
category = "deviantart"
subcategory = "image"
directory_fmt = ["{category}", "{artist}"]
filename_fmt = "{category}_{index}_{title}.{extension}"
pattern = [r"(?:https?://)?([^\.]+\.deviantart\.com/art/.+-(\d+))"]
test = [("http://shimoda7.deviantart.com/art/For-the-sake-of-a-memory-10073852", {
"url": "71345ce3bef5b19bd2a56d7b96e6b5ddba747c2e",
"keyword": "ccac27b8f740fc943afca9460608e02c6cbcdf96",
"content": "6a7c74dc823ebbd457bdd9b3c2838a6ee728091e",
})]
def __init__(self, match):
Extractor.__init__(self)
self.url = "https://" + match.group(1)
self.index = match.group(2)
self.session.cookies["agegate_state"] = "1"
def items(self):
page = self.request(self.url).text
data = self.get_data(page)
data.update(self.get_image(page))
text.nameext_from_url(data["image"], data)
data["title"] = text.unescape(data["title"])
data["description"] = text.unescape(text.unescape(data["description"]))
data["artist"] = text.extract(data["url"], "//", ".")[0]
data["date"] = text.extract(data["date"], ", ", " in ", len(data["title"]))[0]
yield Message.Version, 1
yield Message.Directory, data
yield Message.Url, data["image"], data
def get_data(self, page):
"""Collect metadata for extractor-job"""
return text.extract_all(page, (
('title' , '"og:title" content="', '"'),
('url' , '"og:url" content="', '"'),
('description', '"og:description" content="', '"'),
(None , '<span class="tt-w">', ''),
('date' , 'title="', '"'),
), values={"index": self.index})[0]
def get_image(self, page):
"""Find image-url and -dimensions"""
# try preview
data, pos = text.extract_all(page, (
('image' , '"og:image" content="', '"'),
('width' , '"og:image:width" content="', '"'),
('height', '"og:image:height" content="', '"'),
))
if data["image"].startswith("https://orig"):
return data
# try main image
data, pos = text.extract_all(page, (
(None , 'class="dev-content-normal "', ''),
('image' , ' src="', '"'),
('width' , ' width="', '"'),
('height', ' height="', '"'),
), pos)
if data["image"].startswith("https://orig"):
return data
# try download
test, pos = text.extract(page, 'dev-page-download', '', pos)
if test is not None:
data, pos = text.extract_all(page, (
('image' , 'href="', '"'),
(None , '<span class="text">', ' '),
('width' , '', ' '),
('height', ' ', '<'),
), pos)
response = self.session.head(text.unescape(data["image"]))
data["image"] = response.headers["Location"]
return data
class DeviantartAPI():
"""Minimal interface for the deviantart API"""
def __init__(self, session, client_id="5388",
client_secret="76b08c69cfb27f26d6161f9ab6d061a1"):
self.session = session
self.client_id = client_id
self.client_secret = client_secret
def gallery_all(self, username, offset=0):
"""Yield all Deviation-objects of a specific user """
self.authenticate()
url = "https://www.deviantart.com/api/v1/oauth2/gallery/all"
params = {"username": username, "offset": offset}
while True:
data = self.session.get(url, params=params).json()
yield from data["results"]
if not data["has_more"]:
return
params["offset"] = data["next_offset"]
def authenticate(self):
"""Authenticate the application by requesting a bearer token"""
bearer_token = self._authenticate_impl(
self.client_id, self.client_secret
)
self.session.headers["Authorization"] = bearer_token
@cache(maxage=3600, keyarg=1)
def _authenticate_impl(self, client_id, client_secret):
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:
raise exception.AuthenticationError
return "Bearer " + response.json()["access_token"]