pandoralocal
This commit is contained in:
commit
47182a28d6
10 changed files with 332 additions and 0 deletions
19
README
Normal file
19
README
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
pandoralocal runs a webserver on http://localhost:2620
|
||||||
|
that provides services that are integrated with a pandora instance
|
||||||
|
|
||||||
|
- there should be a way to start pandoralocal with a user session for osx, win32, linux
|
||||||
|
- possibly some gui to start/stop and enable/disable session startup
|
||||||
|
|
||||||
|
once running the configuration is done via web interface
|
||||||
|
there are 2 modes
|
||||||
|
|
||||||
|
a) integration with a pandora instance, this happens via trying to connect to
|
||||||
|
http://local.pad.ma:2620/api/ and if successfull use the local api in the pandora
|
||||||
|
site to manage local volumes(upload/sync), cached videos(playback, download for later playback)
|
||||||
|
|
||||||
|
b) offline annotations(something like speedtrans) with ability to sync/upload to pandora later
|
||||||
|
this uses the same cache for videos but runs on http://127.0.0.1:2620/pad.ma/
|
||||||
|
since there might be no dns lookup for local.pad.ma possible
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
25
bin/pandoralocal
Executable file
25
bin/pandoralocal
Executable file
|
|
@ -0,0 +1,25 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
# GPL 2008
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from glob import glob
|
||||||
|
from optparse import OptionParser
|
||||||
|
|
||||||
|
root = os.path.join(os.path.abspath(os.path.dirname(__file__)), '..')
|
||||||
|
if os.path.exists(os.path.join(root, 'pandoralocal')):
|
||||||
|
sys.path.insert(0, root)
|
||||||
|
|
||||||
|
import pandoralocal
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
parser = OptionParser()
|
||||||
|
parser.add_option('-c', '--config', dest='config', help='config file', default='config.json')
|
||||||
|
(opts, args) = parser.parse_args()
|
||||||
|
|
||||||
|
if None in (opts.config, ):
|
||||||
|
parser.print_help()
|
||||||
|
sys.exit()
|
||||||
|
pandoralocal.main(opts.config)
|
||||||
22
pandoralocal/__init__.py
Normal file
22
pandoralocal/__init__.py
Normal file
|
|
@ -0,0 +1,22 @@
|
||||||
|
# encoding: utf-8
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
import os
|
||||||
|
|
||||||
|
from twisted.web.server import Site
|
||||||
|
from twisted.internet import reactor
|
||||||
|
|
||||||
|
from backend import Backend
|
||||||
|
from server import Server
|
||||||
|
|
||||||
|
from version import __version__
|
||||||
|
|
||||||
|
|
||||||
|
def main(config):
|
||||||
|
base = os.path.abspath(os.path.dirname(__file__))
|
||||||
|
backend = Backend(config)
|
||||||
|
root = Server(base, backend)
|
||||||
|
site = Site(root)
|
||||||
|
port = 2620
|
||||||
|
interface = '127.0.0.1'
|
||||||
|
reactor.listenTCP(port, site, interface=interface)
|
||||||
|
reactor.run()
|
||||||
12
pandoralocal/api.py
Normal file
12
pandoralocal/api.py
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
# encoding: utf-8
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
from server import actions, json_response
|
||||||
|
|
||||||
|
|
||||||
|
def echo(backend, site, data):
|
||||||
|
return json_response(data)
|
||||||
|
actions.register(echo, cache=False)
|
||||||
|
|
||||||
|
def site(backend, site, data):
|
||||||
|
return json_response({'site': site})
|
||||||
|
actions.register(site, cache=False)
|
||||||
15
pandoralocal/backend.py
Normal file
15
pandoralocal/backend.py
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
# encoding: utf-8
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
|
||||||
|
class Backend:
|
||||||
|
def __init__(self, config):
|
||||||
|
self.config = config
|
||||||
|
|
||||||
|
def get_file(self, site, itemId, filename):
|
||||||
|
filename, ext = filename.split('.')
|
||||||
|
resolution, part = filename.split('p')
|
||||||
|
print site, itemId, resolution, part, ext
|
||||||
|
path = ''
|
||||||
|
if resolution == '480' and ext == 'webm':
|
||||||
|
path = '/home/j/.ox/media/44/c4/b1/11a888e96a/480p.webm'
|
||||||
|
return path
|
||||||
198
pandoralocal/server.py
Normal file
198
pandoralocal/server.py
Normal file
|
|
@ -0,0 +1,198 @@
|
||||||
|
# encoding: utf-8
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
import sys
|
||||||
|
import inspect
|
||||||
|
import os
|
||||||
|
import json
|
||||||
|
from urlparse import urlparse
|
||||||
|
|
||||||
|
from twisted.web.resource import Resource
|
||||||
|
from twisted.web.static import File
|
||||||
|
from twisted.web.util import Redirect
|
||||||
|
|
||||||
|
from version import __version__
|
||||||
|
|
||||||
|
def trim(docstring):
|
||||||
|
if not docstring:
|
||||||
|
return ''
|
||||||
|
# Convert tabs to spaces (following the normal Python rules)
|
||||||
|
# and split into a list of lines:
|
||||||
|
lines = docstring.expandtabs().splitlines()
|
||||||
|
# Determine minimum indentation (first line doesn't count):
|
||||||
|
indent = sys.maxint
|
||||||
|
for line in lines[1:]:
|
||||||
|
stripped = line.lstrip()
|
||||||
|
if stripped:
|
||||||
|
indent = min(indent, len(line) - len(stripped))
|
||||||
|
# Remove indentation (first line is special):
|
||||||
|
trimmed = [lines[0].strip()]
|
||||||
|
if indent < sys.maxint:
|
||||||
|
for line in lines[1:]:
|
||||||
|
trimmed.append(line[indent:].rstrip())
|
||||||
|
# Strip off trailing and leading blank lines:
|
||||||
|
while trimmed and not trimmed[-1]:
|
||||||
|
trimmed.pop()
|
||||||
|
while trimmed and not trimmed[0]:
|
||||||
|
trimmed.pop(0)
|
||||||
|
# Return a single string:
|
||||||
|
return '\n'.join(trimmed)
|
||||||
|
|
||||||
|
def json_response(data=None, status=200, text='ok'):
|
||||||
|
if not data:
|
||||||
|
data = {}
|
||||||
|
return {'status': {'code': status, 'text': text}, 'data': data}
|
||||||
|
|
||||||
|
class ApiActions(dict):
|
||||||
|
properties = {}
|
||||||
|
def __init__(self):
|
||||||
|
|
||||||
|
def api(site, data):
|
||||||
|
'''
|
||||||
|
returns list of all known api actions
|
||||||
|
param data {
|
||||||
|
docs: bool
|
||||||
|
}
|
||||||
|
if docs is true, action properties contain docstrings
|
||||||
|
return {
|
||||||
|
status: {'code': int, 'text': string},
|
||||||
|
data: {
|
||||||
|
actions: {
|
||||||
|
'api': {
|
||||||
|
cache: true,
|
||||||
|
doc: 'recursion'
|
||||||
|
},
|
||||||
|
'hello': {
|
||||||
|
cache: true,
|
||||||
|
..
|
||||||
|
}
|
||||||
|
...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
docs = data.get('docs', False)
|
||||||
|
code = data.get('code', False)
|
||||||
|
_actions = self.keys()
|
||||||
|
_actions.sort()
|
||||||
|
actions = {}
|
||||||
|
for a in _actions:
|
||||||
|
actions[a] = self.properties[a]
|
||||||
|
if docs:
|
||||||
|
actions[a]['doc'] = self.doc(a)
|
||||||
|
if code:
|
||||||
|
actions[a]['code'] = self.code(a)
|
||||||
|
response = json_response({'actions': actions})
|
||||||
|
return response
|
||||||
|
self.register(api)
|
||||||
|
|
||||||
|
def doc(self, f):
|
||||||
|
return trim(self[f].__doc__)
|
||||||
|
|
||||||
|
def code(self, name):
|
||||||
|
f = self[name]
|
||||||
|
if name != 'api' and hasattr(f, 'func_closure') and f.func_closure:
|
||||||
|
f = f.func_closure[0].cell_contents
|
||||||
|
info = f.func_code.co_firstlineno
|
||||||
|
return info, trim(inspect.getsource(f))
|
||||||
|
|
||||||
|
def register(self, method, action=None, cache=True):
|
||||||
|
if not action:
|
||||||
|
action = method.func_name
|
||||||
|
self[action] = method
|
||||||
|
self.properties[action] = {'cache': cache}
|
||||||
|
|
||||||
|
def unregister(self, action):
|
||||||
|
if action in self:
|
||||||
|
del self[action]
|
||||||
|
|
||||||
|
def render(self, backend, site, action, data):
|
||||||
|
if action in self:
|
||||||
|
result = self[action](backend, site, data)
|
||||||
|
else:
|
||||||
|
result = json_response(status=404, text='not found')
|
||||||
|
print result
|
||||||
|
return json.dumps(result)
|
||||||
|
|
||||||
|
actions = ApiActions()
|
||||||
|
|
||||||
|
|
||||||
|
class Server(Resource):
|
||||||
|
|
||||||
|
def __init__(self, base, backend):
|
||||||
|
self.base = base
|
||||||
|
self.backend = backend
|
||||||
|
Resource.__init__(self)
|
||||||
|
|
||||||
|
def static_path(self, path):
|
||||||
|
return os.path.join(self.base, 'static', path)
|
||||||
|
|
||||||
|
def get_site(self, request):
|
||||||
|
headers = request.getAllHeaders()
|
||||||
|
#print headers
|
||||||
|
if 'origin' in headers:
|
||||||
|
request.headers['Access-Control-Allow-Origin'] = headers['origin']
|
||||||
|
site = headers['origin']
|
||||||
|
elif 'referer' in headers:
|
||||||
|
u = urlparse(headers['referer'])
|
||||||
|
site = u.scheme + '://' + u.hostname
|
||||||
|
else:
|
||||||
|
site = 'http://' + headers['host']
|
||||||
|
return site
|
||||||
|
|
||||||
|
def getChild(self, name, request):
|
||||||
|
print request
|
||||||
|
if name in ('icon.png', 'favicon.ico'):
|
||||||
|
f = File(self.static_path('png/icon16.png'))
|
||||||
|
f.isLeaf = True
|
||||||
|
return f
|
||||||
|
if request.path == '/api/':
|
||||||
|
return self
|
||||||
|
if request.path.endswith('.webm'):
|
||||||
|
video = request.path
|
||||||
|
site = self.get_site(request)
|
||||||
|
itemId, filename = video[1:].split('/')
|
||||||
|
path = self.backend.get_file(site, itemId, filename)
|
||||||
|
if path:
|
||||||
|
print itemId, filename, 'use', path
|
||||||
|
request.headers['Access-Control-Allow-Origin'] = '*'
|
||||||
|
f = File(path)
|
||||||
|
f.isLeaf = True
|
||||||
|
return f
|
||||||
|
else:
|
||||||
|
url = site + video
|
||||||
|
#url = 'http://padmo.local/B/240p.webm'
|
||||||
|
print "redirect", url
|
||||||
|
return Redirect(url)
|
||||||
|
path = request.path
|
||||||
|
path = path[1:]
|
||||||
|
if not path:
|
||||||
|
path = 'index.html'
|
||||||
|
path = self.static_path(path)
|
||||||
|
f = File(path)
|
||||||
|
if not os.path.isdir(path):
|
||||||
|
f.isLeaf = True
|
||||||
|
return f
|
||||||
|
|
||||||
|
def render_POST(self, request):
|
||||||
|
print 'render_POST'
|
||||||
|
request.headers['Server'] = 'pandoralocal/%s' % __version__
|
||||||
|
site = self.get_site(request)
|
||||||
|
print "POST", request.args
|
||||||
|
if 'action' in request.args:
|
||||||
|
if 'data' in request.args:
|
||||||
|
data = json.loads(request.args['data'][0])
|
||||||
|
else:
|
||||||
|
data = {}
|
||||||
|
action = request.args['action'][0]
|
||||||
|
return actions.render(self.backend, site, action, data)
|
||||||
|
|
||||||
|
def render_GET(self, request):
|
||||||
|
print 'render_GET'
|
||||||
|
request.headers['Server'] = 'pandoralocal/%s' % __version__
|
||||||
|
f = open('static/index.html')
|
||||||
|
data = f.read()
|
||||||
|
f.close()
|
||||||
|
request.headers['Content-Type'] = 'text/html'
|
||||||
|
site = self.get_site(request)
|
||||||
|
data = data.replace('$name', site)
|
||||||
|
return data
|
||||||
2
pandoralocal/static/index.html
Normal file
2
pandoralocal/static/index.html
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
only robots may pass this point
|
||||||
|
$name
|
||||||
BIN
pandoralocal/static/png/icon16.png
Normal file
BIN
pandoralocal/static/png/icon16.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.4 KiB |
1
pandoralocal/version.py
Normal file
1
pandoralocal/version.py
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
__version__ = 0.0
|
||||||
38
setup.py
Normal file
38
setup.py
Normal file
|
|
@ -0,0 +1,38 @@
|
||||||
|
# setup.py
|
||||||
|
# -*- coding: UTF-8 -*-
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
try:
|
||||||
|
from setuptools import setup
|
||||||
|
except:
|
||||||
|
from distutils.core import setup
|
||||||
|
|
||||||
|
def get_version():
|
||||||
|
import os
|
||||||
|
from pandoralocal.version import __version__
|
||||||
|
info = os.path.join(os.path.dirname(__file__), '.bzr/branch/last-revision')
|
||||||
|
if os.path.exists(info):
|
||||||
|
f = open(info)
|
||||||
|
rev = int(f.read().split()[0])
|
||||||
|
f.close()
|
||||||
|
if rev:
|
||||||
|
return u'%s.%s' % (__version__, rev)
|
||||||
|
return '%s'%__version__
|
||||||
|
|
||||||
|
setup(name="pandoralocal",
|
||||||
|
version=get_version() ,
|
||||||
|
scripts=[
|
||||||
|
'bin/pandoralocal',
|
||||||
|
],
|
||||||
|
packages=[
|
||||||
|
'pandoralocal',
|
||||||
|
],
|
||||||
|
author="0x2620",
|
||||||
|
author_email="0x2620@0x2620.org",
|
||||||
|
description="pandoralocal allows users to use local videos on a pan.do/ra site",
|
||||||
|
classifiers = [
|
||||||
|
'Development Status :: 1 - Development/Unstable',
|
||||||
|
'Operating System :: OS Independent',
|
||||||
|
'Programming Language :: Python',
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue