diff --git a/docs/supportedsites.md b/docs/supportedsites.md index a15566df981..639cd082a95 100644 --- a/docs/supportedsites.md +++ b/docs/supportedsites.md @@ -559,6 +559,12 @@ Consider all sites to be NSFW unless otherwise known. Blogs, Posts + + NaverPost + https://post.naver.com/ + Posts, User Profiles + + NaverWebtoon https://comic.naver.com/ diff --git a/gallery_dl/extractor/__init__.py b/gallery_dl/extractor/__init__.py index 22e4fe34123..c05425bcc8f 100644 --- a/gallery_dl/extractor/__init__.py +++ b/gallery_dl/extractor/__init__.py @@ -100,6 +100,7 @@ "myhentaigallery", "myportfolio", "naver", + "naverpost", "naverwebtoon", "newgrounds", "nhentai", diff --git a/gallery_dl/extractor/naverpost.py b/gallery_dl/extractor/naverpost.py new file mode 100644 index 00000000000..56de752c37b --- /dev/null +++ b/gallery_dl/extractor/naverpost.py @@ -0,0 +1,143 @@ +# -*- 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://post.naver.com/""" + +from .common import Extractor, Message +from .. import text, exception +import json +import re + +BASE_PATTERN = r"(?:https?://)?(?:m\.)?post\.naver\.com" + + +class NaverpostExtractor(Extractor): + """Base class for naver post extractors""" + category = "naverpost" + root = "https://post.naver.com" + request_interval = (0.5, 1.5) + + def _call(self, url, params=None): + if params is None: + params = {} + while True: + try: + return self.request(url, params=params) + except exception.HttpError as exc: + if exc.status == 401: + raise exception.AuthenticationError() + if exc.status == 403: + raise exception.AuthorizationError() + if exc.status == 404: + raise exception.NotFoundError(self.subcategory) + self.log.debug(exc) + return + + def _pagination(self, url, params=None): + if params is None: + params = {} + while True: + res = self._call(url, params).text + # the `html` string in the response contains escaped single quotes, + # which would throw a JSONDecodeError exception + res = json.loads(res.replace(r"\'", "'")) + urls = [] + endpoints = text.extract_iter( + res["html"], '
\n', ' ') or + text.extr(page, '', ' ') + ).replace(",", "")), + "url": self.url, + } + return data + + def items(self): + page = self._call(self.url).text + data = self.metadata(page) + + yield Message.Directory, data + + image_classes = ("img_attachedfile", "se_mediaImage") + image_query = r"\?type=w\d+$" + for image in text.extract_iter(page, ""): + img = { + "id": text.extr(image, ' id="', '"'), + "title": text.extr(image, ' title="', '"'), + "attachment-id": text.extr( + image, ' data-attachment-id="', '"'), + "alt": None, + } + classes = text.extr(image, ' class="', '"').split() + if not any(item in classes for item in image_classes): + continue + url = text.extr(image, ' data-src="', '"') + if not re.search(image_query, url): + continue + url = re.sub(image_query, "", url) + img["url"] = url + alt = text.extr(image, ' alt="', '"') + if alt and alt.endswith(".jpg"): + img["alt"] = alt + data["filename"], _, data["extension"] = alt.rpartition(".") + else: + text.nameext_from_url(text.unquote(url), data) + data["image"] = img + yield Message.Url, url, data + + +class NaverpostUserExtractor(NaverpostExtractor): + """Extractor for all posts from a user on post.naver.com""" + subcategory = "user" + pattern = BASE_PATTERN + r"/my\.naver\?memberNo=(\d+)" + example = "https://post.naver.com/my.naver?memberNo=12345" + + def __init__(self, match): + NaverpostExtractor.__init__(self, match) + self.member_no = match.group(1) + + def items(self): + data = {"_extractor": NaverpostPostExtractor} + endpoint = self.root + "/async/my.naver" + params = {"memberNo": self.member_no} + posts = self._pagination(endpoint, params) + for url in posts: + yield Message.Queue, url, data diff --git a/scripts/supportedsites.py b/scripts/supportedsites.py index 470b629d6d4..ea7e93030b2 100755 --- a/scripts/supportedsites.py +++ b/scripts/supportedsites.py @@ -89,6 +89,7 @@ "mastodon.social": "mastodon.social", "myhentaigallery": "My Hentai Gallery", "myportfolio" : "Adobe Portfolio", + "naverpost" : "NaverPost", "naverwebtoon" : "NaverWebtoon", "nhentai" : "nhentai", "nijie" : "nijie", diff --git a/test/results/naverpost.py b/test/results/naverpost.py new file mode 100644 index 00000000000..2eba74a6075 --- /dev/null +++ b/test/results/naverpost.py @@ -0,0 +1,53 @@ +# -*- 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. + +from gallery_dl.extractor import naverpost + +IMAGE_URL_PATTERN = r"(?i)https://post-phinf\.pstatic\.net/.*\.(?:jpe?g|png|gif|webp)" + + +__tests__ = ( +{ + "#url": "https://m.post.naver.com/viewer/postView.nhn?volumeNo=15861102&memberNo=16220685", + "#comment": ".nhn page extension", + "#category": ("", "naverpost", "post"), + "#class": naverpost.NaverpostPostExtractor, + "#pattern": IMAGE_URL_PATTERN, + "#count": 34, + + "title": "[쇼! 음악중심] 180526 방탄소년단 FAKE LOVE 현장 포토", + "description": "[BY MBC예능연구소] [쇼! 음악중심] 589회, 20180526 ※본 콘텐츠는 상업적 용도의 사용을 금합니다.", + "author": "MBC예능연구소", + "date": "dt:2018-05-29 12:09:34", + "views": int, +}, + +{ + "#url": "https://post.naver.com/viewer/postView.naver?volumeNo=31389956&memberNo=29156514", + "#comment": ".naver page extension", + "#category": ("", "naverpost", "post"), + "#class": naverpost.NaverpostPostExtractor, + "#pattern": IMAGE_URL_PATTERN, + "#count": 48, + + "title": "매일 밤 꿈꿔 왔던 드림캐쳐 '바람아' 활동 비하인드 현장", + "description": "[BY 드림캐쳐컴퍼니] 안녕하세요.드림캐쳐 포스트 지기입니다!(*・▽・*)'Odd Eye' 활동이 끝나고 아쉬웠을...", + "author": "드림캐쳐컴퍼니", + "date": "dt:2021-05-03 06:00:09", + "views": int, +}, + +{ + "#url": "https://post.naver.com/my.naver?memberNo=29156514", + "#comment": "up to 20 posts are returned per request", + "#category": ("", "naverpost", "user"), + "#class": naverpost.NaverpostUserExtractor, + "#pattern": naverpost.NaverpostPostExtractor.pattern, + "#range": "1-21", + "#count": 21, +}, + +)