Создание пользовательского блока
Описание
Создание пользовательского блока — это процесс создания нового блока в Puzzle RPA, который позволяет пользователям создавать свои собственные блоки для использования в процессах.
Перечень действий для создания пользовательского блока
- Требуется создать папку с именем блока (например
hello_my_block). - Внутри папки создать структуру файлов (см. ниже).
- Наполнить файлы
block.json,values.xml,code.value.py(илиcode.py),meta.json,libs.py.json,typeToName.json. - Добавить код в
src/__init__.py. - (Опционально) Указать зависимости в
requirements.txtи поместить готовые wheels вwheels/. - Импортировать папку в Puzzle RPA через Настройки → Расширения.
- Перезагрузить Puzzle RPA и проверить блок в тулбоксе.
Структура папки блока
hello_my_block/ # имя папки - обычно совпадает с type блока├─ block.json├─ code.value.py # или code.py├─ meta.json├─ libs.py.json├─ values.xml├─ typeToName.json├─ requirements.txt # опционально├─ wheels/ # опционально, *.wls файлы└─ src/ # папка для python - скриптов и вспомогательных файлов └─ __init__.pyНазначение файлов
block.json- JSON-описание внешнего вида блока;code.value.py/code.py- строка/код, который будет встраиваться в сгенерируемый Python (вызов функции изcode-storage-py);libs.py.json- список импортируемых модулей изcode-storage-py;meta.json- режим, флаги логирования и поведение генерации;values.xml- XML-описание разъёмов/тулбокса (shadow/начальные блоки);typeToName.json- отображаемое название блока для логов/информации;requirements.txt- файл описывающий дополнительные python библиотеки и их версии которые используются блоком;wheels/- папка с wheel-пакетами которые используются блоком;src/- папка для python - скриптов и вспомогательных файлов;src/__init__.py- реализация функций, вызываемых блоком (c декораторами логгера).
Создание файла block.json
type- уникальное имя блока; должно совпадать с именем папки и с<block type="...">вvalues.xml.message0- текст блока; параметры (разъёмы блока) указываются как%1,%2и т.д. Например, “Умножить %1 на 2”.args0: массив описаний разъёмов/полей.inputsInline:true/false— располагает ли разъёмы в одну строку.output- ключ со значениемnullдля блока возвращающего значение.previousStatement- ключ со значениемnullдля блока который может иметь предыдущее соединение сверху (обозначенное выемкой).nextStatement- ключ со значениемnullдля блока который следующее соединение снизу (обозначенное вкладкой).colour: цвет блока в HEX.tooltip- текст подсказки который будет отображаться при наведении на блок.helpUrl- URL справки по блоку.
Пример block.json для блока, который умножает число на 2:
{ "type": "hello_double", "message0": "Умножить %1 на 2 %2", "args0": [ { "type": "input_value", "name": "VALUE1" }, { "type": "input_dummy" } ], "inputsInline": true, "output": null, "colour": "#4A90E2", "tooltip": "Умножает число на 2", "helpUrl": ""}Примечания и типичные параметры args0
input_dummy- пустая строка/видимый текст в блоке (используется для описаний).
{ "type": "input_dummy"}input_value- поле для подсоединения блока со значением (value).
{ "type": "input_value", "name": "VALUE1"}input_statement- для вложенных блоков (statement/тело).
{ "type": "input_statement", "name": "DO"}field_dropdown- выпадающий список (нужны опции вoptions).
{ "type": "field_dropdown", "name": "MODE", "options": [ [ "Опция A", "A" ], [ "Опция B", "B" ] ]}field_checkbox- чекбокс (логическое значение).
{ "type": "field_checkbox", "name": "ENABLED", "checked": true}field_number- числовое поле.
{ "type": "field_number", "name": "NUM", "value": 10, "min": 0, "max": 100, "precision": 1}field_text- текстовое поле.
{ "type": "field_text", "name": "TEXT", "text": "Пример"}field_variable- поле переменных.
{ "type": "field_variable", "name": "VAR", "variable": "item"}field_code_editor- поле для редактирования кода.
{ "type": "field_code_editor", "name": "CODE"}field_image- поле для отображения изображения (Требуется base64 строка изображения, width и height в пикселях, alt - текст изображения когда блок свернут).
{ "type": "field_image", "src": "", "width": 25, "height": 25, "alt": "SQL"}Создание файла code.value.py / code.py
- Если блок возвращает значение — используем
code.value.py. - Если блок не возвращает значение — используем
code.py.
Синтаксис (строка, которая будет вставлена в сгенерированный код)
${vc('VALUE1')}- метка для кодогенерации: значение из разъёмов типа input_value.${fv('FIELD1')}- метка для кодогенерации: значение из полей типа field_text, field_number, field_variable, field_dropdown, field_checkbox.${sc('DO')}- метка для кодогенерации: код из блока типа input_statement (для вложенных блоков).#isInternalLog- специальный маркер для передачи параметра логирования (обычно последний аргумент).
Пример code.value.py (вызов функции из модуля hello_double):
hello_double.double_value(${vc('VALUE1')},#isInternalLog)Примечания:
- Если нужно передать несколько аргументов — перечислить:
${vc('VALUE1')}, ${vc('VALUE2')}. - Если требуется обернуть значения полей типа fv в кавычки, то используйте
'${fv('FIELDVALUE1')}'
Создание файла libs.py.json
- Формат: JSON-массив путей модулей, которые нужно импортировать в сгенерированный код (модули из
code-storage-py). - Префикс
$/означает путь внутри code-storage (пример используемой практики).
Пример:
[ "$/hello_double"]Примечания:
- Откуда брать имена? — соответствуют именам модулей в
code-storage-py.
Создание файла meta.json
Описание ключей:
_modeсо значением"value"— блок возвращает значение (используетсяcode.value.py)._isInternalLogсо значениемtrue/false— включить внутреннее логирование для блока._isDontUseLoggerсо значениемtrue— блок не должен писать лог (подавляет логирование)._onIfUseFuncсо значениемfalse— устаревший флаг (требуется всегда указыватьfalse).
Пример:
{ "_mode": "value", "_isInternalLog": true, "_onIfUseFunc": false,}Условия использования ключей:
| Задача | Рекомендация |
|---|---|
| Блок возвращает значение | _mode: "value" |
| Блок просто выполняет действие | не указывать _mode |
| Нельзя логировать | _isDontUseLogger: true |
| Лог нужен | _isInternalLog: true |
Создание файла typeToName.json
- Служит для записи читаемого названия блока в логах.
- Формат:
{"type_name": "Читаемое название"}.
Пример:
{ "hello_double": "Умножить число на 2"}Создание файла values.xml
- Описывает как блок выглядит в тулбоксе, какие теневые блоки (
shadow) или обычные блоки (block) подставлять в разъёмы блока.
Пример описания блока hello_double в values.xml:
<block type="hello_double"> <value name="VALUE1"> <shadow type="math_number"> <field name="NUM">2</field> </shadow> </value></block>Примеры разных типов полей в values.xml
<!-- Текстовое поле --><shadow type="text"> <field name="TEXT"></field></shadow>
<!-- Числовое поле --><shadow type="math_number"> <field name="NUM">10</field></shadow>
<!-- Переменная --><block type="variables_get"> <field name="VAR">myVar</field></block>Создание файла requirements.txt и wheels/
- Если блок использует библиотеки, отсутствующие в окружении Puzzle RPA, укажите их в файле
requirements.txt. - В папке
wheels/размещаются собранные.whlфайлы совместимых с версией Python (3.11) и операционной системой.
Рекомендации:
- Минимизируйте внешние зависимости.
- Если библиотека уже присутствует в Puzzle RPA — не добавляйте её в файл
requirements.txt, чтобы не вызвать конфликты. - В документации/репозитории проекта храните актуальный список предустановленных библиотек.
Пример содержимого файла requirements.txt:
pandas==2.2.3numpy==1.24.3Создание файла src/__init__.py — реализация функций
- Внутри находятся функции, которые вызываются из файлов
code.value.py/code.py. - Для логирования используются декораторы из модуля
puzzle_logger.
Пример реализации:
from puzzle_logger import log_decorator, window_logger
@window_logger@log_decoratordef double_value(value, puzzle_logger_path=None, block_text=None, block_id=None, window_log=False, **kwargs): if isinstance(value, (int, float)): return value * 2
if isinstance(value, str): try: return float(value) * 2 except ValueError: raise ValueError(f"Некорректный аргумент: '{value}' не является числом")
raise TypeError(f"Тип '{type(value).__name__}' не поддерживается")Примечания:
- Всегда ставьте параметры
puzzle_logger_path=None, block_text=None, block_id=None, window_log=Falseпосле основных параметров, если хотите корректный лог. - Логирование и обработка ошибок важны для поддержки и диагностики блока.
Типичные ошибки и как их избежать
- Не совпадают имена:
typeвblock.json, имя вtypeToName.jsonи<block type>вvalues.xmlдолжны совпадать — иначе блок не подключится или не будет логироваться. Имя папки с файлами расширения должно соответствоватьtypeвblock.json. - Неправильный формат JSON — синтаксис JSON, пути к модулям должны быть корректны.
- Запутанные зависимости — разные версии библиотек в
requirements.txtмогут конфликтовать с системными. Тестируйте на отдельном стенде. - Отсутствуют ключи описания полей или функции не экспортируются — убедитесь, что функция и ключи описания полей существуют и их имена совпадают с вызовом в
code.value.py/code.py. - Блок не появляется в тулбоксе — в чём причина? Часто: ошибки в
values.xml, несовпадениеtype, синтаксическая ошибка вblock.json.
Минимальный шаблон (готовые файлы) — папка hello_double
- Скачать пример блока — hello_double.zip — и распаковать архив в папку блока (например
hello_double). Либо создать папку и файлы вручную по этой инструкции. - Импортировать папку в Puzzle RPA: “Настройки → Расширения”.
Файл: block.json
{ "type": "hello_double", "message0": "Умножить %1 на 2 %2", "args0": [ { "type": "input_value", "name": "VALUE1" }, { "type": "input_dummy" } ], "inputsInline": true, "output": null, "colour": "#4A90E2", "tooltip": "Умножает число на 2", "helpUrl": ""}Файл: values.xml
<block type="hello_double"> <value name="VALUE1"> <shadow type="math_number"> <field name="NUM">2</field> </shadow> </value></block>Файл: code.value.py
hello_double.double_value(${vc('VALUE1')},#isInternalLog)Файл: libs.py.json
[ "$/hello_double"]Файл: meta.json
{ "_mode": "value", "_isInternalLog": true, "_onIfUseFunc": false,}Файл: typeToName.json
{ "hello_double": "Умножить на 2"}Файл: __init__.py
from puzzle_logger import log_decorator, window_logger
@window_logger@log_decoratordef double_value(value, puzzle_logger_path=None, block_text=None, block_id=None, window_log=False, **kwargs): if isinstance(value, (int, float)): return value * 2
if isinstance(value, str): try: return float(value) * 2 except ValueError: raise ValueError(f"Некорректный аргумент: '{value}' не является числом")
raise TypeError(f"Тип '{type(value).__name__}' не поддерживается")