====== Console Extensions Development ====== {{indexmenu_n>7}} We are working on the translation of the Pandora FMS documentation. Sorry for any inconvenience. ===== Console Extensions ===== Extensions are a form to develop new functionality for your Pandora Console as plugins. In this article you will learn how to develop a extension. ==== Kinds of Extensions ==== An extension can be one (or more) of the following types: * **Agent tab**: Extensions that appear in the tab header of the agent's operation and/or edition view. * **Visible**: Extensions that appear in the Pandora FMS Menu. * **Invisible**: Extensions that are loaded and executed in the index.php of the Pandora FMS Menu but that do not appear in the PFMS Menu. ==== Directory of Extensions ==== The directory of extensions is a subdirectory in your local installation of Pandora FMS Console with the name ''extensions''. {{ :wiki:pfms-pandora_console-extensions-directory.png }} This directory contains for each extension the following: **Main file of the extension** This file has the code to load in Pandora FMS Console web. **Subdirectory of extension** this is optional and may contain the Icon image file (a 18 by 18 pixels image) to show next to the name of the extension in the menu and others files as translations, modules, images, etc. ==== Extension Skeleton ==== function () {
} /*-------------------------------------*/ /* Adds the link in the operation menu */ extensions_add_operation_menu_option ('', '', ''); /* Adds the link in the godmode menu */ extensions_add_godmode_menu_option ('', '', '', '') /*-------------------------------------*/ /* Sets the callback function to be called when the extension is selected in the operation menu */ extensions_add_main_function (''); /* Sets the callback function to be called when the extension is selected in the godmode menu */ extensions_add_godmode_function (''); ?> ==== API for Extensions ==== The API for extensions is stil under development and may change in the future. The following sections contain the description of the functions in the API for extensions. === extensions_add_operation_menu_option === extensions_add_operation_menu_option ('', '', '') This function adds the link to the extension with the given name in the **Operations** menu. The third parameter is optional and is the relative path to the icon image ( 18×18 pixels) to apear next to the link. If this last parameter is not defined a plug icon ({{:wiki:extensions.png}}) will be used. === extensions_add_godmode_menu_option === extensions_add_godmode_menu_option ('', '' , '', '') This function adds the link to the extension with the given name in the **Godmode** menu if the user has the required ACL level as indicated by the second parameter. The forth parameter is optional and is the relative path to the icon image ( 18×18 pixels) to apear next to the link. If this parameter is not defined a plug icon ({{:wiki:extensions.png}}) will be used. === extensions_add_main_function === extensions_add_main_function ('') Sets the callback function that will be called when the user clicks on the link to the extension in the operation menu === extensions_add_godmode_function === extensions_add_godmode_function ('') Adds the extension function to call once the user goes to the extension in the Pandora FMS Web Console godmode instead of loading the main function. === extensions_add_login_function === extensions_add_login_function ('') Adds the extension function to call once the user successfully logs into the Pandora FMS Web Console. === extensions_add_godmode_tab_agent === extensions_add_godmode_tab_agent('', '', '', '') Adds one more tab to the agent edit view that when it is selected executes the code of the name function that we pass to it. === extensions_add_opemode_tab_agent === extensions_add_opemode_tab_agent('', '', '', '') Adds one more tab to the agent operating view than when it is selected will execute the code of the name function that we pass to it. === extensions_add_translation_string_function === extensions_add_translation_string_function('') Sets the callback function that will be called from common translation function for extensions. === Father IDs in menu === List of available strings IDs for use in extension API. If use null value or not incluyed param in ''call'' function, the extension appear only in submenu of extension. == Operation == * ''estado'': Monitoring view. * ''network'': Network view. * ''reporting'': Reporting and data visualization. * ''gismaps'': GIS view. * ''eventos'': Events view. * ''workspace'': User's workspace. == Administration == * ''gagente'': Manage monitoring. * ''gmassive'': Massive operations. * ''gmodules'': Manage modules. * ''galertas'': Manage alerts. * ''gusuarios'': Manage users. * ''godgismaps'': Manage GIS. * ''gserver'': Manage servers. * ''glog'': System logs. * ''gsetup'': SetupConfiguración. * ''gdbman'': DB Maintenance. **Administration Enterprise** These elements are only available with Enterprise version. * ''gpolicies'': Manage policies. ==== Example ==== {{ :wiki:pfms-combined-table-agent-module-group.png }} The extension show a table where the columns are Modules groups and the rows the Agent groups. And each cell have a colour with the next meanings: * **Green**: When all modules of Group are ''OK''. * **Yellow**: When at least one monitor in warning. * **Red**: At least one monitor fails. And this extension hang from the Operation menu in Agents. ==== Source code ==== array( "sentence" => "translated sentence" ), /* ... for other language's definitions. */ ); $args = func_get_args(); $string = array_shift($args); $user_language = get_user_language (); if(isset($translates[$user_language][$string])){ return vsprintf($translates[$user_language][$string], $args); } else{ return false; } } /** * Translate the array texts using gettext */ function translate(&$item, $key) { $item = ___($item); } /** * The main function of module groups and the enter point to * execute the code. */ function mainModuleGroups() { global $config; //the useful global var of Pandora Console, it has many data can you use //The big query $sql = "select COUNT(id_agente) AS count, estado FROM tagente_estado WHERE utimestamp != 0 AND id_agente IN (SELECT id_agente FROM tagente WHERE id_grupo = %d AND disabled IS FALSE) AND id_agente_modulo IN (SELECT id_agente_modulo FROM tagente_modulo WHERE id_module_group = %d AND disabled IS FALSE AND delete_pending IS FALSE) GROUP BY estado"; echo "

" . ___("Combine table of agent group and module group") . "

"; echo "

" . ___("This table show in columns the modules group and for rows agents group. The cell show all modules") . "

"; $agentGroups = get_user_groups ($config['id_user']); $modelGroups = get_all_model_groups(); array_walk($modelGroups, 'translate'); //Translate all head titles to language is set $head = $modelGroups; array_unshift($head, ' '); //Metaobject use in print_table $table = null; $table->align[0] = 'right'; //Align to right the first column. $table->style[0] = 'color: #ffffff; background-color: #778866; font-weight: bolder;'; $table->head = $head; //The content of table $tableData = array(); //Create rows and celds foreach ($agentGroups as $idAgentGroup => $name) { $row = array(); array_push($row, $name); foreach ($modelGroups as $idModelGroup => $modelGroup) { $query = sprintf($sql,$idAgentGroup, $idModelGroup); $rowsDB = get_db_all_rows_sql ($query); $states = array(); if ($rowsDB !== false) { foreach ($rowsDB as $rowDB) { $states[$rowDB['estado']] = $rowDB['count']; } } $count = 0; foreach ($states as $idState => $state) { $count = $state; } $color = 'transparent'; //Defaut color for cell if ($count == 0) { $color = '#babdb6'; //Grey when the cell for this model group and agent group hasn't modules. $alinkStart = ''; $alinkEnd = ''; } else { $alinkStart = ''; $alinkEnd = ''; if (array_key_exists(0,$states) && (count($states) == 1)) $color = '#8ae234'; //Green when the cell for this model group and agent has OK state all modules. else { if (array_key_exists(1,$states)) $color = '#cc0000'; //Red when the cell for this model group and agent has at least one module in critical state and the rest in any state. else $color = '#fce94f'; //Yellow when the cell for this model group and agent has at least one in warning state and the rest in green state. } } array_push($row, '
' . $alinkStart . $count . ' modules' . $alinkEnd . '
'); } array_push($tableData,$row); } $table->data = $tableData; print_table($table); echo "

" . ___("The colours meaning:") . "

" . "

"; } extensions_add_operation_menu_option("Modules groups", 'estado', 'module_groups/icon_menu.png'); extensions_add_main_function('mainModuleGroups'); extensions_add_translation_string_function('extensions_translation'); ?>
==== Explanation ==== In the source code there are two parts: * The source code of extension. * The API calls functions. The order of parts is indifferent, but it's better put "The API calls functions" in the bottom of you main file extension because the style guidelines advise to add this part into bottom thus all extensions have more or less a same style. === Source code of extension === In this case for this example has three function in the same file, but if you has a complex code, it is better divide in many files (and it save in subdirectory extension), the functions are: * Function ''extensions_translation()'' \\ This function is for extension specific translation. It is called from the common translation function. * Function ''translate(&$item, $key)'' \\ This function use for callback in the [[https://www.php.net/manual/en/function.array-walk.php|array_walk function]]. Because the main function keep the titles of columns and the titles of rows in array without translations. * Function ''mainModuleGroups()'' \\ This is the heard of extension, and it's huge of lines, I see not all code, see some important parts: * The first is access to config global var. In this var has many configurations and default values for many things of Pandora FMS Console. * The second var is the query in [[https://en.wikipedia.org/wiki/MySQL|MySQL]] into a string. And the ''%d'' is the [[https://en.wikipedia.org/wiki/Printf#printf_format_placeholders|format placeholders]] is for **Id Group** and **Id Module Group** and these are sustitute for value in [[https://www.php.net/manual/en/function.sprintf.php|sprintf function]] into foreach loops. * Some ''echos'' for print the text before the table. * Extract of DB two arrays with one dimension and the index is id, and the content is title for columns (Module groups) and rows (Agent group) in each case. * Translate the **Model Group** array titles. * Make the meta-object $table, fill by rows and print. * Before the ''foreach ~loops'', define into ''$table'' the head and styles of table. * The first loop is for rows (each agent group). * The second loop is for columns in current row (each model group). * Then for each cell, it has two number, id model group and id agent group, with this two number we make a query to database and we obtain the files. * Proccess the result array for obtain other array that is array and the index is a integer of diferents kinds of monitor states and the content is a count of monitor in this state. * Well, the only thing left is to make or fill the content of cell in HTML. The trick is easy. If the count off all states is zero, the background for div in CSS is grey. If ''$states[1] != 0'' or in human language there is one at least of monitor in critical state, the ''div'' has a red color. If the array only have a one cell and it's the normal state, the green is in this ''div''. And others cases, the yellow is the color for ''div''. * Add link in the cell if count is more than ''0''. * Save the row in ''$table'', and start other iteration of foreach. * Print the table. * Print the legend and other notes in the bottom of page. === API calls functions === There are only a few lines of code. The operations in these lines are: * Insert the extension into Pandora FMS menu. * And it's with the call extensions_add_operation_menu_option("Modules groups", 'estado', 'module_groups/icon_menu.png'); Where: * **'Modules groups**' is the name appear in submenu of agents. * **'Status**' is the element that hangs from the extension. * **'module_groups/icon_menu.png**' is the image icon appear in submenu, the path is relative to your extension directory. * Define the main function of this extension . \\ And it's with the call ''extensions_add_main_function('mainModuleGroups');'' where: * **'mainModuleGroups**' is the name of extension main function. The order in which the functions are named is irrelevant. === Directory organization === {{ :wiki:directory_extension_example.png }} The instalation of extension is very easy, because the Pandora FMS Console search new extensions and add into system when new extension is found. You only copy all files of extensions into the directory extension in your [[:es:documentation:pandorafms:installation:01_installing|Pandora FMS Console instalation]]. But you must **set the permissions for the Pandora FMS Console can read the files and subdirectories of extension**. In the screenshot, the extension has a directory structure: * ''module_groups'' * ''icon_menu.png'' * ''module_groups.php'' And the extension directory is for example in ''/var/www/pandora_console''. === Subdirectory === In this case, the example has one subdirectory, and usually any extension must has one subdirectory. The subdirectory has the same name as the name extension and the main file. The subdirectory of the example only has an image icon file (''icon_menu.png''). This icon is shown in the Pandora FMS Menu. [[:en:documentation:start|Go back to Pandora FMS documentation index]]