From 740694534bff8768635f340ed3b1a4c9570667db Mon Sep 17 00:00:00 2001 From: Chris Caron Date: Sun, 9 Jul 2017 21:43:11 -0400 Subject: [PATCH] Inline Testing from within NZBGet; fixes #21 --- Notify.py | 54 +++++++- Notify/nzbget/ScriptBase.py | 245 +++++++++++++++++++++++++++--------- Notify/nzbget/__init__.py | 7 +- 3 files changed, 239 insertions(+), 67 deletions(-) diff --git a/Notify.py b/Notify.py index d022074..2ee71fe 100755 --- a/Notify.py +++ b/Notify.py @@ -31,9 +31,9 @@ # # Info about this Notify NZB Script: # Author: Chris Caron (lead2gold@gmail.com). -# Date: Wed, Jul 5th, 2017. +# Date: Wed, Jul 7th, 2017. # License: GPLv2 (http://www.gnu.org/licenses/gpl.html). -# Script Version: 0.6.0 +# Script Version: 0.6.1 # ########################################################################### @@ -291,6 +291,9 @@ # if they have this extra bit of detail in your logging output. #Debug=no +# You can test your server configuration here. +#TestServers@Test Server Configuration + ### NZBGET QUEUE/POST-PROCESSING SCRIPT ########################################################################### import sys @@ -1150,6 +1153,53 @@ def hhmmss(seconds): notify_type=notify_type, ) + def action_testservers(self, *args, **kwargs): + """ + Execute the TestServers Test Action + """ + + if not self.validate(keys=( + 'Servers', + 'IncludeImage', + 'IncludeFiles', + 'IncludeStats', + 'IncludeLogs', + 'OnFailure', + 'OnSuccess', + )): + return False + + servers = self.parse_list(self.get('Servers', '')) + on_failure = self.parse_bool(self.get('OnFailure')) + on_success = self.parse_bool(self.get('OnSuccess')) + + include_stats = self.parse_bool(self.get('IncludeStats')) + include_files = self.parse_bool(self.get('IncludeFiles')) + include_logs = self.get('IncludeLogs', 'NO').upper() + + # Prepare our Test Response + notify_type = NotifyType.INFO + title='NZBGet-Notify Configuration Test' + body='## NZBGet-Notify Configuration Test ##\r\n' + body += '- **OnFailure**: %s\r\n' % ( + 'Yes' if on_failure else 'No') + body += '- **OnSuccess**: %s\r\n' % ( + 'Yes' if on_success else 'No') + body += '- **Include Statistics**: %s\r\n' % ( + 'Yes' if include_stats else 'No') + body += '- **Include File Listings**: %s\r\n' % ( + 'Yes' if include_files else 'No') + body += '- **Include Log Details**: %s\r\n' % ( + 'Yes' if include_logs else 'No') + + # Preform Notifications + return self.notify( + servers, + title=title, + body=body, + notify_type=notify_type, + ) + def main(self, *args, **kwargs): """CLI """ diff --git a/Notify/nzbget/ScriptBase.py b/Notify/nzbget/ScriptBase.py index 91b612c..2bf58f0 100644 --- a/Notify/nzbget/ScriptBase.py +++ b/Notify/nzbget/ScriptBase.py @@ -142,7 +142,6 @@ from os import X_OK from os import kill from os import getpid -from os import name as os_name from os.path import isdir from os.path import islink from os.path import isfile @@ -156,7 +155,6 @@ from logging import Logger from datetime import datetime from Utils import tidy_path -from urllib import unquote import ssl import traceback @@ -167,7 +165,6 @@ from Logger import init_logger from Logger import destroy_logger -from Utils import ESCAPED_PATH_SEPARATOR from Utils import ESCAPED_WIN_PATH_SEPARATOR from Utils import ESCAPED_NUX_PATH_SEPARATOR from Utils import unescape_xml @@ -254,6 +251,7 @@ '__MACOSX', ) + class EXIT_CODE(object): """List of exit codes for post processing """ @@ -271,13 +269,14 @@ class EXIT_CODE(object): NONE = 95 EXIT_CODES = ( - EXIT_CODE.PARCHECK_CURRENT, - EXIT_CODE.PARCHECK_ALL, - EXIT_CODE.SUCCESS, - EXIT_CODE.FAILURE, - EXIT_CODE.NONE, + EXIT_CODE.PARCHECK_CURRENT, + EXIT_CODE.PARCHECK_ALL, + EXIT_CODE.SUCCESS, + EXIT_CODE.FAILURE, + EXIT_CODE.NONE, ) + class NZBGetDuplicateMode(object): """Defines Duplicate Mode. This is used when Adding NZB-Files directly """ @@ -286,7 +285,7 @@ class NZBGetDuplicateMode(object): SCORE = u'SCORE' # All NZB-Files regardless of their scores are downloaded - ALL = 'ALL' + ALL = 'ALL' # Force download and disable all duplicate checks. FORCE = 'FORCE' @@ -297,21 +296,25 @@ def __init__(self, code=EXIT_CODE.NONE): # Now for your custom code... self.code = code + class NZBGetSuccess(NZBGetExitException): def __init__(self): super(NZBGetExitException, self).\ __init__(code=EXIT_CODE.SUCCESS) + class NZBGetFailure(NZBGetExitException): def __init__(self): super(NZBGetExitException, self).\ __init__(code=EXIT_CODE.FAILURE) + class NZBGetParCheckCurrent(NZBGetExitException): def __init__(self): super(NZBGetExitException, self).\ __init__(code=EXIT_CODE.PARCHECK_CURRENT) + class NZBGetParCheckAll(NZBGetExitException): def __init__(self): super(NZBGetExitException, self).\ @@ -354,88 +357,88 @@ class Health(tuple): # Downloaded and par-checked or unpacked successfully. All # post-processing scripts were successful. The download is # completely OK. - u'ALL': {}, # Use all defaults + u'ALL': {}, # Use all defaults # The download was marked as good using mark(Mark.GOOD) - u'GOOD': {}, # Use all defaults + u'GOOD': {}, # Use all defaults # Download was successful, download health is 100.0%. No par-check # was made (there are no par-files). No unpack was made (there are # no archive files or unpack was disabled for that download or # globally). - u'HEALTH': {}, # Use all defaults + u'HEALTH': {}, # Use all defaults # The hidden history item has status SUCCESS. - u'HIDDEN': {}, # Use all defaults + u'HIDDEN': {}, # Use all defaults # Similar to SUCCESS/ALL but no post-processing scripts were # executed. Downloaded and par-checked successfully. No unpack was # made (there are no archive files or unpack was disabled for that # download or globally). - u'PAR': {}, # Use all defaults + u'PAR': {}, # Use all defaults # Similar to SUCCESS/ALL but no post-processing scripts were # executed. Downloaded and unpacked successfully. Par-check was # successful or was not necessary. - u'UNPACK': {}, # Use all defaults + u'UNPACK': {}, # Use all defaults }, WARNING: { DEFAULT_SUB: {u'has_archive': True, u'is_unpacked': False, }, # Par-check is required by is disabled in settings # (option ParCheck=Manual). - u'DAMAGED': {}, # Use all defaults + u'DAMAGED': {}, # Use all defaults # Download health is below 100.0%. No par-check was made (there # are no par-files). No unpack was made (there are no archive # files or unpack was disabled for that download or globally). - u'HEALTH': {}, # Use all defaults + u'HEALTH': {}, # Use all defaults # The hidden history item has status FAILURE. - u'HIDDEN': {}, # Use all defaults + u'HIDDEN': {}, # Use all defaults # Unpack has failed because the password was not provided or was # wrong. Only for rar5-archives. - u'PASSWORD': {}, # Use all defaults + u'PASSWORD': {}, # Use all defaults # Par-check has detected damage and has downloaded additional # par-files but the repair is disabled in settings # (option ParRepair=no). - u'REPAIRABLE': {}, # Use all defaults + u'REPAIRABLE': {}, # Use all defaults # The URL was fetched successfully but an error occurred during # scanning of the downloaded file. The downloaded file isn't a # proper nzb-file. This status usually means the web-server has # returned an error page (HTML page) instead of the nzb-file. - u'SCAN': {}, # Use all defaults + u'SCAN': {}, # Use all defaults # Downloaded successfully. Par-check and unpack were either # successful or were not performed. At least one of the # post-processing scripts has failed. - u'SCRIPT': {}, # Use all defaults + u'SCRIPT': {}, # Use all defaults # The URL was fetched successfully but downloaded file was not # nzb-file and was skipped by the scanner. - u'SKIPPED': {}, # Use all defaults + u'SKIPPED': {}, # Use all defaults # Unpack has failed due to not enough space on the drive. - u'SPACE': {}, # Use all defaults + u'SPACE': {}, # Use all defaults }, FAILURE: { DEFAULT_SUB: {u'has_archive': True, u'is_unpacked': False, }, # The download was marked as good using mark(Mark.BAD) - u'BAD': {}, # Use all defaults + u'BAD': {}, # Use all defaults # The download was aborted by history check. # Usual case is: download health is below critical health. No # par-check was made (there are no par-files). No unpack was made # (there are no archive files or unpack was disabled for that # download or globally). - u'HEALTH': {}, # Use all defaults + u'HEALTH': {}, # Use all defaults # An error has occurred when moving files from intermediate # directory into the final destination directory. - u'MOVE': {}, # Use all defaults + u'MOVE': {}, # Use all defaults # The par-check has failed. - u'PAR': {}, # Use all defaults + u'PAR': {}, # Use all defaults # The unpack has failed and there are no par-files. - u'UNPACK': {}, # Use all defaults + u'UNPACK': {}, # Use all defaults }, DELETED: { DEFAULT_SUB: {u'has_archive': False, u'is_unpacked': False, }, # The download was deleted by duplicate check. - u'DUPE': {}, # Use all defaults + u'DUPE': {}, # Use all defaults # Fetching of the URL has failed. - u'FETCH': {}, # Use all defaults + u'FETCH': {}, # Use all defaults # The download was manually deleted by user. - u'MANUAL': {}, # Use all defaults + u'MANUAL': {}, # Use all defaults } } @@ -456,7 +459,7 @@ def __new__(self, health): elif not isinstance(health, (tuple, list)): health = (category, subcategory) - health = [ h.upper() for h in filter(bool, health) ] + health = [h.upper() for h in filter(bool, health)] try: if health[0] in Health.HEALTH_MAP: @@ -550,6 +553,33 @@ class PRIORITY(object): # are found in the environment, they are saved to the `config` dictionary SHR_ENVIRO_ID = u'NZBR_' +# Environment ID used when calling tests commands from NZBGet +""" +For example... the below would attempt to execute the function +action_ConnectionTest + +If that didn't exist, it would attempt to execute action_connectiontest +and if that didn't exist, nothing would happen. + +But the point is, it's very easy to simply add the below code and create +a function map to it. This is a new feature introduced after NZBGet v18 +############################################################################ +### OPTIONS ### + +# +# To check connection parameters click the button. +# ConnectionTest@Send Test E-Mail +# +# +# ... + +""" +TST_ENVIRO_ID = u'NZBCP_' + +# The Key Environment Variable that is used to dermine the Test command +# to call (called from NZBGet's Configuration Screen) +TEST_COMMAND = u'%sCOMMAND' % TST_ENVIRO_ID + # Environment ID used when pushing common variables to the server PUSH_ENVIRO_ID = u'NZBPR_' @@ -568,6 +598,7 @@ class PRIORITY(object): SYS_OPTS_RE = re.compile('^%s([A-Z0-9_]+)$' % SYS_ENVIRO_ID) CFG_OPTS_RE = re.compile('^%s([A-Z0-9_]+)$' % CFG_ENVIRO_ID) SHR_OPTS_RE = re.compile('^%s([A-Z0-9_]+)$' % SHR_ENVIRO_ID) +TST_OPTS_RE = re.compile('^%s([A-Z0-9_]+)$' % TST_ENVIRO_ID) DNZB_OPTS_RE = re.compile('^%s%s([A-Z0-9_]+)$' % ( SHR_ENVIRO_ID, SHR_ENVIRO_DNZB_ID, @@ -581,15 +612,15 @@ class PRIORITY(object): # used. GUESS_KEY_MAP = { u'AUDIOCHANNELS': u'audioChannels', u'AUDIOCODEC': u'audioCodec', - u'AUDIOPROFILE': u'audioProfile', u'BONUSNUMBER':u'bonusNumber', - u'BONUSTITLE': u'bonusTitle', u'CONTAINER':u'container', u'DATE': u'date', + u'AUDIOPROFILE': u'audioProfile', u'BONUSNUMBER': u'bonusNumber', + u'BONUSTITLE': u'bonusTitle', u'CONTAINER': u'container', u'DATE': u'date', u'EDITION': u'edition', u'EPISODENUMBER': u'episodeNumber', u'FILMNUMBER': u'filmNumber', u'FILMSERIES': u'filmSeries', u'FORMAT': u'format', u'LANGUAGE': u'language', u'RELEASEGROUP': u'releaseGroup', u'SCREENSIZE': u'screenSize', u'SEASON': u'season', u'SERIES': u'series', u'SPECIAL': u'special', u'SUBTITLELANGUAGE': u'subtitleLanguage', u'TITLE': u'title', - u'TYPE': u'type', u'VIDEOCODEC': u'videoCodec',u'VTYPE': u'vtype', + u'TYPE': u'type', u'VIDEOCODEC': u'videoCodec', u'VTYPE': u'vtype', u'WEBSITE': u'website', u'YEAR': u'year', } @@ -616,6 +647,7 @@ class PRIORITY(object): VALID_HOST_RE = re.compile(r'^[\s]*([^:/\s]+)') VALID_QUERY_RE = re.compile(r'^(.*[/\\])([^/\\]*)$') + class SCRIPT_MODE(object): # After the download of nzb-file is completed NZBGet can call # post-processing scripts (pp-scripts). The scripts can perform further @@ -654,6 +686,12 @@ class SCRIPT_MODE(object): # `ScriptDir`, then choose them in the option `FeedX.Script`. FEED = u'feed' + # To activate a test call to the script, we look for NZBCP_ + # entries. These are populated through calls made available thorugh the + # configuration portion of NZBGet. v1.8 introduced the ability to + # test if your configuration is set up okay. + CONFIG_ACTION = u'action' + # None is detected if you aren't using one of the above types NONE = '' @@ -663,14 +701,18 @@ class SCRIPT_MODE(object): # The order these are listed is very important, # it identifies the order when preforming sanity # checking + SCRIPT_MODE.CONFIG_ACTION, SCRIPT_MODE.POSTPROCESSING, SCRIPT_MODE.SCAN, SCRIPT_MODE.QUEUE, SCRIPT_MODE.SCHEDULER, SCRIPT_MODE.FEED, + + # None should always be the last entry SCRIPT_MODE.NONE, ) + class ScriptBase(object): """The intent is this is the script you run from within your script after overloading the main() function of your class @@ -687,6 +729,10 @@ def __init__(self, logger=True, debug=False, script_mode=None, # Initialize the default character set self.charset = None + # If a configuration test is being executed, this points to the + # function itself. Otherwise this is set to None. + self._config_action = None + # API by default is not configured; it is set up when a call to # an api function is made. self.api = None @@ -722,17 +768,22 @@ def __init__(self, logger=True, debug=False, script_mode=None, self.database_key = database_key # Fetch System Environment (passed from NZBGet) - self.system = dict([(SYS_OPTS_RE.match(k).group(1), v.strip()) \ + self.system = dict([(SYS_OPTS_RE.match(k).group(1), v.strip()) for (k, v) in environ.items() if SYS_OPTS_RE.match(k)]) # Fetch/Load Script Specific Configuration - self.config = dict([(CFG_OPTS_RE.match(k).group(1), v.strip()) \ + self.config = dict([(CFG_OPTS_RE.match(k).group(1), v.strip()) for (k, v) in environ.items() if CFG_OPTS_RE.match(k)]) # Fetch/Load Shared Configuration through push() - self.shared = dict([(SHR_OPTS_RE.match(k).group(1), v.strip()) \ + self.shared = dict([(SHR_OPTS_RE.match(k).group(1), v.strip()) for (k, v) in environ.items() if SHR_OPTS_RE.match(k)]) + # Fetch/Load Test/Command Specific Configuration; This is used + # when issuing commands to a script from the configuration screen + self.test = dict([(TST_OPTS_RE.match(k).group(1), v.strip()) \ + for (k, v) in environ.items() if TST_OPTS_RE.match(k)]) + # Preload nzbheaders based on any DNZB environment variables self.nzbheaders = self.pull_dnzb() @@ -765,7 +816,6 @@ def __init__(self, logger=True, debug=False, script_mode=None, self.debug = self.parse_bool( self.config.get('DEBUG', False)) - # Enabling Character Set as a flag by specifying in the configuration # section of your script #CharSet=no @@ -859,6 +909,9 @@ def __init__(self, logger=True, debug=False, script_mode=None, for k, v in self.shared.items(): self.logger.vvdebug('%s%s=%s' % (SHR_ENVIRO_ID, k, v)) + for k, v in self.test.items(): + self.logger.vvdebug('%s%s=%s' % (TST_ENVIRO_ID, k, v)) + # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # Enforce system/global variables for script processing # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- @@ -871,7 +924,7 @@ def __init__(self, logger=True, debug=False, script_mode=None, environ['%sDEBUG' % SYS_ENVIRO_ID] = NZBGET_BOOL_FALSE if script_mode is not None: - if script_mode in self.script_dict.keys() + [SCRIPT_MODE.NONE,]: + if script_mode in self.script_dict.keys() + [SCRIPT_MODE.NONE, ]: self.script_mode = script_mode if self.script_mode is SCRIPT_MODE.NONE: self.logger.debug('Script mode forced off.') @@ -906,7 +959,7 @@ def __init__(self, logger=True, debug=False, script_mode=None, chdir(self.tempdir) except OSError: self.logger.warning( - 'Temporary directory is not ' + 'accessible: %s' % \ + 'Temporary directory is not ' + 'accessible: %s' % self.tempdir, ) @@ -931,7 +984,7 @@ def __init__(self, logger=True, debug=False, script_mode=None, # we just gracefully move on if this happens pass - def is_unique_instance(self, pidfile=None,die_on_fail=True, + def is_unique_instance(self, pidfile=None, die_on_fail=True, verbose=True): """ Writes a PID file if one is not already present and returns @@ -954,7 +1007,7 @@ def _pid_running(pid): otherwise it returns True if it was found running. """ try: - kill(pid, 0) + kill(pid, 0) except OSError: return False return True @@ -963,9 +1016,9 @@ def _pid_running(pid): self.pidfile = pidfile if not self.pidfile: - self.pidfile = join(self.tempdir, '.run', '%s-%s.pid' % ( - __name__, self.script_mode, - )) + self.pidfile = join(self.tempdir, '.run', '%s-%s.pid' % ( + __name__, self.script_mode, + )) if self.pidfile_tstamp is not None: # PID-File already created and running; test @@ -1040,7 +1093,7 @@ def _pid_running(pid): self.pid, )) - except (ValueError, TypeError), e: + except (ValueError, TypeError): # Bad data if verbose: self.logger.info( @@ -1106,7 +1159,7 @@ def _pid_running(pid): return False try: - fp.write("%s" % str(self.pid)) + fp.write("%s" % str(self.pid)) except: if verbose: self.logger.warning('Could not write PID into PID-File.') @@ -1148,7 +1201,7 @@ def _pid_running(pid): # We wrote our PID file successfully if verbose: self.logger.info( - 'Created PID-File: %s (pid=%d)' % ( + 'Created PID-File: %s (pid=%d)' % ( self.pidfile, self.pid, )) return True @@ -1482,7 +1535,6 @@ def parse_nzbcontent(self, nzbcontent): return results - def parse_url(self, url, default_schema='http', qsd_auth=True): """A function that greatly simplifies the parsing of a url specified by the end user. @@ -1580,7 +1632,7 @@ def parse_url(self, url, default_schema='http', qsd_auth=True): # Parse Query Arugments ?val=key&key=val # while ensureing that all keys are lowercase if qsdata: - result['qsd'] = dict([ (k.lower().strip(), v.strip()) \ + result['qsd'] = dict([(k.lower().strip(), v.strip()) \ for k, v in parse_qsl( qsdata, keep_blank_values=True, @@ -1719,7 +1771,7 @@ def set(self, key, value, use_env=True, use_db=True): self.logger.debug('unset(database) %s"' % key) elif isinstance(value, bool): - # Convert boolean to integer (change True to 1 or False to 0) + # Convert boolean to integer (True to 1 or False to 0) self.database.set(key=key, value=int(value)) self.logger.debug('set(database) %s="%s"' % ( key, @@ -1969,7 +2021,7 @@ def nzb_set(self, key, value, use_env=True, use_db=True): self.logger.debug('nzb_unset(database) %s"' % key) elif isinstance(value, bool): - # Convert boolean to integer (change True to 1 or False to 0) + # Convert boolean to integer (True to 1 or False to 0) self.database.set( key=key, value=int(value), category=Category.NZB) self.logger.debug('nzb_set(database) %s="%s"' % ( @@ -1980,7 +2032,8 @@ def nzb_set(self, key, value, use_env=True, use_db=True): else: self.database.set( key=key, value=value, category=Category.NZB) - self.logger.debug('nzb_set(database) %s="%s"' % (key, value)) + self.logger.debug( + 'nzb_set(database) %s="%s"' % (key, value)) except EnvironmentError: # Database Access Problem @@ -2001,7 +2054,11 @@ def nzb_set(self, key, value, use_env=True, use_db=True): elif isinstance(value, bool): # Convert boolean to integer (change True to 1 or False to 0) - self.database.set(key=key, value=int(value), category=Category.NZB) + self.database.set( + key=key, + value=int(value), + category=Category.NZB, + ) self.logger.debug('nzb_set(database) %s="%s"' % ( key, int(value), @@ -2039,7 +2096,7 @@ def nzb_set(self, key, value, use_env=True, use_db=True): if use_env: if isinstance(value, bool): - # Convert boolean to integer (change True to 1 or False to 0) + # Convert boolean to integer (True to 1 or False to 0) value = str(int(value)) elif not isinstance(value, basestring): @@ -2088,7 +2145,8 @@ def nzb_get(self, key, default=None, use_db=True): value = self.database.get(key=key, category=Category.NZB) if value is not None: # only return if a key was found - self.logger.debug('nzb_get(database) %s="%s"' % (key, value)) + self.logger.debug( + 'nzb_get(database) %s="%s"' % (key, value)) return value except EnvironmentError: @@ -2352,7 +2410,6 @@ def api_connect(self, user=None, password=None, return True - # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # Retrieve System Logs # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- @@ -2863,9 +2920,14 @@ def run(self, *args, **kwargs): # - scheduler_main() # - queue_main() # - feed_main() - # + # - action_() + + if self.script_mode is SCRIPT_MODE.CONFIG_ACTION: + # Line up our action_() script + main_function = self._config_action + # otherwise main() is executed - if hasattr(self, '%s_%s' % (self.script_mode, 'main')): + elif hasattr(self, '%s_%s' % (self.script_mode, 'main')): main_function = getattr( self, '%s_%s' % (self.script_mode, 'main')) @@ -3091,6 +3153,64 @@ def parse_bool(self, arg, default=False): # Handle other types return bool(arg) + def action_sanity_check(self): + """Sanity checking to ensure this really is a Config Test + """ + + if TEST_COMMAND not in environ: + # Nothing more to do + return False + + # Extract our content + command = environ.get(TEST_COMMAND) + if not command: + # Nothing more to do + return False + + if hasattr(self, '%s_%s' % (SCRIPT_MODE.CONFIG_ACTION, command)): + self._config_action = getattr(self, '%s_%s' % ( + SCRIPT_MODE.CONFIG_ACTION, + command, + )) + + if not callable(self._config_action): + self.logger.debug('The internal script variable '\ + '%s is not a function (type=%s)' % ( + (SCRIPT_MODE.CONFIG_ACTION, command()), + type(self._config_action), + )) + + # Reset it's variable + self._config_action = None + return False + + # We're set + return True + + elif hasattr(self, '%s_%s' % ( + SCRIPT_MODE.CONFIG_ACTION, command.lower())): + self._config_action = getattr(self, '%s_%s' % ( + SCRIPT_MODE.CONFIG_ACTION, + command.lower(), + )) + + if not callable(self._config_action): + self.logger.debug('The internal script variable '\ + '%s is not a function (type=%s)' % ( + (SCRIPT_MODE.CONFIG_ACTION, command.lower()), + type(self._config_action), + )) + # Reset it's variable + self._config_action = None + return False + + # We're set + return True + + self.logger.warning('The developer of this script did not'\ + ' create test mapping to this command.') + return False + def detect_mode(self): """ Attempt to detect the script mode based on environment variables @@ -3106,7 +3226,8 @@ def detect_mode(self): if len(self.script_dict.keys()): for k in [ v for v in SCRIPT_MODES \ - if v in self.script_dict.keys() + [SCRIPT_MODE.NONE,]]: + if v in self.script_dict.keys() + [ + SCRIPT_MODE.CONFIG_ACTION, SCRIPT_MODE.NONE,]]: if hasattr(self, '%s_%s' % (k, 'sanity_check')): if getattr(self, '%s_%s' % (k, 'sanity_check'))(): self.script_mode = k diff --git a/Notify/nzbget/__init__.py b/Notify/nzbget/__init__.py index d893353..776a33e 100644 --- a/Notify/nzbget/__init__.py +++ b/Notify/nzbget/__init__.py @@ -2,7 +2,7 @@ # # simplify importing of comonly used modules # -# Copyright (C) 2014 Chris Caron +# Copyright (C) 2014-2017 Chris Caron # # This program is free software; you can redistribute it and/or modify it # under the terms of the GNU Lesser General Public License as published by @@ -15,13 +15,14 @@ # GNU Lesser General Public License for more details. # __title__ = 'pynzbget' -__version__ = '0.2.1' +__version__ = '0.5.1' __author__ = 'Chris Caron ' __license__ = 'GPLv3' -__copywrite__ = 'Copyright 2014 Chris Caron