
Привет, на CKA курсе туторы нарочито вдалбливают последовательность действи: set the cluster, set the namespace, set the pod... И все это происходит в консоли, используя kubectl
- консольная утилита для управления k8s кластерами. Мне настолько это вдолбилось что я решил это автоматизировать. У меня уже был интересный проект автоматизации ssh подключений (sshto), не долго думая, я взял его за основу и написал kui. Это bash скрипт, dialog
обёртка для kubectl
. У диалога есть приятная фича присваивать горячку каждому пункту меню по первому символу. Это очень удобно, в результате весь k8s кластер оказывается у вас на кончиках пальцев.
Я уже упоминал kui в своих публикациях но раз уж сейчас сезон Open source, решился написать самостоятельную статью про kui, OS и вот это вот все.
Зачем делать свое если есть уже много не своего? Да, да, но я недавно побывал в замечательном городе Владимире, в числе прочего зашли в Кузницу-Музей Бородиных. Очень понравилось, дядя кузнец красивым поставленным баритоном рассказал про кузнечное дело что да как, задавал вопросы а за правильные ответы давал сахарок) В кузнице много инструментов: молотков, клещей, оправок... Раньше кузнец все это делал сам, сам делал себе инструменты. Получил заказ, подумал покумекал как делать, сделал инструмент, сделал заказ. Вот так потихоньку в кузнице инструмент и прибывал. Зайдешь в кузницу, посмотришь сколько разного инструмента у кузнеца и сразу понятно опытный кузнец или нет. Вот и я как тот кузнец, инструменты себе делаю сам, не могу не куя.
Будете во Владимире обязательно зайдите в это место, особенно с детьми. Можно самому молотком поколотить, сделать гвоздь на добрую память. К сожалению главная достопримечательность Владимира - Золотые ворота в данный момент на реставрации ну чтож - это повод вернуться и выковать еще один гвоздь.
Так что моя философия - делай своё, делай сам, делай больше, в любом случае будешь вознаграждён опытом и интересно проведешь время, только OS, только хардкор)
Начнем же знакомство с kui'ем. О чем вообще сыр бор, зачем нужен kui? Вот пример обычного рабочего процесса с kubectl
, скажем нам надо посмотреть подики в неймспейсе clickhouse-dev, пишем такое:
$ kubectl --kubeconfig=.kube/config_dev -n clickhouse-dev get po
NAME READY STATUS RESTARTS AGE
chi-cluster-dev01-0-0-0 2/2 Running 0 29d
chi-cluster-dev01-0-1-0 2/2 Running 0 29d
chi-cluster-dev01-0-2-0 2/2 Running 0 29d
Выглядит просто когда можно скопипастить команду но когда сам тыкаешь кнопки, все несколько скучнее, видосик:
Может я просто медленно печатаю? Но я всегда придерживался мнения что работать должен компьютер а человек должен только контролировать его работу и вмешиться по необходимости. Нажми на кнопку, получишь результат. А иначе зачем это все? Попробуем сделать тоже самое с kui'ем:
Согласитесь с kui'ем повеселее чем без kui'я!) А ведь это просто bash скрипт. При желании вы сможете свой kui заточить под ваши нужды без особых проблем. У меня так и было, сначала потенционал kui'я был совсем маленький, выбрать кластер, неймспейс, посмотреть подики, логи. В какой-то момент появилась задача в которой требовалось пробросить порт в подик, сделал и добавил такую функцию в kui. Клиент никак не мог удалить залипший подик, применил --force
и добавил в kui команду Termination. Так потенциал kui постепенно увеличивался и сейчас kui стал очень удобным инструментом. Частенько требуется залезть прямо в под, и потыкать там, вот посмотрите как с помощью kui'я можно бысто пролезть в подик:
Чик-пык и мы внутри. Потом вернулись обратно в kui, выбрали следующий под и пошло-поехало, удобно. Кстати kui запоминает где вы были в последний раз (кластер и неймспейс) и если вы перезапустите kui то начнете там где закончили.
Был забавный диалог с коллегами:

