''' xbmcswift2.module ----------------- This module contains the Module Class. :copyright: (c) 2012 by Jonathan Beluch :license: GPLv3, see LICENSE for more details. ''' from xbmcswift2.xbmcmixin import XBMCMixin from xbmcswift2 import setup_log class Module(XBMCMixin): '''Modules are basically mini plugins except they don't have any functionality until they are registered with a Plugin. ''' def __init__(self, namespace): # Get rid of package prefixes self._namespace = namespace.split('.')[-1] self._view_functions = {} self._routes = [] self._register_funcs = [] self._plugin = None self._url_prefix = None # TODO: Think of a better log name self._log = setup_log(namespace) # TODO: add setter for plugin during registration @property def plugin(self): '''Returns the plugin this module is registered to, or raises a RuntimeError if not registered. ''' if self._plugin is None: # TODO: print called method in the error message raise RuntimeError('Module must be registered in order to call' 'this method.') return self._plugin @property def cache_path(self): '''Returns the module's cache_path.''' return self.plugin.cache_path @property def addon(self): '''Returns the module's addon''' return self.plugin.addon @property def added_items(self): '''Returns this module's added_items''' return self.plugin.added_items @property def handle(self): '''Returns this module's handle''' return self.plugin.handle @property def request(self): '''Returns the current request''' return self.plugin.request @property def log(self): '''Returns the registered plugin's log.''' return self._log @property def url_prefix(self): '''Sets or gets the url prefix of the module. Raises an Exception if this module is not registered with a Plugin. ''' if self._url_prefix is None: # TODO: print called method in the error message raise RuntimeError('Module must be registered in order to call' 'this method.') return self._url_prefix def route(self, url_rule, name=None, options=None): '''A decorator to add a route to a view. name is used to differentiate when there are multiple routes for a given view.''' def decorator(func): '''Adds a url rule for the provided function''' view_name = name or func.__name__ self.add_url_rule(url_rule, func, name=view_name, options=options) return func return decorator def url_for(self, endpoint, explicit=False, **items): '''Returns a valid XBMC plugin URL for the given endpoint name. endpoint can be the literal name of a function, or it can correspond to the name keyword arguments passed to the route decorator. Currently, view names must be unique across all plugins and modules. There are not namespace prefixes for modules. ''' # TODO: Enable items to be passed with keywords of other var names # such as endpoing and explicit # TODO: Figure out how to handle the case where a module wants to # call a parent plugin view. if not explicit and not endpoint.startswith(self._namespace): endpoint = '%s.%s' % (self._namespace, endpoint) return self._plugin.url_for(endpoint, **items) def add_url_rule(self, url_rule, view_func, name, options=None): '''This method adds a URL rule for routing purposes. The provided name can be different from the view function name if desired. The provided name is what is used in url_for to build a URL. The route decorator provides the same functionality. ''' name = '%s.%s' % (self._namespace, name) def register_rule(plugin, url_prefix): '''Registers a url rule for the provided plugin and url_prefix. ''' full_url_rule = url_prefix + url_rule plugin.add_url_rule(full_url_rule, view_func, name, options) # Delay actual registration of the url rule until this module is # registered with a plugin self._register_funcs.append(register_rule) def redirect(self, url): '''Used when you need to redirect to another view, and you only have the final plugin:// url.''' return self._plugin._fake_run(url)