Source code for smqtk.web

"""
SMQTK Web Applications
"""

import inspect
import os

import flask

import smqtk.utils
from smqtk.utils import SmqtkObject
from smqtk.utils import plugin
from smqtk.utils import merge_dict


# noinspection PyAbstractClass
[docs]class SmqtkWebApp (SmqtkObject, flask.Flask, smqtk.utils.Configurable, plugin.Pluggable): """ Base class for SMQTK web applications """
[docs] @classmethod def impl_directory(cls): """ :return: Directory in which this implementation is contained. :rtype: str """ return os.path.dirname(os.path.abspath(inspect.getfile(cls)))
[docs] @classmethod def get_default_config(cls): """ Generate and return a default configuration dictionary for this class. This will be primarily used for generating what the configuration dictionary would look like for this class without instantiating it. This should be overridden in each implemented application class to add appropriate configuration. :return: Default configuration dictionary for the class. :rtype: dict """ return { "flask_app": { "SECRET_KEY": "MySuperUltraSecret", "BASIC_AUTH_USERNAME": "demo", "BASIC_AUTH_PASSWORD": "demo" }, "server": { 'host': "127.0.0.1", 'port': 5000 } }
[docs] @classmethod def from_config(cls, config_dict, merge_default=True): """ Override to just pass the configuration dictionary to constructor """ # Repeated from super method due to overriding how constructor is called if merge_default: merged = cls.get_default_config() merge_dict(merged, config_dict) config_dict = merged return cls(config_dict)
def __init__(self, json_config): """ Initialize application based of supplied JSON configuration :param json_config: JSON configuration dictionary :type json_config: dict """ super(SmqtkWebApp, self).__init__( self.__class__.__name__, static_folder=os.path.join(self.impl_directory(), 'static'), template_folder=os.path.join(self.impl_directory(), 'templates') ) # # Configuration setup # self.json_config = json_config # Factor 'flask_app' configuration properties into self.config for k in self.json_config['flask_app']: self.config[k] = self.json_config['flask_app'][k] # # Security # self.secret_key = self.config['SECRET_KEY']
[docs] def get_config(self): return self.json_config
[docs] def run(self, host=None, port=None, debug=False, **options): """ Override of the run method, drawing running host and port from configuration by default. 'host' and 'port' values specified as argument or keyword will override the app configuration. """ super(SmqtkWebApp, self)\ .run(host=(host or self.json_config['server']['host']), port=(port or self.json_config['server']['port']), debug=debug, **options)
def get_web_applications(reload_modules=False): """ Discover and return SmqtkWebApp implementation classes found in the plugin directory. Keys in the returned map are the names of the discovered classes and the paired values are the actual class type objects. We look for modules (directories or files) that start with and alphanumeric character ('_' prefixed files/directories are hidden, but not recommended). Within a module, we first look for a helper variable by the name ``APPLICATION_CLASS``, which can either be a single class object or an iterable of class objects, to be exported. If the variable is set to None, we skip that module and do not import anything. If the variable is not present, we look for a class by the same na e and casing as the module's name. If neither are found, the module is skipped. :param reload_modules: Explicitly reload discovered modules from source. :type reload_modules: bool :return: Map of discovered class objects of type ``SmqtkWebApp`` whose keys are the string names of the classes. :rtype: dict[str, type] """ import os from smqtk.utils.plugin import get_plugins this_dir = os.path.abspath(os.path.dirname(__file__)) env_var = "APPLICATION_PATH" helper_var = "APPLICATION_CLASS" return get_plugins(__name__, this_dir, env_var, helper_var, SmqtkWebApp, reload_modules=reload_modules)