2 from __future__ import unicode_literals
6 from .common import InfoExtractor
7 from ..compat import compat_str
17 class PeerTubeIE(InfoExtractor):
18 _INSTANCES_RE = r'''(?:
19 # Taken from https://instances.joinpeertube.org/instances
20 tube\.openalgeria\.org|
21 peertube\.pointsecu\.fr|
22 peertube\.nogafa\.org|
24 megatube\.lilomoino\.fr|
25 peertube\.tamanoir\.foucry\.net|
26 peertube\.inapurna\.org|
27 peertube\.netzspielplatz\.de|
28 video\.deadsuperhero\.com|
29 peertube\.devosi\.org|
30 peertube\.1312\.media|
31 tube\.worldofhauru\.xyz|
32 tube\.bootlicker\.party|
34 peertube\.geekshell\.fr|
36 peertube\.peshane\.net|
38 tube\.homecomputing\.fr|
39 videos\.cloudfrancois\.fr|
40 peertube\.viviers-fibre\.net|
45 peertube\.gaialabs\.ch|
46 peertube\.extremely\.online|
47 peertube\.public-infrastructure\.eu|
57 peertube\.geekael\.fr|
59 video\.anormallostpod\.ovh|
62 videos\.iut-orsay\.fr|
63 peertube\.solidev\.net|
64 videos\.symphonie-of-code\.fr|
67 peertube\.gwendalavir\.eu|
68 video\.passageenseine\.fr|
69 videos\.festivalparminous\.org|
70 peertube\.touhoppai\.moe|
71 peertube\.duckdns\.org|
73 peertube\.mastodon\.host|
74 firedragonvideos\.com|
79 peertube\.walkingmountains\.fr|
83 jp\.peertube\.network|
86 peertube\.angristan\.xyz|
87 peertube\.parleur\.net|
96 peertube\.gegeweb\.eu|
99 tube\.conferences-gesticulees\.net|
100 peertube\.datagueule\.tv|
102 meilleurtube\.delire\.party|
103 tube\.mochi\.academy|
107 peertube\.valvin\.fr|
109 video\.colibris-outilslibres\.org|
110 video\.hispagatos\.org|
113 videos\.lecygnenoir\.info|
119 _UUID_RE = r'[\da-fA-F]{8}-[\da-fA-F]{4}-[\da-fA-F]{4}-[\da-fA-F]{4}-[\da-fA-F]{12}'
120 _VALID_URL = r'''(?x)
122 peertube:(?P<host>[^:]+):|
123 https?://(?P<host_2>%s)/(?:videos/(?:watch|embed)|api/v\d/videos)/
126 ''' % (_INSTANCES_RE, _UUID_RE)
128 'url': 'https://peertube.moe/videos/watch/2790feb0-8120-4e63-9af3-c943c69f5e6c',
129 'md5': '80f24ff364cc9d333529506a263e7feb',
131 'id': '2790feb0-8120-4e63-9af3-c943c69f5e6c',
134 'description': 'wow such video, so gif',
135 'thumbnail': r're:https?://.*\.(?:jpg|png)',
136 'timestamp': 1519297480,
137 'upload_date': '20180222',
138 'uploader': 'Luclu7',
139 'uploader_id': '7fc42640-efdb-4505-a45d-a15b1a5496f1',
140 'uploder_url': 'https://peertube.nsa.ovh/accounts/luclu7',
141 'license': 'Unknown',
145 'dislike_count': int,
150 'url': 'https://peertube.tamanoir.foucry.net/videos/watch/0b04f13d-1e18-4f1d-814e-4979aa7c9c44',
151 'only_matching': True,
154 'url': 'https://tube.22decembre.eu/videos/watch/9bb88cd3-9959-46d9-9ab9-33d2bb704c39',
155 'only_matching': True,
157 'url': 'https://tube.22decembre.eu/videos/embed/fed67262-6edb-4d1c-833b-daa9085c71d7',
158 'only_matching': True,
160 'url': 'https://tube.openalgeria.org/api/v1/videos/c1875674-97d0-4c94-a058-3f7e64c962e8',
161 'only_matching': True,
163 'url': 'peertube:video.blender.org:b37a5b9f-e6b5-415c-b700-04a5cd6ec205',
164 'only_matching': True,
168 def _extract_peertube_url(webpage, source_url):
170 r'https?://(?P<host>[^/]+)/videos/watch/(?P<id>%s)'
171 % PeerTubeIE._UUID_RE, source_url)
172 if mobj and any(p in webpage for p in (
174 'There will be other non JS-based clients to access PeerTube',
175 '>We are sorry but it seems that PeerTube is not compatible with your web browser.<')):
176 return 'peertube:%s:%s' % mobj.group('host', 'id')
179 def _extract_urls(webpage, source_url):
180 entries = re.findall(
181 r'''(?x)<iframe[^>]+\bsrc=["\'](?P<url>(?:https?:)?//%s/videos/embed/%s)'''
182 % (PeerTubeIE._INSTANCES_RE, PeerTubeIE._UUID_RE), webpage)
184 peertube_url = PeerTubeIE._extract_peertube_url(webpage, source_url)
186 entries = [peertube_url]
189 def _real_extract(self, url):
190 mobj = re.match(self._VALID_URL, url)
191 host = mobj.group('host') or mobj.group('host_2')
192 video_id = mobj.group('id')
194 video = self._download_json(
195 'https://%s/api/v1/videos/%s' % (host, video_id), video_id)
197 title = video['name']
200 for file_ in video['files']:
201 if not isinstance(file_, dict):
203 file_url = file_.get('fileUrl')
204 if not file_url or not isinstance(file_url, compat_str):
206 file_size = int_or_none(file_.get('size'))
208 file_, lambda x: x['resolution']['label'], compat_str)
209 f = parse_resolution(format_id)
212 'format_id': format_id,
213 'filesize': file_size,
216 self._sort_formats(formats)
218 def account_data(field):
219 return try_get(video, lambda x: x['account'][field], compat_str)
221 category = try_get(video, lambda x: x['category']['label'], compat_str)
222 categories = [category] if category else None
224 nsfw = video.get('nsfw')
226 age_limit = 18 if nsfw else 0
233 'description': video.get('description'),
234 'thumbnail': urljoin(url, video.get('thumbnailPath')),
235 'timestamp': unified_timestamp(video.get('publishedAt')),
236 'uploader': account_data('displayName'),
237 'uploader_id': account_data('uuid'),
238 'uploder_url': account_data('url'),
240 video, lambda x: x['licence']['label'], compat_str),
241 'duration': int_or_none(video.get('duration')),
242 'view_count': int_or_none(video.get('views')),
243 'like_count': int_or_none(video.get('likes')),
244 'dislike_count': int_or_none(video.get('dislikes')),
245 'age_limit': age_limit,
246 'tags': try_get(video, lambda x: x['tags'], list),
247 'categories': categories,