You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
plugin.video.rumble/main.py

693 lines
22 KiB
Python

# -*- coding: utf-8 -*-
import sys, re, os, requests
2 years ago
import xbmc, xbmcplugin, xbmcgui, xbmcaddon, xbmcvfs
import six
from six.moves import urllib
2 years ago
from resources.md5ex import *
# Disable urllib3's "InsecureRequestWarning: Unverified HTTPS request is being made" warnings
from requests.packages.urllib3.exceptions import InsecureRequestWarning
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
try:
import cookielib
except ImportError:
import http.cookiejar as cookielib
2 years ago
2 years ago
try:
import json
except ImportError:
2 years ago
import simplejson as json
BASE_URL = 'https://rumble.com'
PLUGIN_URL = sys.argv[0]
PLUGIN_ID = int(sys.argv[1])
PLUGIN_NAME = PLUGIN_URL.replace("plugin://","")
ADDON = xbmcaddon.Addon()
ADDON_ICON = ADDON.getAddonInfo('icon')
ADDON_NAME = ADDON.getAddonInfo('name')
HOME_DIR = 'special://home/addons/{0}'.format(PLUGIN_NAME)
2 years ago
RESOURCE_DIR = HOME_DIR + 'resources/'
MEDIA_DIR = RESOURCE_DIR + 'media/'
2 years ago
#language
__language__ = ADDON.getLocalizedString
lang = ADDON.getSetting('lang')
r_username = ADDON.getSetting('username')
r_password = ADDON.getSetting('password')
s = requests.session()
2 years ago
if six.PY2:
favorites = xbmc.translatePath(os.path.join(ADDON.getAddonInfo('profile'), 'favorites.dat'))
2 years ago
else:
favorites = xbmcvfs.translatePath(os.path.join(ADDON.getAddonInfo('profile'), 'favorites.dat'))
def createFavorites():
addonID = ADDON.getAddonInfo('id')
if six.PY2:
addon_data_path = xbmc.translatePath(ADDON.getAddonInfo('profile'))
else:
addon_data_path = xbmcvfs.translatePath(ADDON.getAddonInfo('profile'))
if os.path.exists(addon_data_path)==False:
os.mkdir(addon_data_path)
xbmc.sleep(1)
def loadFavorites( return_string = False ):
if os.path.exists(favorites):
fav_str = open(favorites).read()
if return_string:
return fav_str
if fav_str:
return json.loads( fav_str )
else:
createFavorites()
if return_string:
return ''
else:
return []
2 years ago
def notify(message,name=False,iconimage=False,timeShown=5000):
if not name:
name = ADDON_NAME
if not iconimage:
iconimage = ADDON_ICON
xbmc.executebuiltin('Notification(%s, %s, %d, %s)' % (name, message, timeShown, iconimage))
2 years ago
def to_unicode(text, encoding='utf-8', errors='strict'):
# Force text to unicode
2 years ago
if isinstance(text, bytes):
return text.decode(encoding, errors=errors)
return text
def get_search_string(heading='', message=''):
# Ask the user for a search string
2 years ago
search_string = None
keyboard = xbmc.Keyboard(message, heading)
keyboard.doModal()
if keyboard.isConfirmed():
search_string = to_unicode(keyboard.getText())
return search_string
def getRequest(url, data=None, extraHeaders=None):
2 years ago
try:
myHeaders = {
'Accept-Language': 'en-gb,en;q=0.5',
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/108.0.0.0 Safari/537.36',
'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
'Referer': url,
'Cache-Control': 'no-cache',
'Pragma': 'no-cache',
'DNT': '1'
}
if extraHeaders:
myHeaders.update(extraHeaders)
cookieDict = cookielib.CookieJar()
if data:
response = s.post(url, data=data, headers=myHeaders, verify=False, cookies=None, timeout=10)
else:
response = s.get(url, headers=myHeaders, verify=False, cookies=None, timeout=10)
2 years ago
except:
response = ''
return response.text
2 years ago
# main menu
def home_menu():
# Search
addDir( xbmc.getLocalizedString(137), '', 1, MEDIA_DIR + 'search.png', '', '' ,'' )
2 years ago
# Favorites
addDir( xbmc.getLocalizedString(1036), '', 7, MEDIA_DIR + 'favorite.png', '', '', '' )
if r_username and r_password:
# subscriptions
addDir( 'Subscriptions', BASE_URL + '/subscriptions', 3, MEDIA_DIR + 'favorite.png', '', '', 'other' )
2 years ago
# News
addDir( xbmc.getLocalizedString(29916), BASE_URL + '/category/news', 3, MEDIA_DIR + 'news.png', '', '', 'other' )
2 years ago
# Viral
addDir( __language__(30050), BASE_URL + '/category/viral', 3, MEDIA_DIR + 'viral.png', '', '', 'other' )
2 years ago
# Podcasts
addDir( __language__(30051), BASE_URL + '/category/podcasts', 3, MEDIA_DIR +'podcast.png','','','other')
2 years ago
# Battle Leaderboard
addDir( __language__(30052), BASE_URL + '/battle-leaderboard', 3, MEDIA_DIR + 'leader.png', '', '', 'top' )
2 years ago
# Entertainment
addDir( __language__(30053), BASE_URL + '/category/entertainment', 3, MEDIA_DIR + 'entertaiment.png', '', '', 'other' )
2 years ago
# Sports
addDir( xbmc.getLocalizedString(19548), BASE_URL + '/category/sports', 3, MEDIA_DIR + 'sports.png', '', '', 'other' )
2 years ago
# Science
addDir( xbmc.getLocalizedString(29948), BASE_URL + '/category/science', 3, MEDIA_DIR + 'science.png', '', '', 'other' )
2 years ago
# Technology
addDir( __language__(30054), BASE_URL + '/category/technology', 3, MEDIA_DIR + 'technology.png', '', '', 'other' )
2 years ago
# Vlogs
addDir( __language__(30055), BASE_URL + '/category/vlogs', 3, MEDIA_DIR + 'vlog.png', '', '', 'other' )
2 years ago
# Settings
addDir( xbmc.getLocalizedString(5), '', 8, MEDIA_DIR + 'settings.png', '', '', '' )
2 years ago
SetView('WideList')
xbmcplugin.endOfDirectory(PLUGIN_ID, cacheToDisc=False)
# search menu
def search_menu():
# Search Video
addDir( __language__(30100), BASE_URL + '/search/video?q=', 2, MEDIA_DIR + 'search.png', '', '', 'video' )
2 years ago
# Search Channel
addDir( __language__(30101), BASE_URL + '/search/channel?q=',2,MEDIA_DIR + 'search.png', '', '', 'channel' )
# Search User
addDir( __language__(30102), BASE_URL + '/search/channel?q=',2,MEDIA_DIR + 'search.png', '', '', 'user' )
2 years ago
SetView('WideList')
xbmcplugin.endOfDirectory(PLUGIN_ID)
def pagination(url,page,cat,search=False):
2 years ago
if url > '':
page = int(page)
pageUrl = url
if page == 1:
2 years ago
if search:
pageUrl = url + search
elif search and cat == 'video':
pageUrl = url + search + "&page=" + str( page )
elif cat in {'channel', 'user', 'other' }:
pageUrl = url + "?page=" + str( page )
amount = list_rumble( pageUrl, cat )
if amount > 15 and page < 10:
# for next page
page = page + 1
name = __language__(30150) + " " + str( page )
2 years ago
li=xbmcgui.ListItem(name)
linkParams = {
'url': url,
'mode': '3',
'name': name,
'page': str( page ),
'cat': cat,
}
link = buildURL( linkParams )
if search and cat == 'video':
link = link + "&search=" + urllib.parse.quote_plus(search)
xbmcplugin.addDirectoryItem(PLUGIN_ID, link, li, True)
2 years ago
SetView('WideList')
xbmcplugin.endOfDirectory(PLUGIN_ID)
def get_image(data,id):
image_re = re.compile("i.user-image--img--id-"+str(id)+".+?{ background-image: url(.+?);", re.MULTILINE|re.DOTALL|re.IGNORECASE).findall(data)
if image_re !=[]:
image = str(image_re[0]).replace('(', '').replace(')', '')
else:
image = ''
return image
def list_rumble(url, cat):
amount = 0
headers = None
if 'subscriptions' in url:
if not ADDON.getSetting('session'):
login()
headers = { 'cookie': 'u_s=' + ADDON.getSetting('session')}
data = getRequest(url, None, headers)
if 'search' in url:
if cat == 'video':
amount = create_dir_list( data, cat, 'video', True, 1 )
else:
amount = create_dir_list( data, cat, 'channel', True )
elif cat in { 'channel', 'user', 'top', 'other' }:
amount = create_dir_list( data, cat, 'video', False, 2 )
return amount
def create_dir_list( data, cat, type='video', search = False, play=False ):
amount = 0
if type == 'video':
videos = re.compile('a href=([^\>]+)><div class=\"(?:[^\"]+)\"><img class=\"video-item--img\" src=(https:\/\/.+?) alt=(?:[^\>]+)>(?:<span class=\"video-item--watching\">[^\<]+</span>)?(?:<div class=video-item--overlay-rank>(?:[0-9]+)</div>)?</div><(?:[^\>]+)></span></a><div class=\"video-item--info\"><time class=\"video-item--meta video-item--time\" datetime=(.+?)-(.+?)-(.+?)T(?:.+?) title\=\"(?:[^\"]+)\">(?:[^\<]+)</time><h3 class=video-item--title>(.+?)</h3><address(?:[^\>]+)><a rel=author class=\"(?:[^\=]+)=(.+?)><div class=ellipsis-1>(.+?)</div>', re.MULTILINE|re.DOTALL|re.IGNORECASE).findall(data)
if videos:
amount = len(videos)
for link, img, year, month, day, title, channel_link, channel_name in videos:
if '<svg' in channel_name:
channel_name = channel_name.split('<svg')[0] + " (Verified)"
2 years ago
if int(lang) == 0:
video_date = month+'/'+day+'/'+year
2 years ago
else:
video_date = day+'/'+month+'/'+year
video_title = '[B]' + title + '[/B]\n[COLOR gold]' + channel_name + ' - [COLOR lime]' + video_date + '[/COLOR]'
#open get url and open player
addDir( video_title, BASE_URL + link, 4, str(img), str(img), '', cat, False, True, play )
else:
channels = re.compile('a href=(.+?)>\s*<div class=\"channel-item--img\">\s*<i class=\'user-image user-image--img user-image--img--id-(.+?)\'></i>\s*</div>\s*<h3 class=channel-item--title>(.+?)</h3>\s*<span class=channel-item--subscribers>(.+?) subscribers</span>',re.DOTALL).findall(data)
if channels:
amount = len(channels)
for link, img_id, channel_name, subscribers in channels:
# split channel and user
if search:
if cat == 'channel':
if '/c/' not in link:
continue
else:
if '/user/' not in link:
continue
if '<svg' in channel_name:
channel_name = channel_name.split('<svg')[0] + " (Verified)"
img = str( get_image( data, img_id ) )
video_title = '[B]' + channel_name + '[/B]\n[COLOR palegreen]' + subscribers + ' [COLOR yellow]' + __language__(30155) + '[/COLOR]'
#open get url and open player
addDir( video_title, BASE_URL + link, 3, img, img, '', cat, True, True, play )
return amount
2 years ago
def resolver(url):
# playback options - 0: large to small, 1: small to large, 2: quality select
2 years ago
playbackMethod = ADDON.getSetting('playbackMethod')
mediaURL = False
2 years ago
if playbackMethod == '2':
urls = []
data = getRequest(url)
# gets embed id from embed url
embed_id = re.compile(',\"embedUrl\":\"' + BASE_URL + '/embed/(.*?)\/\",', re.MULTILINE|re.DOTALL|re.IGNORECASE).findall(data)
if embed_id:
# use site api to get video urls
data = getRequest(BASE_URL + '/embedJS/u3/?request=video&ver=2&v=' + embed_id[0])
2 years ago
sizes = [ '1080', '720', '480', '360', 'hls' ]
2 years ago
# reverses array - small to large
2 years ago
if playbackMethod == '1':
sizes = sizes[::-1]
for quality in sizes:
2 years ago
matches = re.compile( '"' + quality + '".+?url.+?:"(.*?)"', re.MULTILINE|re.DOTALL|re.IGNORECASE).findall(data)
2 years ago
if matches:
if playbackMethod == '2':
urls.append(( quality, matches[0] ))
else:
mediaURL = matches[0]
break
# quality select
2 years ago
if playbackMethod == '2':
if len(urls) > 0:
selectedIndex = xbmcgui.Dialog().select(
'Select Quality', [(sourceItem[0] or '?') for sourceItem in urls]
)
if selectedIndex != -1:
mediaURL = urls[selectedIndex][1]
if mediaURL:
mediaURL = mediaURL.replace('\/', '/')
return mediaURL
2 years ago
def play_video(name, url, iconimage, play=2):
2 years ago
url = resolver(url)
if url:
li = xbmcgui.ListItem(name, path=url)
li.setArt({"icon": iconimage, "thumb": iconimage})
li.setInfo(type='video', infoLabels={'Title': name, 'plot': ''})
if play == 1:
xbmc.Player().play(item=url, listitem=li)
elif play == 2:
xbmcplugin.setResolvedUrl(int(sys.argv[1]), True, li)
else:
xbmcgui.Dialog().ok( 'Error', 'Video not found' )
2 years ago
def search_items(url,cat):
vq = get_search_string(heading="Search")
if ( not vq ): return False, 0
title = urllib.parse.quote_plus(vq)
2 years ago
pagination(url,1,cat,title)
def getFavorites():
2 years ago
data = loadFavorites()
2 years ago
2 years ago
try:
amount = len(data)
if amount > 0:
for i in data:
2 years ago
name = i[0]
url = i[1]
mode = i[2]
iconimage = i[3]
fanArt = i[4]
description = i[5]
cat = i[6]
folder = ( i[7] == 'True' )
2 years ago
play = i[8]
addDir( name, url, mode, str(iconimage), str(fanArt), str(description), cat, folder, True, int(play) )
2 years ago
SetView('WideList')
xbmcplugin.endOfDirectory(PLUGIN_ID)
else:
xbmcgui.Dialog().ok( xbmc.getLocalizedString(14117), __language__(30155) )
2 years ago
except:
SetView('WideList')
xbmcplugin.endOfDirectory(PLUGIN_ID)
def addFavorite(name, url, fav_mode, iconimage, fanart, description, cat, folder, play):
data = loadFavorites()
data.append((name, url, fav_mode, iconimage, fanart, description, cat, folder, play))
b = open(favorites, 'w')
b.write(json.dumps(data))
b.close()
notify( __language__(30152), name, iconimage )
2 years ago
def rmFavorite(name):
data = loadFavorites()
2 years ago
for index in range(len(data)):
if data[index][0]==name:
del data[index]
b = open(favorites, 'w')
2 years ago
b.write(json.dumps(data))
b.close()
break
notify( __language__(30154), name )
2 years ago
def importFavorites():
if not xbmcgui.Dialog().yesno(
'Import Favorites',
'This will replace the favorites with the plugin.video.rumble.matrix version.\nProceed?',
nolabel = 'Cancel',
yeslabel = 'Ok'
):
return
# no point trying to run this as it didn't exist for python 2
if six.PY2:
return
# make sure path exists
createFavorites()
#load matrix favourites
rumble_matrix_dir = xbmcvfs.translatePath(os.path.join('special://home/userdata/addon_data/plugin.video.rumble.matrix', 'favorites.dat'))
if os.path.exists(rumble_matrix_dir):
rumble_matrix = open(rumble_matrix_dir).read()
if rumble_matrix:
b = open(favorites, 'w')
b.write(rumble_matrix)
b.close()
notify( 'Imported Favorites' )
return
notify( 'Favorites Not Found' )
def login():
login_hash = MD5Ex()
# gets salts
data = getRequest( BASE_URL + '/service.php?name=user.get_salts', {'username': r_username}, [ ( 'Referer', BASE_URL ), ( 'Content-type', 'application/x-www-form-urlencoded' ) ] )
salt = json.loads(data)['data']['salts']
# generate hashes
hashes = login_hash.hash(login_hash.hashStretch(r_password, salt[0], 128) + salt[1]) + ',' + login_hash.hashStretch(r_password, salt[2], 128) + ',' + salt[1]
# login
data = getRequest( BASE_URL + '/service.php?name=user.login', {'username': r_username, 'password_hashes':hashes}, [ ( 'Referer', BASE_URL ), ( 'Content-type', 'application/x-www-form-urlencoded' ) ] )
if data:
session = json.loads(data)['data']['session']
if session:
ADDON.setSetting('session', session)
def addDir(name, url, mode, iconimage, fanart, description, cat, folder=True, fav_context=False, play=False):
2 years ago
linkParams = {
'url': url,
'mode': str( mode ),
'name': name,
'fanart': fanart,
'iconimage': iconimage,
'description': description,
'cat': cat,
}
2 years ago
2 years ago
if play:
linkParams['play'] = str( play )
link = buildURL( linkParams )
li=xbmcgui.ListItem( name )
2 years ago
if folder:
li.setArt({'icon': 'DefaultFolder.png', 'thumb': iconimage})
else:
li.setArt({'icon': 'DefaultVideo.png', 'thumb': iconimage})
if play == 2 and mode == 4:
li.setProperty('IsPlayable', 'true')
li.setInfo(type='Video', infoLabels={'Title': name, 'Plot': description})
if fanart > '':
li.setProperty('fanart_image', fanart)
else:
li.setProperty('fanart_image', HOME_DIR + 'fanart.jpg')
if fav_context:
favorite_str = loadFavorites( True )
2 years ago
2 years ago
try:
2 years ago
name_fav = json.dumps(name)
2 years ago
except:
2 years ago
name_fav = name
2 years ago
try:
contextMenu = []
# checks name via string which I do not like
if name_fav in favorite_str:
contextMenu.append((__language__(30153),'RunPlugin(%s)' %buildURL( {'mode': '6','name': name} )))
2 years ago
else:
fav_params = {
'url': url,
'mode': '5',
'name': name,
'fanart': fanart,
'iconimage': iconimage,
'description': description,
'cat': cat,
'folder': str(folder),
'fav_mode': str(mode),
'play': str(play),
}
contextMenu.append((__language__(30151),'RunPlugin(%s)' %buildURL( fav_params )))
2 years ago
li.addContextMenuItems(contextMenu)
except:
pass
2 years ago
xbmcplugin.addDirectoryItem(handle=PLUGIN_ID, url=link, listitem=li, isFolder=folder)
2 years ago
def buildURL(query):
# Helper function to build a Kodi xbmcgui.ListItem URL.
# :param query: Dictionary of url parameters to put in the URL.
# :returns: A formatted and urlencoded URL string.
return (PLUGIN_URL + '?' + urllib.parse.urlencode({k: v.encode('utf-8') if isinstance(v, six.text_type)
else unicode(v, errors='ignore').encode('utf-8')
for k, v in query.items()}))
def SetView(name):
if name == 'Fanart':
view_num = 502
elif name == 'Wall':
view_num = 500
elif name == 'WideList':
view_num = 55
elif name == 'InfoWall':
view_num = 54
elif name == 'Shift':
view_num = 53
elif name == 'Poster':
view_num = 51
elif name == 'List':
view_num = 50
else:
view_num = 0
if view_num > 0:
try:
xbmc.executebuiltin('Container.SetViewMode(' + str( view_num ) + ')')
except:
pass
2 years ago
def get_params():
return dict(urllib.parse.parse_qsl(sys.argv[2][1:], keep_blank_values=True))
2 years ago
def main():
2 years ago
2 years ago
params=get_params()
try:
url=urllib.parse.unquote_plus(params["url"])
2 years ago
except:
2 years ago
url=None
2 years ago
try:
name=urllib.parse.unquote_plus(params["name"])
2 years ago
except:
2 years ago
name=None
2 years ago
try:
iconimage=urllib.parse.unquote_plus(params["iconimage"])
2 years ago
except:
2 years ago
iconimage=None
2 years ago
try:
mode=int(params["mode"])
except:
2 years ago
mode=None
2 years ago
try:
fanart=urllib.parse.unquote_plus(params["fanart"])
2 years ago
except:
2 years ago
fanart=None
2 years ago
try:
description=urllib.parse.unquote_plus(params["description"])
2 years ago
except:
2 years ago
description=None
2 years ago
try:
subtitle=urllib.parse.unquote_plus(params["subtitle"])
except:
2 years ago
subtitle=None
2 years ago
try:
cat=urllib.parse.unquote_plus(params["cat"])
2 years ago
except:
2 years ago
cat=None
2 years ago
try:
search=urllib.parse.unquote_plus(params["search"])
2 years ago
except:
2 years ago
search=None
2 years ago
try:
page=int(params["page"])
except:
2 years ago
page=1
2 years ago
try:
folder=urllib.parse.unquote_plus(params["folder"])
2 years ago
except:
2 years ago
folder=None
2 years ago
try:
fav_mode=int(params["fav_mode"])
except:
2 years ago
fav_mode=None
2 years ago
try:
play=int(params["play"])
except:
2 years ago
play=1
2 years ago
if mode==None:
home_menu()
elif mode==1:
search_menu()
elif mode==2:
search_items(url,cat)
elif mode==3:
if search and search !=None:
pagination(url, page, cat, search)
2 years ago
else:
pagination(url, page, cat)
2 years ago
elif mode==4:
play_video(name, url, iconimage, play)
elif mode==5:
try:
name = name.split('\\ ')[1]
except:
pass
try:
name = name.split(' - ')[0]
except:
pass
addFavorite( name, url, fav_mode, iconimage, fanart, description, cat, str(folder), str(play) )
2 years ago
elif mode==6:
try:
name = name.split('\\ ')[1]
except:
pass
try:
name = name.split(' - ')[0]
except:
pass
rmFavorite( name )
2 years ago
elif mode==7:
getFavorites()
elif mode==8:
ADDON.openSettings()
elif mode==9:
importFavorites()
2 years ago
if __name__ == "__main__":
main()