6 changed files with 242 additions and 55 deletions
@ -1,63 +1,167 @@ |
|||
#!/usr/bin/env python |
|||
# -*- coding: utf-8 -*- |
|||
'''Wrapper for the command line interface.''' |
|||
|
|||
from os.path import dirname |
|||
from threading import Thread |
|||
from wx.lib.softwareupdate import SoftwareUpdate |
|||
import os |
|||
import sys |
|||
import subprocess |
|||
import time |
|||
import webbrowser |
|||
import wx |
|||
|
|||
|
|||
# Root path |
|||
base_path = dirname(os.path.abspath(__file__)) |
|||
# Include proper dirs |
|||
if hasattr(sys, 'frozen'): |
|||
import libs |
|||
base_path = os.path.dirname(os.path.dirname(os.path.abspath(libs.__file__))) |
|||
print base_path |
|||
else: |
|||
base_path = os.path.dirname(os.path.abspath(__file__)) |
|||
|
|||
# Insert local directories into path |
|||
sys.path.insert(0, os.path.join(base_path, 'libs')) |
|||
lib_dir = os.path.join(base_path, 'libs') |
|||
|
|||
from couchpotato.core.logger import CPLog |
|||
log = CPLog(__name__) |
|||
sys.path.insert(0, base_path) |
|||
sys.path.insert(0, lib_dir) |
|||
|
|||
# Get options via arg |
|||
from couchpotato.runner import getOptions |
|||
from couchpotato.core.helpers.variable import getDataDir |
|||
options = getOptions(base_path, sys.argv[1:]) |
|||
data_dir = getDataDir() |
|||
|
|||
def start(): |
|||
try: |
|||
args = [sys.executable] + sys.argv |
|||
new_environ = os.environ.copy() |
|||
new_environ['cp_main'] = 'true' |
|||
|
|||
if os.name == 'nt': |
|||
for key, value in new_environ.iteritems(): |
|||
if isinstance(value, unicode): |
|||
new_environ[key] = value.encode('iso-8859-1') |
|||
|
|||
subprocess.call(args, env = new_environ) |
|||
return os.path.isfile(os.path.join(data_dir, 'restart')) |
|||
except KeyboardInterrupt, e: |
|||
pass |
|||
except Exception, e: |
|||
log.critical(e) |
|||
return 0 |
|||
|
|||
from couchpotato.runner import runCouchPotato |
|||
if __name__ == '__main__': |
|||
|
|||
if os.environ.get('cp_main', 'false') == 'true': |
|||
|
|||
class TaskBarIcon(wx.TaskBarIcon): |
|||
|
|||
TBMENU_OPEN = wx.NewId() |
|||
TBMENU_SETTINGS = wx.NewId() |
|||
TBMENU_ABOUT = wx.ID_ABOUT |
|||
TBMENU_EXIT = wx.ID_EXIT |
|||
|
|||
def __init__(self, frame): |
|||
wx.TaskBarIcon.__init__(self) |
|||
self.frame = frame |
|||
|
|||
icon = wx.Icon('icon.ico', wx.BITMAP_TYPE_ANY) |
|||
self.SetIcon(icon) |
|||
|
|||
self.Bind(wx.EVT_TASKBAR_LEFT_DCLICK, self.onTaskBarActivate) |
|||
|
|||
self.Bind(wx.EVT_MENU, self.onOpen, id = self.TBMENU_OPEN) |
|||
self.Bind(wx.EVT_MENU, self.onSettings, id = self.TBMENU_SETTINGS) |
|||
self.Bind(wx.EVT_MENU, self.onAbout, id = self.TBMENU_ABOUT) |
|||
self.Bind(wx.EVT_MENU, self.onTaskBarClose, id = self.TBMENU_EXIT) |
|||
|
|||
|
|||
def CreatePopupMenu(self): |
|||
menu = wx.Menu() |
|||
menu.Append(self.TBMENU_OPEN, "Open") |
|||
menu.Append(self.TBMENU_SETTINGS, "Settings") |
|||
menu.Append(self.TBMENU_ABOUT, "About") |
|||
menu.Append(self.TBMENU_EXIT, "Close") |
|||
return menu |
|||
|
|||
def onOpen(self, event): |
|||
url = self.frame.parent.getSetting('base_url') |
|||
webbrowser.open(url) |
|||
|
|||
def onSettings(self, event): |
|||
url = self.frame.parent.getSetting('base_url') + '/settings/' |
|||
webbrowser.open(url) |
|||
|
|||
def onAbout(self, event): |
|||
print 'onAbout' |
|||
|
|||
def onTaskBarActivate(self, evt): |
|||
if not self.frame.IsShown(): |
|||
self.frame.Show(True) |
|||
self.frame.Raise() |
|||
|
|||
def onTaskBarClose(self, evt): |
|||
wx.CallAfter(self.frame.Close) |
|||
|
|||
def makeIcon(self, img): |
|||
if "wxMSW" in wx.PlatformInfo: |
|||
img = img.Scale(16, 16) |
|||
elif "wxGTK" in wx.PlatformInfo: |
|||
img = img.Scale(22, 22) |
|||
|
|||
icon = wx.IconFromBitmap(img.CopyFromBitmap()) |
|||
return icon |
|||
|
|||
|
|||
class MainFrame(wx.Frame): |
|||
|
|||
def __init__(self, parent): |
|||
wx.Frame.__init__(self, None) |
|||
|
|||
self.parent = parent |
|||
self.tbicon = TaskBarIcon(self) |
|||
|
|||
|
|||
class WorkerThread(Thread): |
|||
|
|||
def __init__(self, desktop): |
|||
Thread.__init__(self) |
|||
self._desktop = desktop |
|||
|
|||
self.start() |
|||
|
|||
def run(self): |
|||
|
|||
args = ['--nogit', '--console_log']#, '--quiet'] |
|||
options = getOptions(base_path, args) |
|||
|
|||
try: |
|||
runCouchPotato(options, base_path, sys.argv[1:]) |
|||
runCouchPotato(options, base_path, args, desktop = self._desktop) |
|||
except KeyboardInterrupt, e: |
|||
raise |
|||
except Exception, e: |
|||
log.critical(e) |
|||
else: |
|||
while 1: |
|||
restart = start() |
|||
if not restart: |
|||
break |
|||
|
|||
from couchpotato.core.event import fireEvent |
|||
fireEvent('app.crappy_shutdown', single = True) |
|||
time.sleep(1) |
|||
sys.exit() |
|||
raise |
|||
finally: |
|||
pass |
|||
|
|||
|
|||
class CouchPotatoApp(wx.App, SoftwareUpdate): |
|||
|
|||
settings = {} |
|||
events = {} |
|||
restart = False |
|||
|
|||
def OnInit(self): |
|||
|
|||
# Updater |
|||
base_url = 'http://couchpotatoapp.com/updates/' |
|||
self.InitUpdates(base_url, base_url + 'changelog.txt', |
|||
icon = wx.Icon('icon.ico')) |
|||
|
|||
self.frame = MainFrame(self) |
|||
self.frame.Bind(wx.EVT_CLOSE, self.onClose) |
|||
|
|||
# CouchPotato thread |
|||
self.worker = WorkerThread(self) |
|||
|
|||
return True |
|||
|
|||
def setSettings(self, settings = {}): |
|||
self.settings = settings |
|||
|
|||
def getSetting(self, name): |
|||
return self.settings.get(name) |
|||
|
|||
def addEvents(self, events = {}): |
|||
for name in events.iterkeys(): |
|||
self.events[name] = events[name] |
|||
|
|||
def onClose(self, event): |
|||
onClose = self.events.get('onClose') |
|||
if self.events.get('onClose'): |
|||
onClose(event) |
|||
else: |
|||
self.afterShutdown() |
|||
|
|||
def afterShutdown(self, restart = False): |
|||
self.frame.Destroy() |
|||
self.restart = restart |
|||
|
|||
|
|||
if __name__ == '__main__': |
|||
app = CouchPotatoApp(redirect = False) |
|||
app.MainLoop() |
|||
|
|||
#path = os.path.join(sys.path[0].decode(sys.getfilesystemencoding()), sys.argv[0]) |
|||
#if app.restart: |
|||
# wx.Process.Open(sys.executable + ' ' + path) |
|||
|
@ -1,5 +1 @@ |
|||
CouchPotato Server |
|||
===== |
|||
|
|||
CouchPotato (CP) is an automatic NZB and torrent downloader. You can keep a "movies I want"-list and it will search for NZBs/torrents of these movies every X hours. |
|||
Once a movie is found, it will send it to SABnzbd or download the torrent to a specified directory. |
|||
CouchPotato Desktop |
Binary file not shown.
After Width: | Height: | Size: 28 KiB |
@ -0,0 +1,86 @@ |
|||
from esky import bdist_esky |
|||
from setuptools import setup |
|||
import sys |
|||
import version |
|||
import os |
|||
|
|||
|
|||
# Include proper dirs |
|||
base_path = os.path.dirname(os.path.abspath(__file__)) |
|||
lib_dir = os.path.join(base_path, 'libs') |
|||
|
|||
sys.path.insert(0, base_path) |
|||
sys.path.insert(0, lib_dir) |
|||
|
|||
|
|||
|
|||
# Windows |
|||
if sys.platform == "win32": |
|||
import py2exe |
|||
|
|||
FREEZER = 'py2exe' |
|||
FREEZER_OPTIONS = dict( |
|||
compressed = 0, |
|||
optimize = 0, |
|||
bundle_files = 3, |
|||
dll_excludes = [ |
|||
'MSVCP90.dll', |
|||
'mswsock.dll', |
|||
'powrprof.dll', |
|||
'USP10.dll', |
|||
], |
|||
packages = ['couchpotato', 'libs'], |
|||
includes = [ |
|||
'telnetlib', |
|||
'xml.etree.ElementTree', |
|||
'xml.etree.cElementTree', |
|||
'xml.dom', |
|||
'xml.dom.minidom', |
|||
], |
|||
) |
|||
exeICON = 'icon.ico' |
|||
|
|||
|
|||
# OSX |
|||
elif sys.platform == "darwin": |
|||
import py2app |
|||
|
|||
FREEZER = 'py2app' |
|||
FREEZER_OPTIONS = dict( |
|||
argv_emulation = False, |
|||
iconfile = 'icon.icns', |
|||
plist = dict( |
|||
LSUIElement = True, |
|||
), |
|||
packages = ['couchpotato', 'libs'], |
|||
includes = [ |
|||
'telnetlib', |
|||
'xml.etree.ElementTree', |
|||
'xml.etree.cElementTree', |
|||
'xml.dom', |
|||
'xml.dom.minidom', |
|||
], |
|||
) |
|||
exeICON = None |
|||
|
|||
# Common |
|||
NAME = "CouchPotato" |
|||
APP = [bdist_esky.Executable("CouchPotato.py", gui_only = True, icon = exeICON,)] |
|||
DATA_FILES = ['icon.ico'] |
|||
ESKY_OPTIONS = dict( |
|||
freezer_module = FREEZER, |
|||
freezer_options = FREEZER_OPTIONS, |
|||
bundle_msvcrt = True, |
|||
) |
|||
|
|||
|
|||
# Build the app and the esky bundle |
|||
setup( |
|||
name = NAME, |
|||
scripts = APP, |
|||
version = version.VERSION, |
|||
data_files = DATA_FILES, |
|||
options = dict(bdist_esky = ESKY_OPTIONS), |
|||
) |
|||
|
|||
|
@ -0,0 +1 @@ |
|||
VERSION = '0.5' |
Loading…
Reference in new issue