Files
librenms-librenms/LibreNMS/Plugins.php
Kevin Zink 98ed6bb9dc New plugin system based on Laravel Package Development (#12998)
* use Blade view and Eloquent models for plugins

* move views

* fix style

* fix style

* revert mistake

* Update Plugin.php

delete test property "name"

* rename plugin function to settings

* last but not least - rename in Test.php

* Rename Test to Example

* fix typo

* fix style

* fix style

* fix style

* fix style - I hate tabs...

* Extract view calls

* fix method calls and style

* Move Models the the abstract class

* fix style

* Convert to traits

* Change the Example description

* Fix style

* Fix style

* Fix style

* Convert plugin function to Model static methods and delete .inc.php

* fix style

* fix style

* Use scope

* final methods blows up legacy code

* Config > \LibreNMS\Config

* convert the static string to a static method

* Correct placement in the page

* fix tabs

* fix style

* Rename from tait to hook

to make it easier to understand and be complient

* rename file

* Typo

* Started to change the docu

* change to a more usefully Device_Overview example

* and activate of course

* PluginManager

* fix .gitignore

* only php files in the root folder

* corrected .gitignore with all files :)

* Rename the Hooks and ExampleClass for better readability

* Fix style

* Fix style

* Exception handling (especially if DB is not present)

* Fix style and update schema

* fix indentation

* actually correct indent

* fix migration collation check include utf8mb4_bin

* stop phpstan whining

* A view lines documentation

* add typeHints

* Allow return null on handle

* lint

* fix return types

* fix logic of column collation check

* Fix MenuEntryHook

* switch to longtext instead of json type for now :D

* try phpstan on PHP 7.3

* set phpstan target version to 7.3

* all the typehints

* optional

* more

* Use namespace to prevent view collisions
disambiguate plugin and hook
no magic guessing of names in PluginManager, bad assumptions
remove unused plugins from the DB

* cleanup plugin menu

* cleanup on shutdown and ignore but log query error on cleanup

* instanceof must be called against an instance

* Allow multiple hooks per plugin

* Port plugin ui code to Laravel

* page instead of settings for v1 plugins

* actually working settings pages
a little url cleanup plugin/admin -> plugin/settings

* fix style

* Add page hook

* PHPstan

* Try to fix Illuminate\Http\RedirectResponse

* typehint

* Rewrite the doc

* Fix style

Co-authored-by: PipoCanaja <38363551+PipoCanaja@users.noreply.github.com>
Co-authored-by: Tony Murray <murraytony@gmail.com>
2021-10-19 06:53:28 -05:00

254 lines
6.6 KiB
PHP