Решил увековечить это в kui, переименовал команды cordone и uncordone вот так:

Получилось очень точно по смыслу, не хочешь плодить поды, надень ... эм, закордонь. И к названию подходит как нельзя лучше. Возможно, разработчики kubectl, придумывая названия команд cordone/uncordone, тоже думали об этом, названия команд подозрительно созвучны моему варианту)
Рассмотрим kui поближе. Как я уже сказал это bash скрипт а в качестве "движка" используется диалог. Чтобы все это работало вам понадобится установить dialog
и kubectl
. А для доступа к кластерам необходимо конфиги с ключами положить в папку ~/.kube
- это путь по умолчанию для конфигов kubectl
, файлы должны начинаться со слова config*
. Если вы по каким-то причинам используете другую папку/файлы, подкрутите в kui'е вот эту переменную:
CONFILES=$(echo ~/.kube/{config,config*[!~]}) # k8s confiles list.
Если у вас уже подгорает добавить что-нибудь, вперед, вот как это все выглядит внутри:
#------------------------{ Dialog creator }------------------------
D(){
local type="$1"
local name="$2"
local oklb="$3"
local nolb="$4"
local opts="$5"
shift 5
case $type in
menu) local size='0 0 0';;
inputbox) local size='10 60';;
esac
dialog $opts \
--no-collapse --output-fd 1 \
--colors --aspect 100 \
--ok-label "$oklb" \
--cancel-label "$nolb" \
--$type "$name" \
$size -- "$@"
}
Это функция для упрощеннго создания диалогов. А вот как эта функция используется, простые диалоги ввода данных (inputbox):
#------------------------{ Change ports used in port-forwarding command }--------------------------
local_port(){
new_local=$(D inputbox 'Change local port' CHANGE BACK '--max-input 5' $LOCAL)
LOCAL=${new_local:-$LOCAL}
cmds_regen
}
remote_port(){
new_remote=$(D inputbox 'Change remote port' CHANGE BACK '--max-input 5' $REMOTE)
REMOTE=${new_remote:-$REMOTE}
cmds_regen
}
#------------------------{ Change command to run in pod }------------------------------------------
change_command(){
new_kubcmd=$(D inputbox 'Change command to run in pod' CHANGE BACK '--max-input 255' "$KUBCMD")
KUBCMD="${new_kubcmd:-$KUBCMD}"
cmds_regen
}
...
И пример диалога с выбором (menu), в данном случае кластера:
select_cluster(){
cluster=$(D menu "Select cluster:" SELECT EXIT '--no-items --extra-button --extra-label NODES --help-button --help-label TOP' "${!clusters[@]}")
case $? in
0) kubconfig="${clusters[$cluster]}"
select_namespace;;
2) cluster=${cluster//HELP /}
kubconfig="${clusters[$cluster]}"
select_node top;;
3) kubconfig="${clusters[$cluster]}"
select_node;;
1) bye;;
esac
}
Диалог и соответственно функция возвращают разные коды завершения в зависимости от нажатой кнопки. Кнопок диалог позволяет создавать максимум 4. По умолчанию это кнопки:
ok, код 0
cancel, код 1
help, код 2
extra, код 3
Названия кнопок можно изменять опцией --<имя_кнопки>-label <новое_название_кнопки>
. Для проверки кода используется case
. Самый интересный диалог это выбор действия с объектами: под, джоб, деплой и т.д. Но не все команды актуальны для того или иного типа объектов. Поэтому мне пришлось сделать несколько наборов команд:
cmds_regen(){ # Commands to run, list will be updated
#-----------------------+--------------------------------------------------------------+
# Command name | Description |
#-----------------------+--------------------------------------------------------------+
quick_butt=( '' '' "$quickl" "$quickd"
"$F_BUTT" "Filter items "
"$N_BUTT" "Go to namespace selection "
"$T_BUTT" "Go to object type selection"
"$C_BUTT" "Go to cluster selection "
"$D_BUTT" "Go to node selection"
"$P_BUTT" "Go to top node selection"
"$W_BUTT" "Watch for a while how it's going" )
descr_cmds=( Get "Get this $type"
Describe "Describe this $type"
Output "Set output mode: ${outputypes_list%|}" )
label_cmds=( Label "Set this $type as label" )
scale_cmds=( Scale "Change number of replicas in this $type" )
delete_cmds=( Delete "Delete this $type"
Termination "Delete this $type using --force" )
common_cmds=( "${descr_cmds[@]}"
'' '' # delimiter
Logs "Get $type logs"
'Logs all' "Get $type logs from all containers at once!"
Search "Grep something in logs"
Container 'Select container to run command or get logs from'
Edit "Edit this $type"
'' '' # delimiter
"${delete_cmds[@]}" )
top_cmds=( "Top $type" "Show metrics for this $type"
"Top ${type}s" "Show metrics for all ${type}s" )
rollout_cmds=( 'Rollout restart' "Restart this $type"
'Rollout status ' "Show the status of this $type's rollout"
'Rollout pause ' "Mark the provided this $type as paused"
'Rollout resume ' "Resume this paused $type"
'Rollout undo ' "Undo a previous rollout of this $type"
'Rollout history' "View rollout history of this $type" )
exec_cmds=( "Execute $KUBCMD" "Run command '\Z1$KUBCMD\Z0' in this $type"
'Change command' "Change command(\Z1$KUBCMD\Z0)" )
pfwd_cmds=( 'Port-forward' "Forward local port \Z2$LOCAL\Z0 to pods port \Z2$REMOTE\Z0"
'Change local port' "Change local(\Z2$LOCAL\Z0) port"
"Change pod's port" "Change pod's(\Z2$REMOTE\Z0) port" )
shell_cmds=( 'Interactive shell' "Open interactive shell in this $type" )
cron_cmds=( 'Suspend' "Suspend this $type"
'Unsuspend' "Unsuspend this $type"
Delete "Delete this $type"
Edit "Edit this $type" )
node_cmds=( Condom "Make this $type unschedulable"
Uncondome "Make this $type schedulable again"
Drain "Drain this $type in preparation for maintenance"
'Node pods' "Show pods of this node"
'Top node pods' "Show top pods of this node"
'' '' # delimiter
"${top_cmds[@]}" )
pod_cmds=( "${descr_cmds[@]}"
'' '' # delimiter
Logs "Get $type logs"
'Logs all' "Get $type logs from all containers at once!"
Search "Grep something in logs"
Container 'Select container to run command or get logs from'
'' '' # delimiter
"${delete_cmds[@]}"
'' '' # delimiter
"${top_cmds[@]}"
'' '' # delimiter
"${shell_cmds[@]}"
'' '' # delimiter
"${exec_cmds[@]}"
'' '' # delimiter
"${pfwd_cmds[@]}" )
depl_cmds=( "${common_cmds[@]}"
'' '' # delimiter
"${label_cmds[@]}"
'' '' # delimiter
"${scale_cmds[@]}"
'' '' # delimiter
"${rollout_cmds[@]}"
'' '' # delimiter
"${exec_cmds[@]}"
'' '' # delimiter
"${pfwd_cmds[@]}" )
}; cmds_regen
И применять их в зависимости от объекта:
#------------------------{ Play with objects }-----------------------------------------------------
butt_no_filter=("${quick_butt[@]::4}" "${quick_butt[@]:6:10}")
play_with_event (){ WAIT; data=$(kube get $type $ns 2>&1); GO; echo "$data"; pause; select_type; }
play_with_statefulset (){ all_cmds_check "${common_cmds[@]}" "${scale_cmds[@]}" '' '' "${rollout_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_node (){ all_cmds_check "${descr_cmds[@]}" '' '' "${node_cmds[@]}" "${butt_no_filter[@]:0:10}"; }
play_with_daemonset (){ all_cmds_check "${common_cmds[@]}" '' '' "${rollout_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_cronjob (){ all_cmds_check "${descr_cmds[@]}" '' '' "${cron_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_secret (){ all_cmds_check "${descr_cmds[@]}" '' '' "${delete_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_replicaset (){ all_cmds_check "${common_cmds[@]}" "${scale_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_deployment (){ all_cmds_check "${depl_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_pod (){ all_cmds_check "${pod_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_job (){ all_cmds_check "${common_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_componentstatuse (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_serviceaccount (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_podtemplate (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_limitrange (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_configmap (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_endpoint (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_service (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
play_with_ingress (){ all_cmds_check "${descr_cmds[@]}" "${butt_no_filter[@]}"; }
А вот этот набор:
quick_butt=( '' '' "$quickl" "$quickd"
"$F_BUTT" "Filter items "
"$N_BUTT" "Go to namespace selection "
"$T_BUTT" "Go to object type selection"
"$C_BUTT" "Go to cluster selection "
"$D_BUTT" "Go to node selection"
"$P_BUTT" "Go to top node selection"
"$W_BUTT" "Watch for a while how it's going" )
Пришлось сделать чтобы преодолеть ограничение диалога в 4 кнопки. Можно конечно было добавить подменю, но для вызова подменю нужна кнопка а с ними напряженка. Подменю не удобно и там тоже было бы максимум 4 кнопки. Сделал так, получилось как-бы встроенное подменю. Все дополнительные кнопки/команды на виду и доступны сразу по горячим кнопкам. Да, если основной список длинный они уезжают за экран, но когда долго работаешь с kui'ем горячки запоминаются быстро, достаточно нажать циферку и нужная команда выбрана. Это очень удобно.
Чаще всего приходится возиться с подиками поэтому список команд для подов самый большой:

Можно посмотреть под в разных форматах json|yaml|wide|name. Посмотреть логи. Относительно недавно добавил возможность смотреть логи сразу во всех контейнерах. Когда у пода несколько контейнеров команда kubectl -n.. logs pod/pod_name
ругнется если не указан контейнер:
error: a container name must be specified for pod chi-cluster-dev01-0-0-0, choose one of: [clickhouse clickhouse-bulk]
Но с kui'ем можно сказать Logs all и получить логи сразу всех контейнеров.
Можно удалить под, выполнить в нем команду открыть порт или открыть интерактивную оболочку.
Теперь с помощью подменю Quick selection изменяем тип на deployment:


И видим что деплойментов тут нет О_о

Да, судя по именам подиков это стайтфулсет, нажимаем 3 > enter S > enter

Ага, стайтфулсет, посмотрим что можно с ним делать:

Основная функция тут конечно рестарт, частенько приходится рестартить какой-нибудь стейтфулсет или деплоймент.
Ну и конечно ноды, жмем 5 > enter


Заго.. закордонивание уже описано, еще можно сделать drain, посмотреть статистику топ нод, выбрать все поды ноды и даже топ поды ноды. И все это на кончиках пальцев. Мышь можно не трогать, хотя dialog её поддерживает, элементы меню можно подсвечивать мышью.
Это опенсорс. Есть идеи? Шлите пулреквесты. Нашли ошибку, создайте issue, возможно когда-нибудь исправлю) Кстати вот на днях некто john9x указал на очепятку, поправил, спасибо.
Почему опенсорс? Ну а как иначе, во-первых как я сказал выше философия, во-вторых в проекте используется открытое ПО: didalog
, kubectl
. В-третьих практически все что я знаю и умею, я подчерпнул из открытых источников, поэтому стараюсь вернуть долг опенсорс сообществу, пишу код, пишу статьи когда есть время. Надеюсь кому-то это будет полезно.
В блоге нашей компании есть статья про kui на английском она немного оутдатед и написана в более строгом стиле. Но если у вас есть англоязычные коллеги и вы считаете что им стоит посмотреть на kui, отправьте им это.
Только OS, только хардкор!)
Творите, выдумывайте, пробуйте!)