mirror of
				https://github.com/yt-dlp/yt-dlp.git
				synced 2025-11-04 08:35:12 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			162 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import functools
 | 
						|
import json
 | 
						|
 | 
						|
from .common import InfoExtractor
 | 
						|
from ..utils import (
 | 
						|
    ExtractorError,
 | 
						|
    OnDemandPagedList,
 | 
						|
    clean_html,
 | 
						|
    extract_attributes,
 | 
						|
    get_element_by_class,
 | 
						|
    get_element_html_by_id,
 | 
						|
    parse_count,
 | 
						|
    remove_end,
 | 
						|
    update_url,
 | 
						|
    urlencode_postdata,
 | 
						|
)
 | 
						|
 | 
						|
 | 
						|
class MurrtubeIE(InfoExtractor):
 | 
						|
    _VALID_URL = r'''(?x)
 | 
						|
                        (?:
 | 
						|
                            murrtube:|
 | 
						|
                            https?://murrtube\.net/(?:v/|videos/(?P<slug>[a-z0-9-]+?)-)
 | 
						|
                        )
 | 
						|
                        (?P<id>[A-Z0-9]{4}|[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12})
 | 
						|
                    '''
 | 
						|
    _TESTS = [{
 | 
						|
        'url': 'https://murrtube.net/videos/inferno-x-skyler-148b6f2a-fdcc-4902-affe-9c0f41aaaca0',
 | 
						|
        'md5': '70380878a77e8565d4aea7f68b8bbb35',
 | 
						|
        'info_dict': {
 | 
						|
            'id': 'ca885d8456b95de529b6723b158032e11115d',
 | 
						|
            'ext': 'mp4',
 | 
						|
            'title': 'Inferno X Skyler',
 | 
						|
            'description': 'Humping a very good slutty sheppy (roomate)',
 | 
						|
            'uploader': 'Inferno Wolf',
 | 
						|
            'age_limit': 18,
 | 
						|
            'thumbnail': 'https://storage.murrtube.net/murrtube-production/ekbs3zcfvuynnqfx72nn2tkokvsd',
 | 
						|
            'comment_count': int,
 | 
						|
            'view_count': int,
 | 
						|
            'like_count': int,
 | 
						|
        },
 | 
						|
    }, {
 | 
						|
        'url': 'https://murrtube.net/v/0J2Q',
 | 
						|
        'md5': '31262f6ac56f0ca75e5a54a0f3fefcb6',
 | 
						|
        'info_dict': {
 | 
						|
            'id': '8442998c52134968d9caa36e473e1a6bac6ca',
 | 
						|
            'ext': 'mp4',
 | 
						|
            'uploader': 'Hayel',
 | 
						|
            'title': 'Who\'s in charge now?',
 | 
						|
            'description': 'md5:795791e97e5b0f1805ea84573f02a997',
 | 
						|
            'age_limit': 18,
 | 
						|
            'thumbnail': 'https://storage.murrtube.net/murrtube-production/fb1ojjwiucufp34ya6hxu5vfqi5s',
 | 
						|
            'comment_count': int,
 | 
						|
            'view_count': int,
 | 
						|
            'like_count': int,
 | 
						|
        },
 | 
						|
    }]
 | 
						|
 | 
						|
    def _extract_count(self, name, html):
 | 
						|
        return parse_count(self._search_regex(
 | 
						|
            rf'([\d,]+)\s+<span[^>]*>{name}</span>', html, name, default=None))
 | 
						|
 | 
						|
    def _real_initialize(self):
 | 
						|
        homepage = self._download_webpage(
 | 
						|
            'https://murrtube.net', None, note='Getting session token')
 | 
						|
        self._request_webpage(
 | 
						|
            'https://murrtube.net/accept_age_check', None, 'Setting age cookie',
 | 
						|
            data=urlencode_postdata(self._hidden_inputs(homepage)))
 | 
						|
 | 
						|
    def _real_extract(self, url):
 | 
						|
        video_id = self._match_id(url)
 | 
						|
        if video_id.startswith('murrtube:'):
 | 
						|
            raise ExtractorError('Support for murrtube: prefix URLs is broken')
 | 
						|
        video_page = self._download_webpage(url, video_id)
 | 
						|
        video_attrs = extract_attributes(get_element_html_by_id('video', video_page))
 | 
						|
        playlist = update_url(video_attrs['data-url'], query=None)
 | 
						|
        video_id = self._search_regex(r'/([\da-f]+)/index.m3u8', playlist, 'video id')
 | 
						|
 | 
						|
        return {
 | 
						|
            'id': video_id,
 | 
						|
            'title': remove_end(self._og_search_title(video_page), ' - Murrtube'),
 | 
						|
            'age_limit': 18,
 | 
						|
            'formats': self._extract_m3u8_formats(playlist, video_id, 'mp4'),
 | 
						|
            'description': self._og_search_description(video_page),
 | 
						|
            'thumbnail': update_url(self._og_search_thumbnail(video_page, default=''), query=None) or None,
 | 
						|
            'uploader': clean_html(get_element_by_class('pl-1 is-size-6 has-text-lighter', video_page)),
 | 
						|
            'view_count': self._extract_count('Views', video_page),
 | 
						|
            'like_count': self._extract_count('Likes', video_page),
 | 
						|
            'comment_count': self._extract_count('Comments', video_page),
 | 
						|
        }
 | 
						|
 | 
						|
 | 
						|
