200 lines
6.6 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Copyright 2014, 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 and ugoira from http://www.pixiv.net/"""
2015-10-05 17:15:31 +02:00
from .common import Extractor, Message
from .. import config, text
2014-10-12 21:56:44 +02:00
import re
2015-05-14 19:08:20 +02:00
import json
2014-10-12 21:56:44 +02:00
info = {
"category": "pixiv",
"extractor": "PixivExtractor",
2015-05-14 19:08:20 +02:00
"directory": ["{category}", "{artist-id}-{artist-nick}"],
"filename": "{category}_{artist-id}_{id}{num}.{extension}",
"pattern": [
r"(?:https?://)?(?:www\.)?pixiv\.net/member(?:_illust)?\.php\?id=(\d+)",
],
}
2015-10-05 17:15:31 +02:00
class PixivExtractor(Extractor):
2014-10-12 21:56:44 +02:00
member_url = "http://www.pixiv.net/member_illust.php"
illust_url = "http://www.pixiv.net/member_illust.php?mode=medium"
def __init__(self, match):
2015-10-05 17:15:31 +02:00
Extractor.__init__(self)
self.artist_id = match.group(1)
2015-05-14 19:08:20 +02:00
self.api = PixivAPI(self.session)
2014-10-12 21:56:44 +02:00
def items(self):
2015-05-14 19:08:20 +02:00
self.api.login(
config.get(("extractor", "pixiv", "username")),
config.get(("extractor", "pixiv", "password")),
2015-05-14 19:08:20 +02:00
)
metadata = self.get_job_metadata()
yield Message.Version, 1
2015-04-10 17:32:36 +02:00
yield Message.Headers, self.session.headers
yield Message.Cookies, self.session.cookies
2015-05-14 19:08:20 +02:00
yield Message.Directory, metadata
for work in self.get_works():
work.update(metadata)
pos = work["extension"].rfind("?", -18)
if pos != -1:
timestamp = work["extension"][pos:]
work["extension"] = work["extension"][:pos]
else:
timestamp = ""
2015-05-14 19:08:20 +02:00
if work["type"] == "ugoira":
2015-05-20 10:25:11 +02:00
url, framelist = self.parse_ugoira(work)
2015-05-14 19:08:20 +02:00
work["extension"] = "zip"
yield Message.Url, url, work.copy()
work["extension"] = "txt"
yield Message.Url, "text://"+framelist, work
elif work["page_count"] == 1:
yield Message.Url, work["url"], work
2014-10-12 21:56:44 +02:00
else:
2015-05-14 19:08:20 +02:00
url = work["url"]
ext = work["extension"]
off = url.rfind(".")
if url[off-2] == "p":
off -= 3
2015-05-14 19:08:20 +02:00
if work["id"] > 11319935 and "/img-original/" not in url:
big = "_big"
else:
big = ""
for i in range(work["page_count"]):
work["num"] = "_p{:02}".format(i)
url = "{}{}_p{}.{}{}".format(url[:off], big, i, ext, timestamp)
2015-05-14 19:08:20 +02:00
yield Message.Url, url, work.copy()
def get_works(self):
"""Yield all work-items for a pixiv-member"""
2015-10-03 15:43:02 +02:00
pagenum = 1
2014-10-12 21:56:44 +02:00
while True:
2015-10-03 15:43:02 +02:00
data = self.api.user_works(self.artist_id, pagenum)
2015-05-14 19:08:20 +02:00
for work in data["response"]:
url = work["image_urls"]["large"]
work["num"] = ""
work["url"] = url
work["extension"] = url[url.rfind(".")+1:]
yield work
pinfo = data["pagination"]
if pinfo["current"] == pinfo["pages"]:
2014-10-12 21:56:44 +02:00
return
2015-10-03 15:43:02 +02:00
pagenum = pinfo["next"]
2015-05-20 10:25:11 +02:00
def parse_ugoira(self, data):
"""Parse ugoira data"""
2014-10-12 21:56:44 +02:00
# get illust page
2015-10-03 15:43:02 +02:00
page = self.request(
2015-05-20 10:25:11 +02:00
self.illust_url, params={"illust_id": data["id"]},
2014-10-12 21:56:44 +02:00
).text
# parse page
2015-10-03 15:43:02 +02:00
frames, _ = text.extract(page, ',"frames":[', ']')
2014-10-12 21:56:44 +02:00
2015-05-20 10:25:11 +02:00
# build url
url = re.sub(
r"/img-original/(.+/\d+)[^/]+",
r"/img-zip-ugoira/\g<1>_ugoira1920x1080.zip",
data["url"]
)
2014-10-12 21:56:44 +02:00
# build framelist
2015-05-14 19:08:20 +02:00
framelist = re.sub(
2014-10-12 21:56:44 +02:00
r'\{"file":"([^"]+)","delay":(\d+)\},?',
2015-05-14 19:08:20 +02:00
r'\1 \2\n', frames
2014-10-12 21:56:44 +02:00
)
return url, framelist
def get_job_metadata(self):
"""Collect metadata for extractor-job"""
2015-05-14 19:08:20 +02:00
data = self.api.user(self.artist_id)["response"][0]
return {
"category": info["category"],
"artist-id": self.artist_id,
2015-05-14 19:08:20 +02:00
"artist-name": data["name"],
"artist-nick": data["account"],
}
2014-10-12 21:56:44 +02:00
class PixivAPI():
2015-05-14 19:08:20 +02:00
"""Minimal interface for the Pixiv Public-API for mobile devices
For a better and more complete implementation, see
- https://github.com/upbit/pixivpy
For in-depth information regarding the Pixiv Public-API, see
- http://blog.imaou.com/opensource/2014/10/09/pixiv_api_for_ios_update.html
"""
def __init__(self, session):
self.session = session
2015-05-14 19:08:20 +02:00
self.session.headers.update({
"Referer": "http://www.pixiv.net/",
"User-Agent": "PixivIOSApp/5.8.0",
2015-05-14 19:08:20 +02:00
# "Authorization": "Bearer 8mMXXWT9iuwdJvsVIvQsFYDwuZpRCMePeyagSh30ZdU",
})
def login(self, username, password):
"""Login and gain a Pixiv Public-API access token"""
data = {
"username": username,
"password": password,
"grant_type": "password",
"client_id": "bYGKuGVw91e0NMfPGp44euvGt59s",
"client_secret": "HP3RmkgAmEGro0gn1x9ioawQE8WMfvLXDz3ZqxpK",
}
2015-05-14 19:08:20 +02:00
response = self.session.post(
"https://oauth.secure.pixiv.net/auth/token", data=data
)
if response.status_code not in (200, 301, 302):
raise Exception("login() failed! check username and password.\n"
"HTTP %s: %s" % (response.status_code, response.text))
try:
token = self._parse(response)
self.session.headers["Authorization"] = (
"Bearer " + token["response"]["access_token"]
)
except:
raise Exception("Get access_token error! Response: %s" % (token))
def user(self, user_id):
"""Query information about a pixiv user"""
response = self.session.get(
"https://public-api.secure.pixiv.net/v1/users/"
"{user}.json".format(user=user_id)
)
return self._parse(response)
def user_works(self, user_id, page, per_page=20):
"""Query information about the works of a pixiv user"""
params = {
'page': page,
'per_page': per_page,
'image_sizes': 'large',
}
response = self.session.get(
"https://public-api.secure.pixiv.net/v1/users/"
"{user}/works.json".format(user=user_id), params=params
)
return self._parse(response)
2015-05-14 19:08:20 +02:00
@staticmethod
def _parse(response):
"""Parse a Pixiv Public-API response"""
return json.loads(response.text)