<?php
/**
* Plugins.php
*
* -Description-
*
* This 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 3 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.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>
*
* @author LibreNMS Group
*
* @link https://www.librenms.org
*
* @copyright 2016
*/
namespace LibreNMS;
use App\Models\Plugin;
use Log;
/**
* Handles loading of plugins
*
* @author LibreNMS Group
*
* @link https://www.librenms.org
*
* @copyright 2016
*
* Supported hooks
* <ul>
* <li>menu</li>
* <li>device_overview_container</li>
* <li>port_container</li>
* </ul>
*/
class Plugins
{
/**
* Array of plugin hooks
*
* @var array|null
*/
private static $plugins = null;
/**
* Start loading active plugins
*
* @return bool
*/
public static function start()
{
if (! is_null(self::$plugins)) {
return false;
}
self::$plugins = [];
$plugin_dir = Config::get('plugin_dir');
if (! file_exists($plugin_dir)) {
return false;
}
$plugin_files = Plugin::isActive()->get()->toArray();
foreach ($plugin_files as $plugins) {
$plugin_file = $plugin_dir . '/' . $plugins['plugin_name'] . '/' . $plugins['plugin_name'] . '.php';
$plugin_info = pathinfo($plugin_file);
if ($plugin_info['extension'] !== 'php') {
continue;
}
if (! is_file($plugin_file)) {
continue;
}
self::load($plugin_file, $plugin_info['filename']);
}
return true;
}
/**
* Load plugin
*
* @param string $file Full path and filename of plugin
* @param string $pluginName Plugin name without any namespace
* @return object|null
*/
public static function load($file, $pluginName)
{
chdir(Config::get('install_dir') . '/html');
$plugin = self::getInstance($file, $pluginName);
if (! is_null($plugin)) {
$class = get_class($plugin);
$hooks = get_class_methods($class);
foreach ((array) $hooks as $hookName) {
if ($hookName[0] != '_') {
self::$plugins[$hookName][] = $class;
}
}
}
chdir(Config::get('install_dir'));
return $plugin;
}
/**
* Get an instance of this plugin
* Search various namespaces and include files if needed.
*
* @param string $file
* @param string $pluginName
* @return object|null
*/
private static function getInstance($file, $pluginName)
{
$ns_prefix = 'LibreNMS\\Plugins\\';
$ns_psr4 = $ns_prefix . $pluginName . '\\' . $pluginName;
$ns_plugin = $ns_prefix . $pluginName;
$ns_global = $pluginName;
if (class_exists($ns_psr4)) {
return new $ns_psr4;
}
if (class_exists($ns_plugin)) {
return new $ns_plugin;
}
// Include file because it's not psr4 (may have been included by previous class_exists calls
include_once $file;
if (class_exists($ns_global)) {
return new $ns_global;
}
return null;
}
/**
* Get all plugins implementing a specific hook.
*
* @param string $hook Name of the hook to get count for
* @return int|bool
*/
public static function countHooks($hook)
{
// count all plugins implementing a specific hook
self::start();
if (! empty(self::$plugins[$hook])) {
return count(self::$plugins[$hook]);
} else {
return false;
}
}
/**
* Call hook for plugin.
*
* @param string $hook Name of hook to call
* @param array|false $params Optional array of parameters for hook
* @return string
*/
public static function call($hook, $params = false)
{
chdir(Config::get('install_dir') . '/html');
self::start();
ob_start();
if (! empty(self::$plugins[$hook])) {
foreach (self::$plugins[$hook] as $name) {
try {
if (! is_array($params)) {
@call_user_func([$name, $hook]);
} else {
@call_user_func_array([$name, $hook], $params);
}
} catch (\Exception $e) {
Log::error($e);
}
}
}
$output = ob_get_contents();
ob_end_clean();
chdir(Config::get('install_dir'));
return $output;
}
/**
* Get count of hooks.
*
* @return int
*/
public static function count()
{
self::start();
return count(self::$plugins);
}
public static function scanNew()
{
$countInstalled = 0;
if (file_exists(\LibreNMS\Config::get('plugin_dir'))) {
$plugin_files = array_diff(scandir(\LibreNMS\Config::get('plugin_dir')), ['..', '.']);
$plugin_files = array_diff($plugin_files, Plugin::versionOne()->pluck('plugin_name')->toArray());
foreach ($plugin_files as $name) {
if (is_dir(\LibreNMS\Config::get('plugin_dir') . '/' . $name)
&& is_file(\LibreNMS\Config::get('plugin_dir') . '/' . $name . '/' . $name . '.php')) {
Plugin::create(['plugin_name' => $name, 'plugin_active' => false, 'version' => 1]);
$countInstalled++;
}
}
}
return $countInstalled;
}
public static function scanRemoved()
{
$countRemoved = 0;
if (file_exists(\LibreNMS\Config::get('plugin_dir'))) {
$plugin_files = array_diff(scandir(\LibreNMS\Config::get('plugin_dir')), ['.', '..', '.gitignore']);
$plugins = Plugin::versionOne()->whereNotIn('plugin_name', $plugin_files)->select('plugin_id')->get();
foreach ($plugins as $plugin) {
if ($plugin->delete()) {
$countRemoved++;
}
}
}
return $countRemoved;
}
}