Project

General

Profile

Documentation - Advanced Plugin Helper

Encapsulate presentation logic and Redmine patch management in PORO

Redmine Plugin Version Redmine Version Language Support Version Stage

The Advanced Plugin Helper plugin is a Redmine plugin for developers. It is helping to keep Rails helper and views light in favour of encapsulating presentation login in plain old ruby classes (PORO). It also provides a patch api to easily register Redmine patches for Redmine 4 or 5 as well as an exception notifier.


Plugin Features

The Advanced Plugin Helper addresses Redmine plugin developer seeking to write reusable and easy maintainable code.

Presenter API

  • Inherit presenter classes from AdvancedPluginHelper::BasePresenter.
  • Register the newly created presenter to be loaded with your plugin.

Patch API

  • Register your patch files to be loaded with your plugin under Redmine 4 or 5.
  • Apply custom modifications to be loaded with your plugin under Redmine 4 or 5.

Subclass associations

  • Allows to add superclass associations to subclasses when they are missing, see Rails Issue 20678

Exception notifier

  • Get notified via email when something went wrong in your Redmine instance.

Getting Started

This quick introduction shows you the most convenient way for trying out the plugin with your Redmine instance.

You need a running Redmine instance in order to install the plugin. If you need help with the installation, please refer to Redmine.org.

Downloading the plugin

Attention

Don't clone the default branch: For production you need to clone the master branch explicitly!

Navigate into the /plugins directory of your Redmine instance.

Download the latest version of this plugin and add it as advanced_plugin_helper folder into your plugin directory.

cd (REDMINE_ROOT_DIR)/plugins
git clone -b master https://github.com/xmera-circle/advanced_plugin_helper

Restart Redmine

Navigate into the root directoy of Redmine.

Run the restart Rake task. This will touch restart.txt in order to force Redmine's application server to restart and load
everything from scratch. This will register the new plugin.

cd (REDMINE_ROOT_DIR)
rake restart

Plugin Usage

Create a custom presenter class

First , you need a app/presenters directory in your plugin. When created it will be automatically added to Rails autoload_path when restarting Redmine.

Second , create a presenter class as subclass of AdvancedPluginHelper::BasePresenter like so:

module MyRedminePlugin
  class MyModelPresenter < AdvancedPluginHelper::BasePresenter
    presents :my_model  # you can use 'my_model' as any instance variable of your model
    
    def helper_method_for_my_model
      # method body here
    end
  end
end

The AdvancedPluginHelper::BasePresenter class supports:

Save the file as app/presenters/my_model_presenter.rb.

Third , register your presenter. This could be done in lib/my_redmine_plugin.rb

module MyRedminePlugin
  class << self
    def setup
      AdvancedPluginHelper::Presenter.register('MyRedminePlugin::MyModelPresenter', 'MyModel')
    end
  end
end

Fourth , make sure that MyRedminePlugin.setup is loaded when booting Redmine. For doing so, add this to your init.rb:

require File.expand_path('lib/my_redmine_plugin', __dir__)

Redmine::Plugin.register :my_redmine_plugin do
  name 'My Redmine Plugin'
  author '(your_name)'
  description '(plugin_description)'
  version '(current_version)'
  url '(plugin_url)'
  author_url '(your_github_profile)'

  requires_redmine version_or_higher: '4.2.0'
  requires_redmine_plugin :advanced_plugin_helper, version_or_higher: '0.2.0'
end

MyRedminePlugin.setup

Finally , in your view you can use your presenter like so:

<%= show(@my_model).helper_method_for_my_model %>

Of course, replace my_redmine_plugin and similar placeholder with your plugin name, class name, etc.

The AdvancedPluginHelper::PresentersHelper is added as helper to these classes by default:

  • ActionMailer::Base,
  • ApplicationController,
  • SettingsController,
  • ProjectsController,
  • QueriesController,
  • NewsController.

You can extend the list by adding your class to the array:

AdvancedPluginHelper.klasses << YourKlass

Register your Redmine patches

When you need to patch Redmine code you can register your patch in lib/my_redmine_plugin.rb:

module MyRedminePlugin
  class << self
    def setup
      AdvancedPluginHelper::Patch.register(data)
    end
    
    private

    def data
      { klass: Issue, patch: MyRedminePlugin::Extensions::IssuePatch, strategy: :include }
    end
  end
end

The strategy could be one of :include, :prepend, :helper.

Strategy Recommendation
:include when adding new methods
:prepend when overriding existing methods
:helper when adding a helper module

Apply other modifications

When you need to apply custom modifications, which does not require a patch file, you can do that with this command:

module MyRedminePlugin
  class << self
    def setup
      AdvancedPluginHelper::Patch.apply do
        # your code here
      end
    end
  end
end

IMPORTANT

Make sure that your code is idempotent since Rails/Redmine loads it several times during the apps boot process!

Subclass associations

When you have created a subclass and will try to add associations to the corresponding superclass, you will notice that the subclass won't have that associations. It will have the methods to access the association but you cannot call them. This is a bug in Rails since 2015 (see above).

In order to patch that behavior, you can register your subclass like below. This will create missing associations.

module MyRedminePlugin
  class << self
    def setup
      AdvancedPluginHelper::Associations.register(MySubclass)
    end
  end
end

Configure the exception notifier

# In config/additional_environment.rb:

require File.expand_path('plugins/advanced_plugin_helper/lib/advanced_plugin_helper/notifier', __dir__)
require File.expand_path('plugins/advanced_plugin_helper/lib/exception_notifier/custom_mail_notifier', __dir__)
require 'exception_notification/rails'

ExceptionNotification.configure do |config|
  if AdvancedPluginHelper::Notifier.email_delivery_enabled?
    config.add_notifier :custom_mail, AdvancedPluginHelper::Notifier.custom_mail
    config.error_grouping = AdvancedPluginHelper::Notifier.error_grouping
    config.ignore_if do |_exception, _options|
      AdvancedPluginHelper::Notifier.disabled?
    end
  end
end

Configuration with environment variables

In order to configure sender and recipients set the corresponding env vars:
DEFAULT_EXCEPTION_NOTIFIER_RECIPIENTS (default: nil)
EXCEPTION_NOTIFIER_SENDER (default: Setting.mail_from as set in Administration » Setting » Email notifications)
EXCEPTION_NOTIFIER_RECIPIENTS (default: nil)

You can also disable the notifier via EXCEPTION_NOTIFIER_DISABLED (default: false).

DEFAULT_EXCEPTION_NOTIFIER_RECIPIENTS and EXCEPTION_NOTIFIER_RECIPIENTS will be merged into a single list of recipients.

Configuration with class variables

DEFAULT_EXCEPTION_NOTIFIER_RECIPIENTS can be overridden with AdvancedPluginHelper::Notifier.default_exception_recipients=(your-comma-separated-list-of-recipients).

EXCEPTION_NOTIFIER_RECIPIENTS can be overridden with AdvancedPluginHelper::Notifier.custom_exception_recipients=(your-comma-separated-list-of-recipients).

EXCEPTION_NOTIFIER_DISABLED can be overridden with AdvancedPluginHelper::Notifier.disabled=(true-or-false)

Changelog

All notable changes to this plugin will be reported in the changelog.

License

Copyright (C) 2022-2023 Liane Hampe (), xmera Solutions GmbH.

This plugin program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.