mirror of
https://github.com/soimort/you-get.git
synced 2025-01-24 05:55:02 +03:00
rebuild flickr extractor
This commit is contained in:
parent
d0a7655727
commit
91b196baef
@ -1,39 +1,228 @@
|
||||
#!/usr/bin/env python
|
||||
|
||||
__all__ = ['flickr_download']
|
||||
__all__ = ['flickr_download_main']
|
||||
|
||||
from ..common import *
|
||||
|
||||
def flickr_download(url, output_dir='.', merge=True, info_only=False, **kwargs):
|
||||
page = get_html(url)
|
||||
title = match1(page, r'<meta property="og:title" content="([^"]*)"')
|
||||
photo_id = match1(page, r'"id":"([0-9]+)"')
|
||||
import json
|
||||
|
||||
try: # extract video
|
||||
html = get_html('https://secure.flickr.com/apps/video/video_mtl_xml.gne?photo_id=%s' % photo_id)
|
||||
node_id = match1(html, r'<Item id="id">(.+)</Item>')
|
||||
secret = match1(html, r'<Item id="photo_secret">(.+)</Item>')
|
||||
pattern_url_photoset = r'https?://www\.flickr\.com/photos/.+/(?:(?:sets)|(?:albums))?/([^/]+)'
|
||||
pattern_url_photostream = r'https?://www\.flickr\.com/photos/([^/]+)(?:/|(?:/page))?$'
|
||||
pattern_url_single_photo = r'https?://www\.flickr\.com/photos/[^/]+/(\d+)'
|
||||
pattern_url_gallery = r'https?://www\.flickr\.com/photos/[^/]+/galleries/(\d+)'
|
||||
pattern_url_group = r'https?://www\.flickr\.com/groups/([^/]+)'
|
||||
pattern_url_favorite = r'https?://www\.flickr\.com/photos/([^/]+)/favorites'
|
||||
|
||||
html = get_html('https://secure.flickr.com/video_playlist.gne?node_id=%s&secret=%s' % (node_id, secret))
|
||||
app = match1(html, r'APP="([^"]+)"')
|
||||
fullpath = unescape_html(match1(html, r'FULLPATH="([^"]+)"'))
|
||||
url = app + fullpath
|
||||
pattern_inline_title = r'<title>([^<]*)</title>'
|
||||
pattern_inline_api_key = r'api\.site_key\s*=\s*"([^"]+)"'
|
||||
pattern_inline_img_url = r'"url":"([^"]+)","key":"[^"]+"}}'
|
||||
pattern_inline_NSID = r'"nsid"\s*:\s*"([^"]+)"'
|
||||
pattern_inline_video_mark = r'("mediaType":"video")'
|
||||
|
||||
# (api_key, method, ext, page)
|
||||
tmpl_api_call = (
|
||||
'https://api.flickr.com/services/rest?'
|
||||
'&format=json&nojsoncallback=1'
|
||||
# UNCOMMENT FOR TESTING
|
||||
#'&per_page=5'
|
||||
'&per_page=500'
|
||||
# this parameter CANNOT take control of 'flickr.galleries.getPhotos'
|
||||
# though the doc said it should.
|
||||
# it's always considered to be 500
|
||||
'&api_key=%s'
|
||||
'&method=flickr.%s'
|
||||
'&extras=url_sq,url_q,url_t,url_s,url_n,url_m,url_z,url_c,url_l,url_h,url_k,url_o,media'
|
||||
'%s&page=%d'
|
||||
)
|
||||
|
||||
tmpl_api_call_video_info = (
|
||||
'https://api.flickr.com/services/rest?'
|
||||
'&format=json&nojsoncallback=1'
|
||||
'&method=flickr.video.getStreamInfo'
|
||||
'&api_key=%s'
|
||||
'&photo_id=%s'
|
||||
'&secret=%s'
|
||||
)
|
||||
|
||||
tmpl_api_call_photo_info = (
|
||||
'https://api.flickr.com/services/rest?'
|
||||
'&format=json&nojsoncallback=1'
|
||||
'&method=flickr.photos.getInfo'
|
||||
'&api_key=%s'
|
||||
'&photo_id=%s'
|
||||
)
|
||||
|
||||
# looks that flickr won't return urls for all sizes
|
||||
# we required in 'extras field without a acceptable header
|
||||
dummy_header = {
|
||||
'User-Agent':'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:53.0) Gecko/20100101 Firefox/53.0'
|
||||
}
|
||||
def get_content_headered(url):
|
||||
return get_content(url, dummy_header)
|
||||
|
||||
def get_photoset_id(url, page):
|
||||
return match1(url, pattern_url_photoset)
|
||||
|
||||
def get_photo_id(url, page):
|
||||
return match1(url, pattern_url_single_photo)
|
||||
|
||||
def get_gallery_id(url, page):
|
||||
return match1(url, pattern_url_gallery)
|
||||
|
||||
def get_api_key(page):
|
||||
match = match1(page, pattern_inline_api_key)
|
||||
# this happens only when the url points to a gallery page
|
||||
# that contains no inline api_key(and never makes xhr api calls)
|
||||
# in fact this might be a better approch for getting a temporary api key
|
||||
# since there's no place for a user to add custom infomation that may
|
||||
# misguide the regex in the homepage
|
||||
if not match:
|
||||
return match1(get_html('https://flickr.com'), r'"site_key"\s*:\s*"([^"]+)"')
|
||||
return match
|
||||
|
||||
def get_NSID(url, page):
|
||||
return match1(page, pattern_inline_NSID)
|
||||
|
||||
# [
|
||||
# (
|
||||
# regex_match_url,
|
||||
# remote_api_method,
|
||||
# additional_query_parameter_for_method,
|
||||
# parser_for_additional_parameter,
|
||||
# field_where_photourls_are_saved
|
||||
# )
|
||||
# ]
|
||||
url_patterns = [
|
||||
# www.flickr.com/photos/{username|NSID}/sets|albums/{album-id}
|
||||
(
|
||||
pattern_url_photoset,
|
||||
'photosets.getPhotos',
|
||||
'photoset_id',
|
||||
get_photoset_id,
|
||||
'photoset'
|
||||
),
|
||||
# www.flickr.com/photos/{username|NSID}/{pageN}?
|
||||
(
|
||||
pattern_url_photostream,
|
||||
# according to flickr api documentation, this method needs to be
|
||||
# authenticated in order to filter photo visible to the calling user
|
||||
# but it seems works fine anonymously as well
|
||||
'people.getPhotos',
|
||||
'user_id',
|
||||
get_NSID,
|
||||
'photos'
|
||||
),
|
||||
# www.flickr.com/photos/{username|NSID}/galleries/{gallery-id}
|
||||
(
|
||||
pattern_url_gallery,
|
||||
'galleries.getPhotos',
|
||||
'gallery_id',
|
||||
get_gallery_id,
|
||||
'photos'
|
||||
),
|
||||
# www.flickr.com/groups/{groupname|groupNSID}/
|
||||
(
|
||||
pattern_url_group,
|
||||
'groups.pools.getPhotos',
|
||||
'group_id',
|
||||
get_NSID,
|
||||
'photos'
|
||||
),
|
||||
# www.flickr.com/photos/{username|NSID}/favorites/*
|
||||
(
|
||||
pattern_url_favorite,
|
||||
'favorites.getList',
|
||||
'user_id',
|
||||
get_NSID,
|
||||
'photos'
|
||||
)
|
||||
]
|
||||
|
||||
def flickr_download_main(url, output_dir = '.', merge = False, info_only = False, **kwargs):
|
||||
urls = None
|
||||
size = 'o' # works for collections only
|
||||
title = None
|
||||
if 'stream_id' in kwargs:
|
||||
size = kwargs['stream_id']
|
||||
if match1(url, pattern_url_single_photo):
|
||||
url, title = get_single_photo_url(url)
|
||||
urls = [url]
|
||||
else:
|
||||
urls, title = fetch_photo_url_list(url, size)
|
||||
index = 0
|
||||
for url in urls:
|
||||
mime, ext, size = url_info(url)
|
||||
|
||||
print_info(site_info, title, mime, size)
|
||||
print_info('Flickr.com', title, mime, size)
|
||||
if not info_only:
|
||||
download_urls([url], title, ext, size, output_dir, merge=merge, faker=True)
|
||||
suffix = '[%d]' % index
|
||||
download_urls([url], title + suffix, ext, False, output_dir, None, False, False)
|
||||
index = index + 1
|
||||
|
||||
except: # extract images
|
||||
image = match1(page, r'<meta property="og:image" content="([^"]*)')
|
||||
ext = 'jpg'
|
||||
_, _, size = url_info(image)
|
||||
def fetch_photo_url_list(url, size):
|
||||
for pattern in url_patterns:
|
||||
# FIXME: fix multiple matching since the match group is dropped
|
||||
if match1(url, pattern[0]):
|
||||
return fetch_photo_url_list_impl(url, *pattern[1:], size)
|
||||
raise NotImplementedError('Flickr extractor is not supported for %s.' % url)
|
||||
|
||||
print_info(site_info, title, ext, size)
|
||||
if not info_only:
|
||||
download_urls([image], title, ext, size, output_dir, merge=merge)
|
||||
def fetch_photo_url_list_impl(url, method, id_field, id_parse_func, collection_name, size):
|
||||
page = get_html(url)
|
||||
api_key = get_api_key(page)
|
||||
ext_field = ''
|
||||
if id_parse_func:
|
||||
ext_field = '&%s=%s' % (id_field, id_parse_func(url, page))
|
||||
page_number = 1
|
||||
urls = []
|
||||
while True:
|
||||
call_url = tmpl_api_call % (api_key, method, ext_field, page_number)
|
||||
photoset = json.loads(get_content_headered(call_url))[collection_name]
|
||||
pagen = photoset['page']
|
||||
pages = photoset['pages']
|
||||
for info in photoset['photo']:
|
||||
url = get_url_of_largest(info, api_key, size)
|
||||
urls.append(url)
|
||||
page_number = page_number + 1
|
||||
# the typeof 'page' and 'pages' may change in different methods
|
||||
if str(pagen) == str(pages):
|
||||
break
|
||||
return urls, match1(page, pattern_inline_title)
|
||||
|
||||
# image size suffixes used in inline json 'key' field
|
||||
# listed in descending order
|
||||
size_suffixes = ['o', 'k', 'h', 'l', 'c', 'z', 'm', 'n', 's', 't', 'q', 'sq']
|
||||
|
||||
def get_orig_video_source(api_key, pid, secret):
|
||||
parsed = json.loads(get_content_headered(tmpl_api_call_video_info % (api_key, pid, secret)))
|
||||
for stream in parsed['streams']['stream']:
|
||||
if stream['type'] == 'orig':
|
||||
return stream['_content'].replace('\\', '')
|
||||
return None
|
||||
|
||||
def get_url_of_largest(info, api_key, size):
|
||||
if info['media'] == 'photo':
|
||||
sizes = size_suffixes
|
||||
if size in sizes:
|
||||
sizes = sizes[sizes.index(size):]
|
||||
for suffix in sizes:
|
||||
if 'url_' + suffix in info:
|
||||
return info['url_' + suffix].replace('\\', '')
|
||||
return None
|
||||
else:
|
||||
return get_orig_video_source(api_key, info['id'], info['secret'])
|
||||
|
||||
def get_single_photo_url(url):
|
||||
page = get_html(url)
|
||||
pid = get_photo_id(url, page)
|
||||
title = match1(page, pattern_inline_title)
|
||||
if match1(page, pattern_inline_video_mark):
|
||||
api_key = get_api_key(page)
|
||||
reply = get_content(tmpl_api_call_photo_info % (api_key, get_photo_id(url, page)))
|
||||
secret = json.loads(reply)['photo']['secret']
|
||||
return get_orig_video_source(api_key, pid, secret), title
|
||||
#last match always has the best resolution
|
||||
match = match1(page, pattern_inline_img_url)
|
||||
return 'https:' + match.replace('\\', ''), title
|
||||
|
||||
site_info = "Flickr.com"
|
||||
download = flickr_download
|
||||
download_playlist = playlist_not_supported('flickr')
|
||||
download = flickr_download_main
|
||||
download_playlist = playlist_not_supported('flickr');
|
||||
|
Loading…
Reference in New Issue
Block a user