9 changed files with 206 additions and 12 deletions
Split View
Diff Options
-
6devscripts/buildserver.py
-
112devscripts/create-github-release.py
-
7devscripts/release.sh
-
5youtube_dl/compat.py
-
10youtube_dl/downloader/hls.py
-
1youtube_dl/extractor/extractors.py
-
2youtube_dl/extractor/twitch.py
-
73youtube_dl/extractor/vidio.py
-
2youtube_dl/extractor/youtube.py
@ -0,0 +1,112 @@ |
|||
#!/usr/bin/env python |
|||
from __future__ import unicode_literals |
|||
|
|||
import base64 |
|||
import json |
|||
import mimetypes |
|||
import netrc |
|||
import optparse |
|||
import os |
|||
import sys |
|||
|
|||
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) |
|||
|
|||
from youtube_dl.compat import ( |
|||
compat_basestring, |
|||
compat_input, |
|||
compat_getpass, |
|||
compat_print, |
|||
compat_urllib_request, |
|||
) |
|||
from youtube_dl.utils import ( |
|||
make_HTTPS_handler, |
|||
sanitized_Request, |
|||
) |
|||
|
|||
|
|||
class GitHubReleaser(object): |
|||
_API_URL = 'https://api.github.com/repos/rg3/youtube-dl/releases' |
|||
_UPLOADS_URL = 'https://uploads.github.com/repos/rg3/youtube-dl/releases/%s/assets?name=%s' |
|||
_NETRC_MACHINE = 'github.com' |
|||
|
|||
def __init__(self, debuglevel=0): |
|||
self._init_github_account() |
|||
https_handler = make_HTTPS_handler({}, debuglevel=debuglevel) |
|||
self._opener = compat_urllib_request.build_opener(https_handler) |
|||
|
|||
def _init_github_account(self): |
|||
try: |
|||
info = netrc.netrc().authenticators(self._NETRC_MACHINE) |
|||
if info is not None: |
|||
self._username = info[0] |
|||
self._password = info[2] |
|||
compat_print('Using GitHub credentials found in .netrc...') |
|||
return |
|||
else: |
|||
compat_print('No GitHub credentials found in .netrc') |
|||
except (IOError, netrc.NetrcParseError): |
|||
compat_print('Unable to parse .netrc') |
|||
self._username = compat_input( |
|||
'Type your GitHub username or email address and press [Return]: ') |
|||
self._password = compat_getpass( |
|||
'Type your GitHub password and press [Return]: ') |
|||
|
|||
def _call(self, req): |
|||
if isinstance(req, compat_basestring): |
|||
req = sanitized_Request(req) |
|||
# Authorizing manually since GitHub does not response with 401 with |
|||
# WWW-Authenticate header set (see |
|||
# https://developer.github.com/v3/#basic-authentication) |
|||
b64 = base64.b64encode( |
|||
('%s:%s' % (self._username, self._password)).encode('utf-8')).decode('ascii') |
|||
req.add_header('Authorization', 'Basic %s' % b64) |
|||
response = self._opener.open(req).read().decode('utf-8') |
|||
return json.loads(response) |
|||
|
|||
def list_releases(self): |
|||
return self._call(self._API_URL) |
|||
|
|||
def create_release(self, tag_name, name=None, body='', draft=False, prerelease=False): |
|||
data = { |
|||
'tag_name': tag_name, |
|||
'target_commitish': 'master', |
|||
'name': name, |
|||
'body': body, |
|||
'draft': draft, |
|||
'prerelease': prerelease, |
|||
} |
|||
req = sanitized_Request(self._API_URL, json.dumps(data).encode('utf-8')) |
|||
return self._call(req) |
|||
|
|||
def create_asset(self, release_id, asset): |
|||
asset_name = os.path.basename(asset) |
|||
url = self._UPLOADS_URL % (release_id, asset_name) |
|||
# Our files are small enough to be loaded directly into memory. |
|||
data = open(asset, 'rb').read() |
|||
req = sanitized_Request(url, data) |
|||
mime_type, _ = mimetypes.guess_type(asset_name) |
|||
req.add_header('Content-Type', mime_type or 'application/octet-stream') |
|||
return self._call(req) |
|||
|
|||
|
|||
def main(): |
|||
parser = optparse.OptionParser(usage='%prog VERSION BUILDPATH') |
|||
options, args = parser.parse_args() |
|||
if len(args) != 2: |
|||
parser.error('Expected a version and a build directory') |
|||
|
|||
version, build_path = args |
|||
|
|||
releaser = GitHubReleaser(debuglevel=0) |
|||
|
|||
new_release = releaser.create_release( |
|||
version, name='youtube-dl %s' % version, draft=True, prerelease=True) |
|||
release_id = new_release['id'] |
|||
|
|||
for asset in os.listdir(build_path): |
|||
compat_print('Uploading %s...' % asset) |
|||
releaser.create_asset(release_id, os.path.join(build_path, asset)) |
|||
|
|||
|
|||
if __name__ == '__main__': |
|||
main() |
@ -0,0 +1,73 @@ |
|||
# coding: utf-8 |
|||
from __future__ import unicode_literals |
|||
|
|||
import re |
|||
|
|||
from .common import InfoExtractor |
|||
from ..utils import int_or_none |
|||
|
|||
|
|||
class VidioIE(InfoExtractor): |
|||
_VALID_URL = r'https?://(?:www\.)?vidio\.com/watch/(?P<id>\d+)-(?P<display_id>[^/?#&]+)' |
|||
_TESTS = [{ |
|||
'url': 'http://www.vidio.com/watch/165683-dj_ambred-booyah-live-2015', |
|||
'md5': 'cd2801394afc164e9775db6a140b91fe', |
|||
'info_dict': { |
|||
'id': '165683', |
|||
'display_id': 'dj_ambred-booyah-live-2015', |
|||
'ext': 'mp4', |
|||
'title': 'DJ_AMBRED - Booyah (Live 2015)', |
|||
'description': 'md5:27dc15f819b6a78a626490881adbadf8', |
|||
'thumbnail': 're:^https?://.*\.jpg$', |
|||
'duration': 149, |
|||
'like_count': int, |
|||
}, |
|||
}, { |
|||
'url': 'https://www.vidio.com/watch/77949-south-korea-test-fires-missile-that-can-strike-all-of-the-north', |
|||
'only_matching': True, |
|||
}] |
|||
|
|||
def _real_extract(self, url): |
|||
mobj = re.match(self._VALID_URL, url) |
|||
video_id, display_id = mobj.group('id', 'display_id') |
|||
|
|||
webpage = self._download_webpage(url, display_id) |
|||
|
|||
title = self._og_search_title(webpage) |
|||
|
|||
m3u8_url, duration, thumbnail = [None] * 3 |
|||
|
|||
clips = self._parse_json( |
|||
self._html_search_regex( |
|||
r'data-json-clips\s*=\s*(["\'])(?P<data>\[.+?\])\1', |
|||
webpage, 'video data', default='[]', group='data'), |
|||
display_id, fatal=False) |
|||
if clips: |
|||
clip = clips[0] |
|||
m3u8_url = clip.get('sources', [{}])[0].get('file') |
|||
duration = clip.get('clip_duration') |
|||
thumbnail = clip.get('image') |
|||
|
|||
m3u8_url = m3u8_url or self._search_regex( |
|||
r'data(?:-vjs)?-clip-hls-url=(["\'])(?P<url>.+?)\1', webpage, 'hls url') |
|||
formats = self._extract_m3u8_formats(m3u8_url, display_id, 'mp4', entry_protocol='m3u8_native') |
|||
|
|||
duration = int_or_none(duration or self._search_regex( |
|||
r'data-video-duration=(["\'])(?P<duartion>\d+)\1', webpage, 'duration')) |
|||
thumbnail = thumbnail or self._og_search_thumbnail(webpage) |
|||
|
|||
like_count = int_or_none(self._search_regex( |
|||
(r'<span[^>]+data-comment-vote-count=["\'](\d+)', |
|||
r'<span[^>]+class=["\'].*?\blike(?:__|-)count\b.*?["\'][^>]*>\s*(\d+)'), |
|||
webpage, 'like count', fatal=False)) |
|||
|
|||
return { |
|||
'id': video_id, |
|||
'display_id': display_id, |
|||
'title': title, |
|||
'description': self._og_search_description(webpage), |
|||
'thumbnail': thumbnail, |
|||
'duration': duration, |
|||
'like_count': like_count, |
|||
'formats': formats, |
|||
} |
Write
Preview
Loading…
Cancel
Save