Добавление собственного модуля
В данной статье описано, как написать собственный модуль для ispmanager, который появится в панели в разделе Модули.
Для добавления своего модуля в ispmanager нужно добавить в директорию /usr/local/mgr5/etc/plugins/ispmgr xml-файл со специальной структурой:
Создайте файл exampleplugin.xml со следующим содержанием:
<?xml version="1.0" encoding="UTF-8"?>
<mgrdata>
<plugin name="exampleplugin">
<dist>1</dist>
<weight>1</weight>
<pricelist>12345</pricelist>
<license>ExampleLicense</license>
</free>
<settings>exampleplugin.settings</settings>
<msg name="desc_short" lang="ru">Модуль Example</msg>
<msg name="desc_short" lang="en">Module Example</msg>
<msg name="desc_full" lang="ru">Пример добавления модуля в ispmanager</msg>
<msg name="desc_full" lang="en">Example of adding custom module to ispmanager</msg>
<msg name="msg_help_links" dist="1" lang="ru"><a href="http://example.com" target="_blank">Описание ссылки для версий Lite, Pro, Host</a></msg>
<msg name="msg_help_links" dist="3" lang="ru"><a href="https://rudocs.ispmanager.com/ispmanager-business/integratsiya-c-viruslive" target="_blank">Описание ссылки для версии Business</a></msg>
<msg name="msg_help_links" dist="1" lang="en"><a href="http://example.com" target="_blank">Link description for Lite, Pro, Host versions</a></msg>
<msg name="msg_help_links" dist="3" lang="en"><a href="http://example.com" target="_blank">Link description for Business version</a></msg>
</plugin>
</mgrdata>
Описание тэгов:
- Параметр name - имя модуля;
- <dist> - опционально, указывает в какой версии будет использоваться модуль: 1 - lite, pro, host, 3 - business. При отсутствии будет использоваться во всех версиях;
- <weight> - опционально, позволяет управлять порядком модуля во вкладке “Модули”. Модули выстраиваются в порядке от большего веса к меньшему. При отсутствии веса модуль будет в конце списка.
- <pricelist> - опционально, id тарифа в биллинговой системе (если используется BILLmanager);
- <license> - опционально, имя лицензии в биллинговой системе (если используется BILLmanager);
- </free> - опционально, добавляет для модуля метку о том, что он бесплатный;
- <settings> - имя действия, которое будет выполняться при нажатии на кнопку настроек модуля;
- <msg> - текстовые сообщения для отображения во вкладке модули
У тэгов <msg> есть 2 параметра:
- lang - язык панели (“ru” - русский, “eng” - английский), в котором будет отображаться сообщение;
- dist - версия ispmanager, в которой будет отображаться сообщение, аналогично тэгу <dist>.
Сообщения:
- desc_short - краткое название модуля, используется в качестве заголовка;
- desc_full - подробное описание модуля;
- msg_help_links - опционально для добавления ссылок в меню “Показать полезные ссылки”.
Обратите внимание!
Тэг <weight> используется с релиза 6.90 (от 16.01.2024). В более ранних версиях использовать вместо него <group>integration</group> или <group>antiviruses</group> для добавления в соответствующий коллапс на текущей форме. Без этого Модуль будет в самом низу без группы и с пустой графой “Категории”
Функции установки и удаления модуля реализуются через работу с пакетом с именем “ispmanager-plugin-example”. Кнопки Установить и Удалить запускают установку или удаление пакетов соответственно через менеджер пакетов.
Признаком установленного пакета считается наличие файла с именем вида ispmgr_mod_exampleplugin.xml в директории /usr/local/mgr5/etc/xml.
Для работы обработчика необходимо наличие файла с именем вида ispmgr_mod_exampleplugin.xml в директории /usr/local/mgr5/etc/xml, где exampleplugin должно совпадать со значением тэга name.
Обработчику в STDIN подается xml от панели. В ответ обработчик должен вернуть корректный xml в STDOUT.
Для отладки своего обработчика можно включить для core_module уровень логирования 9 (Основное меню → Настройки логирования). В логе панели /usr/local/mgr5/var/ispmgr.log будут записываться xml с запросом к обработчику и ответом от него. В случае некорректного xml в ответе, логироваться он не будет.
Создайте файл со следующим содержимым:
<?xml version="1.0" encoding="UTF-8"?>
<mgrdata>
<plugin name="exampleplugin">
<dist>1</dist>
<settings>exampleplugin.settings</settings>
<free/>
</plugin>
<handler name="exampleplugin.py" type="xml">
<func name="exampleplugin.settings"/>
</handler>
<metadata name="exampleplugin.settings" type="form" mgr="ispmgr">
<form>
<field name="example_field" noname="yes" fullwidth="yes">
<textdata name="example_msg" type="msg"/>
</field>
<field name="example_input">
<input type="text" name="example_input" required="yes"/>
</field>
</form>
<buttons>
<button name="ok" type="ok"/>
<button name="cancel" type="cancel"/>
</buttons>
</metadata>
<lang name="ru">
<messages name="exampleplugin.settings">
<msg name="title">Пример формы</msg>
<msg name="example_msg">Пример сообщения</msg>
<msg name="example_input">Пример поля для ввода</msg>
<msg name="msg_create">Сохранить</msg>
</messages>
<messages name="mgrerror_exampleplugin">
<msg name="msg_error_exampleplugin">Возникла ошибка в работе примера формы: __object__</msg>
<msg name="wrong_value">некорректное введенное значение __value__</msg>
</messages>
</lang>
<lang name="en">
<messages name="exampleplugin.settings">
<msg name="title">Form example</msg>
<msg name="example_msg">Message example</msg>
<msg name="example_input">Input field example</msg>
<msg name="msg_create">Save</msg>
</messages>
<messages name="mgrerror_exampleplugin">
<msg name="msg_error_exampleplugin">Error in form example: __object__</msg>
<msg name="wrong_value">incorrect input value __value__</msg>
</messages>
</lang>
</mgrdata>
Описание тэгов:
- Значения в подтэгах тэга <plugin> должны совпадать со значениями из файла exampleplugin.xml;
- Параметр name тэга <handler> содержит имя файла, содержащего обработчик;
- <metadata> - содержит описание формы;
- <lang> - позволяет контролировать, для какого языка будут использоваться сообщения, описанные внутри него.
Сам обработчик должен быть помещен в директорию /usr/local/mgr5/addon. Имя файла должно совпадать со значением параметра name тэга <handler>.
Для примера используем следующий обработчик:
#!/usr/bin/env python
from sys import stdin
import os
import xml.etree.ElementTree as etree
#Заполнение формы значениями
def Get(root):
etree.SubElement(root, 'example_input').text = 'Do not change'
#Обработка нажатия на кнопку “Сохранить”
def Set(root):
val = os.getenv('PARAM_example_input')
if val == 'Do not change':
file = open(r'/usr/local/mgr5/var/exampleplugin.output', 'w')
file.write('Good job!\n')
file.close()
etree.SubElement(root, 'ok')
else:
#Формирование сообщения об ошибке
error = etree.SubElement(root, 'error')
error.set('code', '1')
error.set('type', 'exampleplugin')
error.set('object', 'wrong_value')
param = etree.SubElement(error, 'param')
param.text = 'wrong_value'
param.set('name', 'object')
param.set('type', 'msg')
param = etree.SubElement(error, 'param')
param.set('name', 'value')
param.text = val
if __name__ == "__main__":
#получение значения параметра, переданного панелью из переменных окружения
func = os.getenv('PARAM_func')
correct_function = False
#Проверка, что вызывается нужная функция
#Вызов функции по ее имени
if func == 'exampleplugin.settings':
correct_function = True
#Вызов и формы "Модули"
elif func == 'plugin.settings':
if os.getenv('PARAM_plugin_settings') == 'exampleplugin':
correct_function = True
if correct_function:
#Чтение xml, переданного из панели, из stdin
root = etree.parse(stdin).getroot()
#Проверка, что нажата кнопка "Сохранить”
if os.getenv('PARAM_clicked_button') == 'ok':
Set(root)
else:
Get(root)
print etree.tostring(root, encoding='utf8')
quit(0)
При нажатии на кнопку настроек модуля будет вызываться обработчик и показываться форма с заполненным значением поля. При изменении поля и нажатии на кнопку Сохранить будет сгенерирована ошибка.