|
|
|
from couchpotato import addView
|
|
|
|
from couchpotato.core.event import fireEvent, addEvent
|
|
|
|
from couchpotato.core.helpers.variable import getExt
|
|
|
|
from couchpotato.core.logger import CPLog
|
|
|
|
from couchpotato.environment import Env
|
|
|
|
from flask.helpers import send_from_directory
|
|
|
|
from flask.templating import render_template_string
|
|
|
|
from multipartpost import MultipartPostHandler
|
|
|
|
from urlparse import urlparse
|
|
|
|
import cookielib
|
|
|
|
import glob
|
|
|
|
import math
|
|
|
|
import os.path
|
|
|
|
import re
|
|
|
|
import socket
|
|
|
|
import time
|
|
|
|
import traceback
|
|
|
|
import urllib
|
|
|
|
import urllib2
|
|
|
|
|
|
|
|
log = CPLog(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class Plugin(object):
|
|
|
|
|
|
|
|
enabled_option = 'enabled'
|
|
|
|
auto_register_static = True
|
|
|
|
|
|
|
|
needs_shutdown = False
|
|
|
|
running = []
|
|
|
|
|
|
|
|
http_last_use = {}
|
|
|
|
http_time_between_calls = 0
|
|
|
|
|
|
|
|
def registerPlugin(self):
|
|
|
|
addEvent('app.shutdown', self.doShutdown)
|
|
|
|
addEvent('plugin.running', self.isRunning)
|
|
|
|
|
|
|
|
def conf(self, attr, value = None, default = None):
|
|
|
|
return Env.setting(attr, self.getName().lower(), value = value, default = default)
|
|
|
|
|
|
|
|
def getName(self):
|
|
|
|
return self.__class__.__name__
|
|
|
|
|
|
|
|
def renderTemplate(self, parent_file, template, **params):
|
|
|
|
|
|
|
|
template = open(os.path.join(os.path.dirname(parent_file), template), 'r').read()
|
|
|
|
return render_template_string(template, **params)
|
|
|
|
|
|
|
|
def registerStatic(self, plugin_file, add_to_head = True):
|
|
|
|
|
|
|
|
# Register plugin path
|
|
|
|
self.plugin_path = os.path.dirname(plugin_file)
|
|
|
|
|
|
|
|
# Get plugin_name from PluginName
|
|
|
|
s1 = re.sub('(.)([A-Z][a-z]+)', r'\1_\2', self.__class__.__name__)
|
|
|
|
class_name = re.sub('([a-z0-9])([A-Z])', r'\1_\2', s1).lower()
|
|
|
|
|
|
|
|
path = 'static/' + class_name + '/'
|
|
|
|
addView(path + '<path:filename>', self.showStatic, static = True)
|
|
|
|
|
|
|
|
if add_to_head:
|
|
|
|
for f in glob.glob(os.path.join(self.plugin_path, 'static', '*')):
|
|
|
|
fireEvent('register_%s' % ('script' if getExt(f) in 'js' else 'style'), path + os.path.basename(f))
|
|
|
|
|
|
|
|
def showStatic(self, filename):
|
|
|
|
d = os.path.join(self.plugin_path, 'static')
|
|
|
|
return send_from_directory(d, filename)
|
|
|
|
|
|
|
|
def createFile(self, path, content):
|
|
|
|
|
|
|
|
self.makeDir(os.path.dirname(path))
|
|
|
|
|
|
|
|
try:
|
|
|
|
f = open(path, 'w')
|
|
|
|
f.write(content)
|
|
|
|
f.close()
|
|
|
|
except Exception, e:
|
|
|
|
log.error('Unable writing to file "%s": %s' % (path, e))
|
|
|
|
|
|
|
|
def makeDir(self, path):
|
|
|
|
try:
|
|
|
|
if not os.path.isdir(path):
|
|
|
|
os.makedirs(path, Env.getPermission('folder'))
|
|
|
|
return True
|
|
|
|
except Exception, e:
|
|
|
|
log.error('Unable to create folder "%s": %s' % (path, e))
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
# http request
|
|
|
|
def urlopen(self, url, timeout = 10, params = {}, headers = {}, multipart = False, show_error = True):
|
|
|
|
|
|
|
|
socket.setdefaulttimeout(timeout)
|
|
|
|
|
|
|
|
# Fill in some headers
|
|
|
|
if not headers.get('Referer'):
|
|
|
|
headers['Referer'] = urlparse(url).hostname
|
|
|
|
if not headers.get('User-Agent'):
|
|
|
|
headers['User-Agent'] = ''
|
|
|
|
|
|
|
|
host = urlparse(url).hostname
|
|
|
|
self.wait(host)
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
if multipart:
|
|
|
|
log.info('Opening multipart url: %s, params: %s' % (url, [x for x in params.iterkeys()]))
|
|
|
|
request = urllib2.Request(url, params, headers)
|
|
|
|
|
|
|
|
cookies = cookielib.CookieJar()
|
|
|
|
opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cookies), MultipartPostHandler)
|
|
|
|
|
|
|
|
data = opener.open(request).read()
|
|
|
|
else:
|
|
|
|
log.info('Opening url: %s, params: %s' % (url, [x for x in params.iterkeys()]))
|
|
|
|
data = urllib.urlencode(params) if len(params) > 0 else None
|
|
|
|
request = urllib2.Request(url, data, headers)
|
|
|
|
|
|
|
|
data = urllib2.urlopen(request).read()
|
|
|
|
except IOError:
|
|
|
|
if show_error:
|
|
|
|
log.error('Failed opening url in %s: %s %s' % (self.getName(), url, traceback.format_exc(1)))
|
|
|
|
raise
|
|
|
|
|
|
|
|
self.http_last_use[host] = time.time()
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
def wait(self, host = ''):
|
|
|
|
now = time.time()
|
|
|
|
|
|
|
|
last_use = self.http_last_use.get(host, 0)
|
|
|
|
|
|
|
|
wait = math.ceil(last_use - now + self.http_time_between_calls)
|
|
|
|
|
|
|
|
if wait > 0:
|
|
|
|
log.debug('Waiting for %s, %d seconds' % (self.getName(), wait))
|
|
|
|
time.sleep(last_use - now + self.http_time_between_calls)
|
|
|
|
|
|
|
|
def beforeCall(self, handler):
|
|
|
|
#log.debug('Calling %s.%s' % (self.getName(), handler.__name__))
|
|
|
|
self.isRunning('%s.%s' % (self.getName(), handler.__name__))
|
|
|
|
|
|
|
|
def afterCall(self, handler):
|
|
|
|
self.isRunning('%s.%s' % (self.getName(), handler.__name__), False)
|
|
|
|
|
|
|
|
def doShutdown(self):
|
|
|
|
self.shuttingDown(True)
|
|
|
|
|
|
|
|
def shuttingDown(self, value = None):
|
|
|
|
if value is None:
|
|
|
|
return self.needs_shutdown
|
|
|
|
|
|
|
|
self.needs_shutdown = value
|
|
|
|
|
|
|
|
def isRunning(self, value = None, boolean = True):
|
|
|
|
if value is None:
|
|
|
|
return self.running
|
|
|
|
|
|
|
|
if boolean:
|
|
|
|
self.running.append(value)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
self.running.remove(value)
|
|
|
|
except:
|
|
|
|
log.error("Something went wrong when finishing the plugin function. Could not find the 'is_running' key")
|
|
|
|
|
|
|
|
|
|
|
|
def getCache(self, cache_key, url = None):
|
|
|
|
cache = Env.get('cache').get(cache_key)
|
|
|
|
if cache:
|
|
|
|
if not Env.setting('development'): log.debug('Getting cache %s' % cache_key)
|
|
|
|
return cache
|
|
|
|
|
|
|
|
if url:
|
|
|
|
try:
|
|
|
|
data = self.urlopen(url)
|
|
|
|
self.setCache(cache_key, data)
|
|
|
|
return data
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def setCache(self, cache_key, value, timeout = 300):
|
|
|
|
log.debug('Setting cache %s' % cache_key)
|
|
|
|
Env.get('cache').set(cache_key, value, timeout)
|
|
|
|
return value
|
|
|
|
|
|
|
|
def isDisabled(self):
|
|
|
|
return not self.isEnabled()
|
|
|
|
|
|
|
|
def isEnabled(self):
|
|
|
|
return self.conf(self.enabled_option) or self.conf(self.enabled_option) == None
|