delete videoobrepl
This commit is contained in:
parent
f46eab4b67
commit
8ab772c3d7
2 changed files with 0 additions and 408 deletions
|
|
@ -1,22 +0,0 @@
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
# Copyright(C) 2010 John Obbele, based on previous work from solsTiCe
|
|
||||||
# d'Hiver # <solstice.dhiver@gmail.com>
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, version 3 of the License.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program; if not, write to the Free Software
|
|
||||||
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
||||||
|
|
||||||
|
|
||||||
from .videoobrepl import VideoobRepl
|
|
||||||
|
|
||||||
__all__ = ['VideoobRepl']
|
|
||||||
|
|
@ -1,386 +0,0 @@
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
# Copyright(C) 2010 John Obbele
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, version 3 of the License.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program; if not, write to the Free Software
|
|
||||||
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
||||||
|
|
||||||
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import errno
|
|
||||||
from cmd import Cmd
|
|
||||||
from subprocess import Popen, PIPE
|
|
||||||
from weboob.capabilities.video import ICapVideo
|
|
||||||
from weboob.tools.application.console import ConsoleApplication
|
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['VideoobRepl']
|
|
||||||
|
|
||||||
|
|
||||||
# EVIL GLOBAL VARIABLES {{{
|
|
||||||
# shell escape strings
|
|
||||||
BOLD = '[1m'
|
|
||||||
NC = '[0m' # no color
|
|
||||||
|
|
||||||
# A list of tuples: (player , play_from_stdin_cmd)
|
|
||||||
# FIXME: lookup preference in freedesktop MIME database
|
|
||||||
PLAYERS = [
|
|
||||||
('parole', 'parole fd://0'),
|
|
||||||
('totem', 'totem fd://0'),
|
|
||||||
('mplayer', 'mplayer -really-quiet -'),
|
|
||||||
('vlc', 'vlc -'),
|
|
||||||
('xine', 'xine stdin:/'),
|
|
||||||
]
|
|
||||||
# }}}
|
|
||||||
|
|
||||||
class DefaultOptions():
|
|
||||||
"""Dummy options object.
|
|
||||||
|
|
||||||
Should be replaced by a proper one from optparse.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.lang = "fr"
|
|
||||||
self.quality = "hd"
|
|
||||||
self.verbose = True
|
|
||||||
|
|
||||||
class MyPlayer():
|
|
||||||
"""Black magic invoking a video player to this world.
|
|
||||||
|
|
||||||
Presently, due to strong disturbances in the holidays of the ether
|
|
||||||
world, the video player used is chosen from a static list of
|
|
||||||
programs. See PLAYERS for more information.
|
|
||||||
|
|
||||||
You MAY want to move it into a separate weboob.tools.applications
|
|
||||||
module.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, options=DefaultOptions()):
|
|
||||||
"@param options [object] requires the bool. attribute 'verbose'"
|
|
||||||
self.options = options
|
|
||||||
|
|
||||||
self.player = None
|
|
||||||
for (binary,cmd_stdin) in PLAYERS:
|
|
||||||
if self._find_in_path(os.environ['PATH'], binary):
|
|
||||||
self.player = binary
|
|
||||||
self.player_stdin = cmd_stdin
|
|
||||||
break
|
|
||||||
if not self.player:
|
|
||||||
raise OSError(errno.ENOENT, "video player not found")
|
|
||||||
|
|
||||||
if self.options.verbose:
|
|
||||||
print "Video player is (%s,%s)" % (self.player,
|
|
||||||
self.player_stdin)
|
|
||||||
|
|
||||||
|
|
||||||
def play(self, video):
|
|
||||||
"""Play a video object, using programs from the PLAYERS list.
|
|
||||||
|
|
||||||
This function dispatch calls to either _play_default or
|
|
||||||
_play_rtmp for special rtmp streams using SWF verification.
|
|
||||||
"""
|
|
||||||
if video.url.find('rtmp') == 0:
|
|
||||||
self._play_rtmp(video)
|
|
||||||
else:
|
|
||||||
self._play_default(video)
|
|
||||||
|
|
||||||
def _play_default(self, video):
|
|
||||||
"Play video.url with the video player."
|
|
||||||
cmd = self.player + " " + video.url
|
|
||||||
args = cmd.split()
|
|
||||||
|
|
||||||
print "invoking [%s]" % cmd
|
|
||||||
os.spawnlp(os.P_NOWAIT, args[0], *args)
|
|
||||||
|
|
||||||
def _play_rtmp(self, video):
|
|
||||||
""""Download data with rtmpdump and pipe them to a video player.
|
|
||||||
|
|
||||||
You need a working version of rtmpdump installed and the SWF
|
|
||||||
object url in order to comply with SWF verification requests
|
|
||||||
from the server. The last one is retrieved from the non-standard
|
|
||||||
non-API compliant 'swf_player' attribute of the 'video' object.
|
|
||||||
"""
|
|
||||||
|
|
||||||
if not self._find_in_path(os.environ['PATH'], 'rtmpdump'):
|
|
||||||
raise OSError(errno.ENOENT, "\'rtmpdump\' binary not found")
|
|
||||||
|
|
||||||
video_url = video.url
|
|
||||||
try:
|
|
||||||
player_url = video.swf_player
|
|
||||||
rtmp = 'rtmpdump -r %s --swfVfy %s' % (video_url, player_url)
|
|
||||||
|
|
||||||
except AttributeError:
|
|
||||||
error("Your video object does not have a 'swf_player' "
|
|
||||||
"attribute. SWF verification will be disabled and "
|
|
||||||
"may prevent correct video playback.")
|
|
||||||
|
|
||||||
rtmp = 'rtmpdump -r %s' % video_url
|
|
||||||
|
|
||||||
if not self.options.verbose:
|
|
||||||
rtmp += ' --quiet'
|
|
||||||
|
|
||||||
print ':: Streaming from %s' % video_url
|
|
||||||
print ':: to %s' % self.player_stdin
|
|
||||||
p1 = Popen(rtmp.split(), stdout=PIPE)
|
|
||||||
p2 = Popen(self.player_stdin.split(),
|
|
||||||
stdin=p1.stdout, stderr=PIPE)
|
|
||||||
|
|
||||||
def _find_in_path(self,path, filename):
|
|
||||||
for i in path.split(':'):
|
|
||||||
if os.path.exists('/'.join([i, filename])):
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def error(str):
|
|
||||||
"Shortcut to print >>sys.stderr"
|
|
||||||
print >> sys.stderr, "Error:", str
|
|
||||||
|
|
||||||
def print_keys_values(tuplets, indent=0, highlight=False):
|
|
||||||
"""Pretty print a list of (key, values) tuplets."""
|
|
||||||
first_column_width = max(len(k) for (k,v) in tuplets)
|
|
||||||
for (key,value) in tuplets:
|
|
||||||
# calm down typography nitpickers
|
|
||||||
key = key + ":"
|
|
||||||
# assert first column width
|
|
||||||
key = key + " " * (first_column_width - len(key) + 1)
|
|
||||||
# add uniform indentation if needed
|
|
||||||
key = " " * indent + key
|
|
||||||
# call 911
|
|
||||||
if highlight:
|
|
||||||
key = BOLD + key + NC
|
|
||||||
print key, value
|
|
||||||
|
|
||||||
class MyCmd(Cmd):
|
|
||||||
"""Read-Eval-Print-Loop object build from the 'cmd' framework.
|
|
||||||
|
|
||||||
It's just a command dispatcher, so get done with it.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, consoleApplication, player):
|
|
||||||
"""
|
|
||||||
@param consoleApplication an instance of ConsoleApplication
|
|
||||||
@param player an instance of MyPlayer()
|
|
||||||
"""
|
|
||||||
Cmd.__init__(self)
|
|
||||||
self.prompt = BOLD + 'weboob> ' + NC
|
|
||||||
self.intro = 'Type "help" to see available commands.'
|
|
||||||
|
|
||||||
self.player = player
|
|
||||||
|
|
||||||
# engine / console application initialisation
|
|
||||||
# (loading ALL backends by default)
|
|
||||||
self.engine = consoleApplication
|
|
||||||
self.enabled_backends = []
|
|
||||||
self.available_backends = []
|
|
||||||
self.engine.load_backends(ICapVideo)
|
|
||||||
for b in self.engine.weboob.iter_backends(caps=ICapVideo):
|
|
||||||
self.enabled_backends.append(b)
|
|
||||||
self.available_backends.append(b)
|
|
||||||
|
|
||||||
self.videos = [] # videos list cache
|
|
||||||
|
|
||||||
def do_quit(self, arg):
|
|
||||||
"""quit the command line interpreter"""
|
|
||||||
print "Byebye !"
|
|
||||||
return True
|
|
||||||
|
|
||||||
def do_exit(self, arg):
|
|
||||||
"""quit the command line interpreter"""
|
|
||||||
return self.do_quit(arg)
|
|
||||||
|
|
||||||
def do_EOF(self, arg):
|
|
||||||
"""quit the command line interpreter when ^D is pressed"""
|
|
||||||
print ""
|
|
||||||
return self.do_quit(arg)
|
|
||||||
|
|
||||||
# By default, an emptyline repeats the previous command.
|
|
||||||
# Overriding this function disables the behaviour.
|
|
||||||
def emptyline(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
# Called when command prefix is not recognized
|
|
||||||
def default(self, line):
|
|
||||||
error('don\'t know how to %s' % line)
|
|
||||||
|
|
||||||
# uncomment the leading '_' to use it as a debug function
|
|
||||||
def _completedefault(self, text, line, begidx, endidx):
|
|
||||||
error('don\'t know how to complete '
|
|
||||||
'(text, line, begidx, endidx) =\n'
|
|
||||||
'(%s,%s,%d,%d)' %
|
|
||||||
(text, line, begidx, endidx))
|
|
||||||
|
|
||||||
def _completion_helper(self, text, choices):
|
|
||||||
"""Complete TEST with string from CHOICES."""
|
|
||||||
if text:
|
|
||||||
return [x for x in choices if x.startswith(text)]
|
|
||||||
else:
|
|
||||||
return choices
|
|
||||||
|
|
||||||
# The global help option can be ignored as long as you are to lazy
|
|
||||||
# to implement it yourself.
|
|
||||||
#def do_help(self, arg): pass
|
|
||||||
|
|
||||||
### dedicated commands
|
|
||||||
### fun starts here
|
|
||||||
###
|
|
||||||
|
|
||||||
# TODO: do_status
|
|
||||||
# TODO: toggle_nsfw
|
|
||||||
# TODO: retrieve video from page_url
|
|
||||||
|
|
||||||
def do_backends(self, line):
|
|
||||||
"""backends ACTION [backend0 backend1] …
|
|
||||||
|
|
||||||
ACTION is one of the following:
|
|
||||||
- add: enable backends
|
|
||||||
- rm | remove: disable backends
|
|
||||||
- only: enable only the following backends
|
|
||||||
- view: list enabled and available backends
|
|
||||||
if no arguments are given, default to 'view'
|
|
||||||
"""
|
|
||||||
if not line:
|
|
||||||
args = ["view"] # default behaviour
|
|
||||||
else:
|
|
||||||
args = line.split()
|
|
||||||
|
|
||||||
if args[0] in ["add", "only", "rm", "remove"]:
|
|
||||||
|
|
||||||
if args[0] == "add":
|
|
||||||
action = self.enabled_backends.append
|
|
||||||
elif args[0] == "only":
|
|
||||||
self.enabled_backends = [] # reset
|
|
||||||
action = self.enabled_backends.append
|
|
||||||
elif args[0] == "remove" or args[0] == "rm":
|
|
||||||
action = self.enabled_backends.remove
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
for b in self.available_backends:
|
|
||||||
if b.name in args[1:]:
|
|
||||||
action(b)
|
|
||||||
|
|
||||||
self.enabled_backends.sort()
|
|
||||||
|
|
||||||
# FIXME: do we really need it ?
|
|
||||||
# reload engine
|
|
||||||
self.engine.deinit()
|
|
||||||
names = tuple(x.name for x in self.enabled_backends)
|
|
||||||
self.engine.load_backends(ICapVideo, names=names)
|
|
||||||
|
|
||||||
else: # else "view"
|
|
||||||
availables = " ".join(
|
|
||||||
x.name for x in self.available_backends)
|
|
||||||
enabled = " ".join(
|
|
||||||
x.name for x in self.enabled_backends)
|
|
||||||
print_keys_values([("Available backends", str(availables)),
|
|
||||||
("Enabled backends ", str(enabled))],
|
|
||||||
highlight=True)
|
|
||||||
|
|
||||||
def do_search(self, pattern):
|
|
||||||
"""search [PATTERN]
|
|
||||||
|
|
||||||
Search for videos.
|
|
||||||
If no patterns are given, display the last entries.
|
|
||||||
"""
|
|
||||||
|
|
||||||
if pattern:
|
|
||||||
format = u'Search pattern: %s' % pattern
|
|
||||||
else:
|
|
||||||
format = u'Latest videos'
|
|
||||||
|
|
||||||
# create generator, retrieve videos and add them to self.videos
|
|
||||||
videos_g = self.engine.do('iter_search_results',
|
|
||||||
pattern=pattern, nsfw=True,
|
|
||||||
max_results=10)
|
|
||||||
self.videos = [] # reset
|
|
||||||
for i, (backend, video) in enumerate(videos_g):
|
|
||||||
self.videos.append((backend,video))
|
|
||||||
|
|
||||||
|
|
||||||
# code factorisatorminator: display the list of videos
|
|
||||||
self.do_ls("")
|
|
||||||
|
|
||||||
def complete_backends(self, text, line, begidx, endidx):
|
|
||||||
choices = None
|
|
||||||
|
|
||||||
if line.count(' ') == 1:
|
|
||||||
choices = ["add", "remove", "view", "only"]
|
|
||||||
else:
|
|
||||||
choices = [x.name for x in self.available_backends]
|
|
||||||
|
|
||||||
if choices:
|
|
||||||
return self._completion_helper(text, choices)
|
|
||||||
|
|
||||||
|
|
||||||
def do_ls(self, line):
|
|
||||||
"""ls
|
|
||||||
|
|
||||||
Re-display the last list of videos.
|
|
||||||
"""
|
|
||||||
for i, (backend, video) in enumerate(self.videos):
|
|
||||||
print "%s(%d) %s %s(%s)" % (BOLD, i, video.title, NC,
|
|
||||||
backend.name)
|
|
||||||
print_keys_values([
|
|
||||||
("url", video.url),
|
|
||||||
("duration", "%s seconds" % video.duration),
|
|
||||||
("rating", "%.2f/%.2f" % (video.rating or 0,
|
|
||||||
video.rating_max or 0))],
|
|
||||||
indent=4)
|
|
||||||
|
|
||||||
|
|
||||||
def do_play(self, line):
|
|
||||||
"""play NUMBER
|
|
||||||
|
|
||||||
Play a previously listed video.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
id = int(line)
|
|
||||||
except ValueError:
|
|
||||||
error("invalid number")
|
|
||||||
return False
|
|
||||||
|
|
||||||
try:
|
|
||||||
(backend, video) = self.videos[id]
|
|
||||||
id = video.id
|
|
||||||
except IndexError:
|
|
||||||
error("unknown video number")
|
|
||||||
return False
|
|
||||||
|
|
||||||
# FIXME: do we really have to unload/reload backends ?
|
|
||||||
self.engine.deinit()
|
|
||||||
names = (backend.name,) if backend is not None else None
|
|
||||||
self.engine.load_backends(ICapVideo, names=names)
|
|
||||||
|
|
||||||
# XXX: copy&paste from weboob-cli,
|
|
||||||
# don't ask me anything about it :(
|
|
||||||
for backend, video in self.engine.do('get_video', id):
|
|
||||||
if video is None:
|
|
||||||
continue
|
|
||||||
self.player.play(video)
|
|
||||||
|
|
||||||
|
|
||||||
class VideoobRepl(ConsoleApplication):
|
|
||||||
APPNAME = 'videoob-repl'
|
|
||||||
VERSION = '0.2'
|
|
||||||
COPYRIGHT = 'Copyright(C) 2010 John Obbele'
|
|
||||||
|
|
||||||
def add_application_options(self, group):
|
|
||||||
group.add_option('-C', '--configured',
|
|
||||||
action='store_true',
|
|
||||||
help='load configured backends')
|
|
||||||
|
|
||||||
def main(self, argv):
|
|
||||||
player = MyPlayer(DefaultOptions())
|
|
||||||
console = self
|
|
||||||
MyCmd(console, player).cmdloop()
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue