Source code for projectroles.plugins

"""Plugin point definitions and plugin API for apps based on projectroles"""

from django.conf import settings
from djangoplugins.point import PluginPoint


# Local costants
PLUGIN_TYPES = {
    'project_app': 'ProjectAppPluginPoint',
    'backend': 'BackendPluginPoint',
    'site_app': 'SiteAppPluginPoint',
}

# From djangoplugins
ENABLED = 0
DISABLED = 1
REMOVED = 2


# Plugin Mixins ----------------------------------------------------------------


[docs] class ProjectModifyPluginMixin: """ Mixin for project plugin API extensions for additional actions to be performed for project and role modifications. Used if e.g. updating external resources based on SODAR Core projects. Add this into your project app or backend plugin if you want to implement additional modification features. It is not supported on site app plugins. """
[docs] def perform_project_modify( self, project, action, project_settings, old_data=None, old_settings=None, request=None, ): """ Perform additional actions to finalize project creation or update. :param project: Current project object (Project) :param action: Action to perform (CREATE or UPDATE) :param project_settings: Project app settings (dict) :param old_data: Old project data in case of an update (dict or None) :param old_settings: Old app settings in case of update (dict or None) :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def revert_project_modify( self, project, action, project_settings, old_data=None, old_settings=None, request=None, ): """ Revert project creation or update if errors have occurred in other apps. :param project: Current project object (Project) :param action: Action which was performed (CREATE or UPDATE) :param project_settings: Project app settings (dict) :param old_data: Old project data in case of update (dict or None) :param old_settings: Old app settings in case of update (dict or None) :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def perform_role_modify(self, role_as, action, old_role=None, request=None): """ Perform additional actions to finalize role assignment creation or update. :param role_as: RoleAssignment object :param action: Action to perform (CREATE or UPDATE) :param old_role: Role object for previous role in case of an update :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def revert_role_modify(self, role_as, action, old_role=None, request=None): """ Revert role assignment creation or update if errors have occurred in other apps. :param role_as: RoleAssignment object :param action: Action which was performed (CREATE or UPDATE) :param old_role: Role object for previous role in case of an update :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def perform_role_delete(self, role_as, request=None): """ Perform additional actions to finalize role assignment deletion. :param role_as: RoleAssignment object :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def revert_role_delete(self, role_as, request=None): """ Revert role assignment deletion deletion if errors have occurred in other apps. :param role_as: RoleAssignment object :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def perform_owner_transfer( self, project, new_owner, old_owner, old_owner_role, request=None ): """ Perform additional actions to finalize project ownership transfer. :param project: Project object :param new_owner: SODARUser object for new owner :param old_owner: SODARUser object for previous owner :param old_owner_role: Role object for new role of previous owner :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def revert_owner_transfer( self, project, new_owner, old_owner, old_owner_role, request=None ): """ Revert project ownership transfer if errors have occurred in other apps. :param project: Project object :param new_owner: SODARUser object for new owner :param old_owner: SODARUser object for previous owner :param old_owner_role: Role object for new role of previous owner :param request: Request object or None """ # TODO: Implement this in your app plugin pass
[docs] def perform_project_sync(self, project): """ Synchronize existing projects to ensure related data exists when the syncmodifyapi management comment is called. Should mostly be used in development when the development databases have been e.g. modified or recreated. :param project: Current project object (Project) """ # TODO: Implement this in your app plugin pass
[docs] def perform_project_setting_update( self, plugin_name, setting_name, value, old_value, project=None, user=None, ): """ Perform additional actions when updating a single app setting with PROJECT scope. :param plugin_name: Name of app plugin for the setting, "projectroles" is used for projectroles settings (string) :param setting_name: Setting name (string) :param value: New value for setting :param old_value: Previous value for setting :param project: Project object or None :param user: User object or None """ # TODO: Implement this in your app plugin pass
[docs] def revert_project_setting_update( self, plugin_name, setting_name, value, old_value, project=None, user=None, ): """ Revert updating a single app setting with PROJECT scope if errors have occurred in other apps. :param plugin_name: Name of app plugin for the setting, "projectroles" is used for projectroles settings (string) :param setting_name: Setting name (string) :param value: New value for setting :param old_value: Previous value for setting :param project: Project object or None :param user: User object or None """ # TODO: Implement this in your app plugin pass
[docs] def perform_project_archive( self, project, ): """ Perform additional actions to finalize project archiving or unarchiving. The state being applied can be derived from the project.archive attr. :param project: Project object (Project) """ # TODO: Implement this in your app plugin pass
[docs] def revert_project_archive( self, project, ): """ Revert project archiving or unarchiving if errors have occurred in other apps. The state being originally set can be derived from the project.archive attr. :param project: Project object (Project) """ # TODO: Implement this in your app plugin pass
# Plugin Points ----------------------------------------------------------------
[docs] class ProjectAppPluginPoint(PluginPoint): """Projectroles plugin point for registering project specific apps""" #: UI URLs urls = [] #: App settings definition #: #: Example :: #: #: app_settings = { #: 'example_setting': { #: 'scope': 'PROJECT', # PROJECT/USER/SITE #: 'type': 'STRING', # STRING/INTEGER/BOOLEAN #: 'default': 'example', #: 'label': 'Project setting', # Optional, defaults to name/key #: 'placeholder': 'Enter example setting here', # Optional #: 'description': 'Example project setting', # Optional #: 'options': ['example', 'example2'], # Optional, only for #: settings of type STRING or INTEGER #: 'user_modifiable': True, # Optional, show/hide in forms #: 'local': False, # Optional, show/hide in forms on target #: site, sync value from source to target site if false #: 'project_types': [PROJECT_TYPE_PROJECT], # Optional, may #: contain PROJECT_TYPE_CATEGORY and/or PROJECT_TYPE_PROJECT #: } #: } # TODO: Define project specific settings in your app plugin, example above app_settings = {} # DEPRECATED, will be removed in the next SODAR Core release project_settings = {} #: Iconify icon # TODO: Implement this in your app plugin icon = 'mdi:help-rhombus-outline' #: Entry point URL ID (must take project sodar_uuid as "project" argument) # TODO: Implement this in your app plugin entry_point_url_id = 'home' #: Description string # TODO: Implement this in your app plugin description = 'TODO: Write a description for your plugin' #: Required permission for accessing the app # TODO: Implement this in your app plugin (can be None) app_permission = None #: Enable or disable general search from project title bar # TODO: Implement this in your app plugin search_enable = False #: List of search object types for the app # TODO: Implement this in your app plugin search_types = [] #: Search results template # TODO: Implement this in your app plugin search_template = None #: App card template for the project details page # TODO: Implement this in your app plugin details_template = None #: App card title for the project details page # TODO: Implement this in your app plugin (can be None) details_title = None #: Position in plugin ordering # TODO: Implement this in your app plugin (must be an integer) plugin_ordering = 50 #: Optional custom project list column definition #: #: Example :: #: #: project_list_columns = { #: 'column_id': { #: 'title': 'Column Title', #: 'width': 100, # Desired width of column in pixels #: 'description': 'Description', # Optional description string #: 'active': True, # Boolean for whether the column is active #: 'ordering': 50, # Integer for ordering the columns #: 'align': 'left' # Alignment of content #: } #: } # TODO: Define project list column data in your app plugin (optional) project_list_columns = {} #: Display application for categories in addition to projects # TODO: Override this in your app plugin if needed category_enable = False #: Names of plugin specific Django settings to display in siteinfo # TODO: Override this in your app plugin if needed info_settings = []
[docs] def get_object(self, model, uuid): """ Return object based on a model class and the object's SODAR UUID. :param model: Object model class :param uuid: sodar_uuid of the referred object :return: Model object or None if not found :raise: NameError if model is not found """ try: return model.objects.get(sodar_uuid=uuid) except model.DoesNotExist: return None
[docs] def search(self, search_terms, user, search_type=None, keywords=None): """ Return app items based on one or more search terms, user, optional type and optional keywords. :param search_terms: Search terms to be joined with the OR operator (list of strings) :param user: User object for user initiating the search :param search_type: String :param keywords: List (optional) :return: List of PluginSearchResult objects """ # TODO: Implement this in your app plugin # TODO: Implement display of results in the app's search template return []
[docs] def update_cache(self, name=None, project=None, user=None): """ Update cached data for this app, limitable to item ID and/or project. :param name: Item name to limit update to (string, optional) :param project: Project object to limit update to (optional) :param user: User object to denote user triggering the update (optional) """ # TODO: Implement this in your app plugin return None
[docs] def get_statistics(self): """ Return app statistics as a dict. Should take the form of {id: {label, value, url (optional), description (optional)}}. :return: Dict """ # TODO: Implement this in your app plugin return {}
[docs] def get_project_list_value(self, column_id, project, user): """ Return a value for the optional additional project list column specific to a project. :param column_id: ID of the column (string) :param project: Project object :param user: User object (current user) :return: String (may contain HTML), integer or None """ # TODO: Implement this in your app plugin (optional) return None
[docs] def validate_form_app_settings(self, app_settings, project=None, user=None): """ Validate app settings form data and return a dict of errors. :param app_settings: Dict of app settings :param project: Project object or None :param user: User object or None :return: dict in format of {setting_name: 'Error string'} """ # TODO: Implement this in your app plugin (optional) return None
[docs] class BackendPluginPoint(PluginPoint): """Projectroles plugin point for registering backend apps""" #: Iconify icon # TODO: Implement this in your backend plugin icon = 'mdi:help-rhombus-outline' #: Description string # TODO: Implement this in your backend plugin description = 'TODO: Write a description for your plugin' #: URL of optional javascript file to be included # TODO: Implement this in your backend plugin if applicable javascript_url = None #: URL of optional css file to be included # TODO: Implement this in your backend plugin if applicable css_url = None #: Names of plugin specific Django settings to display in siteinfo # TODO: Override this in your app plugin if needed info_settings = []
[docs] def get_api(self): """Return API entry point object.""" # TODO: Implement this in your backend plugin raise NotImplementedError
[docs] def get_statistics(self): """ Return backend statistics as a dict. Should take the form of {id: {label, value, url (optional), description (optional)}}. :return: Dict """ # TODO: Implement this in your backend plugin return {}
[docs] def get_object(self, model, uuid): """ Return object based on a model class and the object's SODAR UUID. :param model: Object model class :param uuid: sodar_uuid of the referred object :return: Model object or None if not found :raise: NameError if model is not found """ try: return model.objects.get(sodar_uuid=uuid) except model.DoesNotExist: return None
[docs] class SiteAppPluginPoint(PluginPoint): """Projectroles plugin point for registering site-wide apps""" #: Iconify icon # TODO: Implement this in your site app plugin icon = 'mdi:help-rhombus-outline' #: Description string # TODO: Implement this in your site app plugin description = 'TODO: Write a description for your plugin' #: Entry point URL ID # TODO: Implement this in your app plugin entry_point_url_id = 'home' #: Required permission for displaying the app # TODO: Implement this in your site app plugin (can be None) app_permission = None #: User settings definition #: #: Example :: #: #: app_settings = { #: 'example_setting': { #: 'scope' : 'USER', # always USER #: 'type': 'STRING', # STRING/INTEGER/BOOLEAN #: 'default': 'example', #: 'placeholder': 'Enter example setting here', # Optional #: 'description': 'Example user setting', # Optional #: 'options': ['example', 'example2'], # Optional, only for #: settings of type STRING or INTEGER #: 'user_modifiable': True, # Optional, show/hide in forms #: 'local': False, # Optional, show/hide in forms on target #: site #: } #: } # TODO: Define user specific settings in your app plugin, example above app_settings = {} #: List of names for plugin specific Django settings to display in siteinfo # TODO: Override this in your app plugin if needed info_settings = []
[docs] def get_statistics(self): """ Return app statistics as a dict. Should take the form of {id: {label, value, url (optional), description (optional)}}. :return: Dict """ # TODO: Implement this in your app plugin return {}
[docs] def get_messages(self, user=None): """ Return a list of messages to be shown to users. :param user: User object (optional) :return: List of dicts or and empty list if no messages """ ''' Example of valid return data: return [{ 'content': 'Message content in here, can contain html', 'color': 'info', # Corresponds to bg-* in Bootstrap 'dismissable': True # False for non-dismissable }] ''' # TODO: Implement this in your site app plugin return []
[docs] def get_object(self, model, uuid): """ Return object based on a model class and the object's SODAR UUID. :param model: Object model class :param uuid: sodar_uuid of the referred object :return: Model object or None if not found :raise: NameError if model is not found """ try: return model.objects.get(sodar_uuid=uuid) except model.DoesNotExist: return None
[docs] def validate_form_app_settings(self, app_settings, user=None): """ Validate app settings form data and return a dict of errors. :param app_settings: Dict of app settings :param user: User object or None :return: dict in format of {setting_name: 'Error string'} """ # TODO: Implement this in your app plugin (optional) return None
# Data Classes -----------------------------------------------------------------
[docs] class PluginSearchResult: """ Class representing a list of search results from a specific plugin for one or more search types. Expected to be returned from search() implementations. """ #: Category of the result set, used in templates (string) category = None #: Title to be displayed for this set of search results in the UI (string) title = None #: List of one or more search type keywords for these results search_types = [] #: List or QuerySet of result objects items = [] def __init__(self, category, title, search_types, items): """ Initialize PluginSearchResult. :param category: Category of the result set, used in templates (string) :param title: Title to be displayed for this set of search results in the UI (string) :param search_types: List of one or more search type keywords for the results :param items: List or QuerySet of result objects """ self.category = category self.title = title self.search_types = search_types if not isinstance(search_types, list) or len(search_types) < 1: raise ValueError( 'At least one type keyword must be provided in search_types' ) self.items = items
# Plugin API -------------------------------------------------------------------
[docs] def get_active_plugins(plugin_type='project_app', custom_order=False): """ Return active plugins of a specific type. :param plugin_type: "project_app", "site_app" or "backend" (string) :param custom_order: Order by plugin_ordering for project apps (boolean) :return: List or None :raise: ValueError if plugin_type is not recognized """ if plugin_type not in PLUGIN_TYPES.keys(): raise ValueError( 'Invalid value for plugin_type. Accepted values: {}'.format( ', '.join(PLUGIN_TYPES.keys()) ) ) plugins = eval(PLUGIN_TYPES[plugin_type]).get_plugins() if plugins: ret = [ p for p in plugins if ( p.is_active() and ( plugin_type in ['project_app', 'site_app'] or p.name in settings.ENABLED_BACKEND_PLUGINS ) ) ] return sorted( ret, key=lambda x: ( x.plugin_ordering if custom_order and plugin_type == 'project_app' else x.name ), ) return None
[docs] def change_plugin_status(name, status, plugin_type='app'): """ Change the status of a selected plugin in the database. :param name: Plugin name (string) :param status: Status (int, see djangoplugins) :param plugin_type: Type of plugin ("app", "backend" or "site") :raise: ValueError if plugin_type is invalid or plugin with name not found """ # NOTE: Used to forge plugin to a specific status for e.g. testing if plugin_type == 'app': plugin = ProjectAppPluginPoint.get_plugin(name) elif plugin_type == 'backend': plugin = BackendPluginPoint.get_plugin(name) elif plugin_type == 'site': plugin = SiteAppPluginPoint.get_plugin(name) else: raise ValueError('Invalid plugin_type: "{}"'.format(plugin_type)) if not plugin: raise ValueError( 'Plugin of type "{}" not found with name "{}"'.format( plugin_type, name ) ) plugin = plugin.get_model() plugin.status = status plugin.save()
[docs] def get_app_plugin(plugin_name, plugin_type=None): """ Return active app plugin. :param plugin_name: Plugin name (string) :param plugin_type: Plugin type (string or None for all types) :return: Plugin object or None if not found or inactive """ if plugin_type: plugin_types = [PLUGIN_TYPES[plugin_type]] else: plugin_types = PLUGIN_TYPES.values() for t in plugin_types: try: return eval(t).get_plugin(plugin_name) except Exception: pass
[docs] def get_backend_api(plugin_name, force=False, **kwargs): """ Return backend API object. NOTE: May raise an exception from plugin.get_api(). :param plugin_name: Plugin name (string) :param force: Return plugin regardless of status in ENABLED_BACKEND_PLUGINS :param kwargs: Optional kwargs for API :return: Plugin object or None if not found """ if plugin_name in settings.ENABLED_BACKEND_PLUGINS or force: try: plugin = BackendPluginPoint.get_plugin(plugin_name) except BackendPluginPoint.DoesNotExist: return None return plugin.get_api(**kwargs) if plugin.is_active() else None
# Plugins within projectroles --------------------------------------------------
[docs] class RemoteSiteAppPlugin(SiteAppPluginPoint): """Site plugin for remote site and project management""" #: Name (used as plugin ID) name = 'remotesites' #: Title (used in templates) title = 'Remote Site Access' #: UI URLs urls = [] #: Iconify icon icon = 'mdi:cloud-sync' #: Description string description = 'Management of remote SODAR sites and remote project access' #: Entry point URL ID entry_point_url_id = 'projectroles:remote_sites' #: Required permission for displaying the app app_permission = 'userprofile.update_remote'