Skip to content

Commit

Permalink
Weasyl Extractor (#977)
Browse files Browse the repository at this point in the history
* weasyl extractor

* @kattjevfel suggested changes

* @mikf changes
  • Loading branch information
Zanny authored Sep 25, 2020
1 parent d5fa716 commit ebb7737
Show file tree
Hide file tree
Showing 3 changed files with 209 additions and 0 deletions.
2 changes: 2 additions & 0 deletions docs/supportedsites.rst
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ Twitter https://twitter.com/ |twitter-C|
VSCO https://vsco.co/ Collections, individual Images, User Profiles
Wallhaven https://wallhaven.cc/ individual Images, Search Results Optional (`API Key <configuration.rst#extractorwallhavenapi-key>`__)
Warosu https://warosu.org/ Threads
Weasyl https://www.weasyl.com/ |weasyl-C|
Webtoon https://www.webtoons.com/ Comics, Episodes
Weibo https://www.weibo.com/ Images from Statuses, User Profiles
WikiArt.org https://www.wikiart.org/ Artists, Artist Listings, Artworks
Expand Down Expand Up @@ -163,4 +164,5 @@ Turboimagehost https://www.turboimagehost.com/ individual Images
.. |reddit-C| replace:: individual Images, Submissions, Subreddits, User Profiles
.. |smugmug-C| replace:: Albums, individual Images, Images from Users and Folders
.. |twitter-C| replace:: Bookmarks, Likes, Media Timelines, Search Results, Timelines, Tweets
.. |weasyl-C| replace:: Folders, Journals, Journals, Submissions, Submissions
.. |yuki-S| replace:: yuki.la 4chan archive
1 change: 1 addition & 0 deletions gallery_dl/extractor/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@
"vsco",
"wallhaven",
"warosu",
"weasyl",
"webtoons",
"weibo",
"wikiart",
Expand Down
206 changes: 206 additions & 0 deletions gallery_dl/extractor/weasyl.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
# -*- coding: utf-8 -*-

# 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.

"""Extractors for https://www.weasyl.com/"""

from .common import Extractor, Message
from .. import text
import re

BASE_PATTERN = r"(?:https://)?(?:www\.)?weasyl.com/"


class WeasylExtractor(Extractor):
category = "weasyl"
directory_fmt = ("{category}", "{owner_login}")
filename_fmt = "{submitid}_{title}.{extension}"
archive_fmt = "{submitid}"
root = "https://www.weasyl.com"

def __init__(self, match):
Extractor.__init__(self, match)

@staticmethod
def populate_submission(data):
# Some submissions don't have content and can be skipped
if "submission" in data["media"]:
data["url"] = data["media"]["submission"][0]["url"]
data["extension"] = text.ext_from_url(data["url"])
return True
return False

def request_submission(self, submitid):
return self.request(
"{}/api/submissions/{}/view".format(self.root, submitid)).json()

def retrieve_journal(self, id):
data = self.request(
"{}/api/journals/{}/view".format(self.root, id)).json()
data["extension"] = "html"
data["html"] = "text:" + data["content"]
return data

def submissions(self):
nextid = 0
while nextid is not None:
url = "{}/api/users/{}/gallery?nextid={}".format(
self.root, self.owner_login, nextid
)
folderid = self.folderid if hasattr(self, "folderid") else None
if folderid:
url += "&folderid={}".format(self.folderid)
json = self.request(url).json()
for data in json["submissions"]:
if self.populate_submission(data):
data["folderid"] = folderid
# Do any submissions have more than one url? If so
# a urllist of the submission array urls would work.
yield Message.Url, data["url"], data
nextid = json["nextid"]


class WeasylSubmissionExtractor(WeasylExtractor):
subcategory = "submission"
pattern = (BASE_PATTERN +
r"(?:~[\w-]+/submissions|submission)/(\d+)/?([\w-]+)?")
test = (
"https://www.weasyl.com/submission/2031/a-wesley", {
"keyword": {
"url": "https://cdn.weasyl.com/~fiz/submissions/2031/41ebc1c29"
"40be928532785dfbf35c37622664d2fbb8114c3b063df969562fc5"
"1/fiz-a-wesley.png",
}
}
)

def __init__(self, match):
WeasylExtractor.__init__(self, match)
self.submitid = int(match.group(1))
if len(match.groups()) == 3:
self.title = match.group(2)

def items(self):
yield Message.Version, 1
data = self.request_submission(self.submitid)
yield Message.Directory, data
if self.populate_submission(data):
yield Message.Url, data["url"], data


class WeasylSubmissionsExtractor(WeasylExtractor):
subcategory = "submissions"
pattern = BASE_PATTERN + r"(?:~([\w-]+)/?|submissions/([\w-]+))$"
test = (
("https://www.weasyl.com/~tanidareal", {
"count": ">= 200"
}),
("https://www.weasyl.com/submissions/tanidareal", {
"count": ">= 200"
})
)

def __init__(self, match):
WeasylExtractor.__init__(self, match)
self.owner_login = match.group(1) if match.group(1) else match.group(2)

def items(self):
yield Message.Version, 1
yield Message.Directory, {"owner_login": self.owner_login}
yield from self.submissions()


class WeasylFolderExtractor(WeasylExtractor):
subcategory = "folder"
directory_fmt = ("{category}", "{owner_login}", "{folder_name}")
pattern = BASE_PATTERN + r"submissions/([\w-]+)\?folderid=(\d+)"
test = (
"https://www.weasyl.com/submissions/tanidareal?folderid=7403", {
"count": ">= 12"
}
)

def __init__(self, match):
WeasylExtractor.__init__(self, match)
self.owner_login = match.group(1)
self.folderid = int(match.group(2))

def items(self):
yield Message.Version, 1
iter = self.submissions()
# Folder names are only on single submission api calls
msg, url, data = next(iter)
details = self.request_submission(data["submitid"])
yield Message.Directory, details
yield msg, url, data
yield from iter


class WeasylJournalExtractor(WeasylExtractor):
subcategory = "journal"
filename_fmt = "{journalid}_{title}.{extension}"
archive_fmt = "{journalid}"
pattern = BASE_PATTERN + r"journal/(\d+)/?([\w-]+)?"
test = (
("https://www.weasyl.com/journal/17647", {
"keyword": {
"content":
"<p><a>javascript:alert(42);</a></p><p>No more of that!</p>",
"title": "bbcode",
}
}),
("https://www.weasyl.com/journal/17647/bbcode", {
"keyword": {
"content":
"<p><a>javascript:alert(42);</a></p><p>No more of that!</p>",
"title": "bbcode",
}
})
)

def __init__(self, match):
WeasylExtractor.__init__(self, match)
self.journalid = int(match.group(1))
if match.group(2):
self.title = match.group(2)

def items(self):
yield Message.Version, 1
data = self.retrieve_journal(self.journalid)
if hasattr(self, "title"):
data["title"] = self.title
else:
data["title"] = data["title"].lower()
yield Message.Directory, data
yield Message.Url, data["html"], data


class WeasylJournalsExtractor(WeasylExtractor):
subcategory = "journals"
filename_fmt = "{journalid}_{title}.{extension}"
archive_fmt = "{journalid}"
pattern = BASE_PATTERN + r"journals/([\w-]+)"
test = (
"https://www.weasyl.com/journals/charmander", {
"count": ">= 2",
}
)

def __init__(self, match):
WeasylExtractor.__init__(self, match)
self.owner_login = match.group(1)

def items(self):
yield Message.Version, 1
yield Message.Directory, {"owner_login": self.owner_login}
response = self.request("{}/journals/{}".format(
self.root, self.owner_login
))

for journal in re.finditer(r'"/journal/(\d+)/([\w-]+)"',
response.text):
data = self.retrieve_journal(int(journal.group(1)))
data["title"] = journal.group(2)
yield Message.Url, data["html"], data

0 comments on commit ebb7737

Please sign in to comment.