class MurrtubeUserIE(InfoExtractor):
 | 
						|
    _WORKING = False
 | 
						|
    IE_DESC = 'Murrtube user profile'
 | 
						|
    _VALID_URL = r'https?://murrtube\.net/(?P<id>[^/]+)$'
 | 
						|
    _TESTS = [{
 | 
						|
        'url': 'https://murrtube.net/stormy',
 | 
						|
        'info_dict': {
 | 
						|
            'id': 'stormy',
 | 
						|
        },
 | 
						|
        'playlist_mincount': 27,
 | 
						|
    }]
 | 
						|
    _PAGE_SIZE = 10
 | 
						|
 | 
						|
    def _download_gql(self, video_id, op, note=None, fatal=True):
 | 
						|
        result = self._download_json(
 | 
						|
            'https://murrtube.net/graphql',
 | 
						|
            video_id, note, data=json.dumps(op).encode(), fatal=fatal,
 | 
						|
            headers={'Content-Type': 'application/json'})
 | 
						|
        return result['data']
 | 
						|
 | 
						|
    def _fetch_page(self, username, user_id, page):
 | 
						|
        data = self._download_gql(username, {
 | 
						|
            'operationName': 'Media',
 | 
						|
            'variables': {
 | 
						|
                'limit': self._PAGE_SIZE,
 | 
						|
                'offset': page * self._PAGE_SIZE,
 | 
						|
                'sort': 'latest',
 | 
						|
                'userId': user_id,
 | 
						|
            },
 | 
						|
            'query': '''\
 | 
						|
query Media($q: String, $sort: String, $userId: ID, $offset: Int!, $limit: Int!) {
 | 
						|
  media(q: $q, sort: $sort, userId: $userId, offset: $offset, limit: $limit) {
 | 
						|
    id
 | 
						|
    __typename
 | 
						|
  }
 | 
						|
}'''},
 | 
						|
            f'Downloading page {page + 1}')
 | 
						|
        if data is None:
 | 
						|
            raise ExtractorError(f'Failed to retrieve video list for page {page + 1}')
 | 
						|
 | 
						|
        media = data['media']
 | 
						|
 | 
						|
        for entry in media:
 | 
						|
            yield self.url_result('murrtube:{}'.format(entry['id']), MurrtubeIE.ie_key())
 | 
						|
 | 
						|
    def _real_extract(self, url):
 | 
						|
        username = self._match_id(url)
 | 
						|
        data = self._download_gql(username, {
 | 
						|
            'operationName': 'User',
 | 
						|
            'variables': {
 | 
						|
                'id': username,
 | 
						|
            },
 | 
						|
            'query': '''\
 | 
						|
query User($id: ID!) {
 | 
						|
  user(id: $id) {
 | 
						|
    id
 | 
						|
    __typename
 | 
						|
  }
 | 
						|
}'''},
 | 
						|
            'Downloading user info')
 | 
						|
        if data is None:
 | 
						|
            raise ExtractorError('Failed to fetch user info')
 | 
						|
 | 
						|
        user = data['user']
 | 
						|
 | 
						|
        entries = OnDemandPagedList(functools.partial(
 | 
						|
            self._fetch_page, username, user.get('id')), self._PAGE_SIZE)
 | 
						|
 | 
						|
        return self.playlist_result(entries, username)
 |