Как избежать запятых в аргументе массива Icinga
У меня есть сценарий Python, который принимает переменное количество целых чисел и работает с ними. Значения передаются в сценарий с аргументом командной строки -s
, Когда я запускаю скрипт локально, он работает нормально:
python check_myScript.py -s 1 2 3 4
Внутри целые числа превращаются в массив через запятую, используя этот код:
# /usr/lib/naios/plugins/check_myScript.py
import argparse
parser = argparse.ArgumentParser
parser.add_argument('-s', '--myNumbers', nargs='*', default="")
args = parser.parse_args()
Если я тогда позвоню print args.myNumbers
в скрипте я получаю массив, который выглядит так:
['1', '2', '3', '4']
Мне не удалось получить этот же вывод на Icinga Web. Я подозреваю, что это как-то связано с тем, как Icinga разрешает макросы. Мой код выглядит следующим образом:
# /etc/icinga2/conf.d/myScript.conf
# . . .
object Service "myService" {
import "generic-service"
host_name = "myHost"
check_command = "myCheckCommand"
vars.someNumbers = "1,2,3,4"
}
#/etc/icinga2/conf.d/commands.conf
# . . .
object CheckCommand "myCheckCommand" {
command = [PluginDir + "/check_myScript.py",]
arguments = {"-s" = "$someNumbers$"}
}
Вот некоторые из входов, которые я попробовал для vars.someNumbers =
и их соответствующие выводы:
INPUT | OUTPUT
----------------------------------------------
"'1' '2' '3' '4'" → ["'1' '2' '3' '4'"]
"['1' '2' '3' '4']" → ["['1' '2' '3' '4']"]
"['1','2','3','4']" → ["['1', '2', '3', '4']"]
['1','2','3','4'] → error
[1,2,3,4] → ['4']
[1 2 3 4] → error
"1,2,3,4" → ['1, 2, 3, 4']
Обновление: после долгих потрясений мне удалось заставить массивы выглядеть одинаково. Сначала я должен был изменить arguments
часть моей команды проверки для отключения ключей повтора:
arguments = {
"-s" = {
value = "$someNumbers$"
repeat_key = false
}
}
И мне пришлось использовать вход ["1", "2", "3", "4"]
, (Что интересно, ['1', '2', '3', '4']
не работает, так что, очевидно, есть разница между одинарными и двойными кавычками.)
Тем не менее, есть новая проблема: мой .join()
Метод ведет себя по-другому на Icinga, чем на моей локальной машине. На моей локальной машине, после .join()
, Я получил:
1,2,3,4
тогда как на Icinga я получаю:
1, 2, 3, 4
(обратите внимание на пробелы)
Я пробовал добавлять .replace(" ","")
к моему .join()
команда, но это не имеет никакого эффекта.
Обновление 2: у Icinga, кажется, есть какая-то странная одержимость запятыми и пробелами. Этот тест иллюстрирует проблему:
# test.py
sentence = ['this','is','a','sentence']
print sentence # → ['this', 'is', 'a', 'sentence']
print '-'.join(sentence) # → this-is-a-sentence
print ','.join(sentence) # → this, is, a, sentence
Вот еще более простой пример проблемы:
print "I,don't,want,spaces!" # → I, don't, want, spaces!
Попробуйте, как я мог бы, я еще не смог удалить пробелы (что является проблемой для моего приложения, так как предполагается, что он динамически формирует URL с переданными параметрами).
Обновление 3: мой текущий обходной путь для моей конкретной ситуации заключается в использовании %2C
в моем .join()
Метод, который является кодировкой URL для запятой.
1 ответ
Я думаю, что лучший способ - использовать массив в Icinga 2
object Service "myService" {
import "generic-service"
host_name = "myHost"
check_command = "myCheckCommand"
vars.someNumbers = [1, 2, 3, 4]
}
Затем с помощью repeat_key false
он должен работать.
arguments = {
"-s" = {
value = "$someNumbers$"
repeat_key = false
}
}
Помните, что Icinga 2 не позволит вам вводить "пробелы" в отдельные части командной строки. Это функция безопасности, поэтому вы не можете добавить какой-либо код оболочки.
Icinga 2 создает список аргументов для выполнения, чтобы пробелы были включены в один аргумент.
Вы также можете разделить Icinga 2, но это расширенное использование:
vars.someNumbers = "1,2,3,4"
object CheckCommand "myCheckCommand" {
command = [PluginDir + "/check_myScript.py",]
arguments = {
"-s" = {
value = {{
return macro("$someNumbers$").split(",")
}}
repeat_key = false
}
}