1
0
Fork 0
mirror of https://github.com/yt-dlp/yt-dlp synced 2025-01-18 23:03:05 +01:00

[extractor/camsoda] Add extractor (#5465)

Authored by: zulaport
This commit is contained in:
zulaport 2022-11-08 19:23:24 -08:00 committed by GitHub
parent fad689c7b6
commit 8fddc232bf
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 60 additions and 0 deletions

View file

@ -255,6 +255,7 @@ from .camdemy import (
CamdemyFolderIE CamdemyFolderIE
) )
from .cammodels import CamModelsIE from .cammodels import CamModelsIE
from .camsoda import CamsodaIE
from .camtasia import CamtasiaEmbedIE from .camtasia import CamtasiaEmbedIE
from .camwithher import CamWithHerIE from .camwithher import CamWithHerIE
from .canalalpha import CanalAlphaIE from .canalalpha import CanalAlphaIE

View file

@ -0,0 +1,59 @@
import random
from .common import InfoExtractor
from ..utils import ExtractorError, traverse_obj
class CamsodaIE(InfoExtractor):
_VALID_URL = r'https?://www\.camsoda\.com/(?P<id>[\w-]+)'
_TESTS = [{
'url': 'https://www.camsoda.com/lizzhopf',
'info_dict': {
'id': 'lizzhopf',
'ext': 'mp4',
'title': 'lizzhopf (lizzhopf) Nude on Cam. Free Live Sex Chat Room - CamSoda',
'description': str,
'is_live': True,
'age_limit': 18,
},
'skip': 'Room is offline',
}]
def _real_extract(self, url):
video_id = self._match_id(url)
webpage = self._download_webpage(url, video_id, headers=self.geo_verification_headers())
data = self._download_json(
f'https://camsoda.com/api/v1/video/vtoken/{video_id}', video_id,
query={'username': f'guest_{random.randrange(10000, 99999)}'},
headers=self.geo_verification_headers())
if not data:
raise ExtractorError('Unable to find configuration for stream.')
elif data.get('private_servers'):
raise ExtractorError('Model is in private show.', expected=True)
elif not data.get('stream_name'):
raise ExtractorError('Model is offline.', expected=True)
stream_name = traverse_obj(data, 'stream_name', expected_type=str)
token = traverse_obj(data, 'token', expected_type=str)
formats = []
for server in traverse_obj(data, ('edge_servers', ...)):
formats = self._extract_m3u8_formats(
f'https://{server}/{stream_name}_v1/index.m3u8?token={token}',
video_id, ext='mp4', m3u8_id='hls', fatal=False, live=True)
if formats:
break
if not formats:
self.raise_no_formats('No active streams found', expected=True)
self._sort_formats(formats)
return {
'id': video_id,
'title': self._html_extract_title(webpage),
'description': self._html_search_meta('description', webpage, default=None),
'is_live': True,
'formats': formats,
'age_limit': 18,
}