Last active 19 hours ago

install.sh Raw
1#!/bin/bash
2
3set -e
4
5# Цвета для вывода
6ANSI_RED=$(tput -Txterm setaf 1)
7ANSI_GREEN=$(tput -Txterm setaf 2)
8ANSI_YELLOW=$(tput -Txterm setaf 3)
9ANSI_RESET=$(tput -Txterm sgr0)
10
11# ==================== ФУНКЦИИ ====================
12
13# Проверка прав root
14check_root() {
15 if [ "$EUID" -ne 0 ]; then
16 echo -e "${ANSI_RED}Ошибка: скрипт должен запускаться от root${ANSI_RESET}"
17 exit 1
18 fi
19}
20
21# Запрос имени хоста
22ask_hostname() {
23 local current_hostname=$(hostname)
24 read -e -p "${ANSI_YELLOW}Введите имя хоста: (текущее: $current_hostname) ${ANSI_RESET}" -i "${current_hostname}" NEW_HOSTNAME
25 echo "$NEW_HOSTNAME"
26}
27
28# Запрос - создавать ли пользователя
29ask_create_user() {
30 local choice
31 echo -e "${ANSI_YELLOW}Создать нового пользователя? (y/n):${ANSI_RESET}"
32 read -p "Create user [Y/N]? " choice
33 if [[ "$choice" =~ ^[Yy]$ ]]; then
34 CREATE_USER="yes"
35 else
36 CREATE_USER="no"
37 fi
38}
39
40# Запрос имени пользователя
41ask_username() {
42 local username
43 echo -e "${ANSI_YELLOW}Введите имя нового пользователя:${ANSI_RESET}" >&2
44 read -p "Username: " username
45 if [ -z "$username" ]; then
46 echo -e "${ANSI_RED}Имя пользователя не может быть пустым.${ANSI_RESET}" >&2
47 exit 1
48 fi
49 echo "$username"
50}
51
52# Запрос пароля
53ask_password() {
54 local username=$1
55 local user_pass
56 echo -e "${ANSI_YELLOW}Введите пароль для пользователя $username:${ANSI_RESET}" >&2
57 read -s -p "Password: " user_pass
58 echo >&2
59 if [ -z "$user_pass" ]; then
60 echo -e "${ANSI_RED}Пароль не может быть пустым.${ANSI_RESET}" >&2
61 exit 1
62 fi
63 echo "$user_pass"
64}
65
66# Запрос настройки прокси
67ask_proxy() {
68 echo -e "\n${ANSI_YELLOW}Использовать прокси-сервер apt-cacher-ng? (y/n):${ANSI_RESET}"
69 read -p "Use proxy? " USE_PROXY
70 if [[ "$USE_PROXY" =~ ^[Yy]$ ]]; then
71 echo -e "${ANSI_YELLOW}Введите URL прокси-сервера (например: http://192.168.111.87:3142):${ANSI_RESET}"
72 read -e -p "Proxy URL: " -i "http://192.168.111.87:3142" PROXY_URL
73 if [ -n "$PROXY_URL" ]; then
74 configure_proxy "$PROXY_URL"
75 fi
76 fi
77}
78
79# Настройка прокси для apt
80configure_proxy() {
81 local proxy_url=$1
82 echo -e "${ANSI_GREEN}Настройка прокси для apt...${ANSI_RESET}"
83 cat > /etc/apt/apt.conf.d/02aptproxy << EOF
84Acquire::http::proxy "$proxy_url";
85Acquire::ftp::proxy "$proxy_url";
86EOF
87 echo -e "${ANSI_GREEN}Прокси настроен в /etc/apt/apt.conf.d/02aptproxy${ANSI_RESET}"
88}
89
90# Установка имени хоста
91set_hostname() {
92 local hostname=$1
93 echo -e "\n${ANSI_GREEN}=== Установка имени хоста: $hostname ===${ANSI_RESET}"
94 hostnamectl set-hostname "$hostname"
95 if grep -q "^127.0.1.1" /etc/hosts; then
96 sed -i "s/^127.0.1.1.*/127.0.1.1\t$hostname/" /etc/hosts
97 else
98 echo "127.0.1.1\t$hostname" >> /etc/hosts
99 fi
100 echo -e "${ANSI_GREEN}Имя хоста установлено.${ANSI_RESET}"
101}
102
103# Добавление репозитория MEPHI
104add_mephi_repository() {
105 . /etc/os-release 2>/dev/null || { echo "Failed to determine release"; exit 1; }
106 [ -z "$VERSION_CODENAME" ] && echo "VERSION_CODENAME not found" && exit 1
107
108 local conf_file="/etc/apt/sources.list.d/debian_mephi.list"
109
110 if grep -qs "mirror.mephi.ru.*$VERSION_CODENAME" "$conf_file" 2>/dev/null; then
111 echo -e "Repository for $VERSION_CODENAME already exists in $conf_file"
112 else
113 echo -e "Adding repository for Debian $VERSION_CODENAME"
114 cat >> "$conf_file" << EOF
115
116# Added $(date '+%Y-%m-%d')
117deb http://mirror.mephi.ru/debian/ $VERSION_CODENAME main non-free-firmware
118deb-src http://mirror.mephi.ru/debian/ $VERSION_CODENAME main non-free-firmware
119EOF
120 echo "Repository added, updating package list..."
121 apt update
122 fi
123}
124
125# Обновление системы
126update_system() {
127 echo -e "${ANSI_GREEN}=== Обновление системы ===${ANSI_RESET}"
128 apt update
129 apt -y upgrade
130}
131
132# Установка пакетов
133install_packages() {
134 local packages=(
135 pv nano curl wget sudo lsb-release iptables
136 unzip pigz zstd
137 ncdu gdu
138 screen tmux lynx
139 htop btop iftop mtr ioping
140 git jq yq pwgen
141 bind9-dnsutils net-tools ssh-audit
142 cloud-guest-utils qemu-guest-agent
143 console-setup
144 )
145
146 echo -e "${ANSI_GREEN}Installing packages...${ANSI_RESET}"
147 apt-get update
148 apt-get install -y "${packages[@]}"
149}
150
151# Настройка SSH
152configure_ssh() {
153 echo -e "${ANSI_GREEN}Allowing SSH root login${ANSI_RESET}"
154 sed -i 's/^.*PermitRootLogin.*$/PermitRootLogin yes/' /etc/ssh/sshd_config
155 systemctl restart sshd
156 echo -e "${ANSI_GREEN}Ok.${ANSI_RESET}"
157}
158
159# Настройка таймзоны и локали
160configure_locale() {
161 echo "tzdata tzdata/Areas select Europe" | debconf-set-selections
162 echo "tzdata tzdata/Zones/Europe select Moscow" | debconf-set-selections
163 DEBIAN_FRONTEND=noninteractive dpkg-reconfigure tzdata
164
165 cat > /etc/locale.gen << EOF
166ru_RU.UTF-8 UTF-8
167en_US.UTF-8 UTF-8
168EOF
169
170 locale-gen
171 update-locale LANG=en_US.UTF-8
172 locale -a | grep -E "en_US|ru_RU"
173 source /etc/default/locale
174
175 dpkg-reconfigure console-setup
176 dpkg-reconfigure keyboard-configuration
177}
178
179# Настройка journald
180configure_journald() {
181 mkdir -p /etc/systemd/journald.conf.d
182 cat > /etc/systemd/journald.conf.d/size.conf << EOF
183[Journal]
184SystemMaxUse=1G
185SystemMaxFileSize=50M
186MaxLevelStore=warning
187EOF
188
189 systemctl restart systemd-journald
190}
191
192# Создание пользователя
193create_user() {
194 local username=$1
195 local password=$2
196
197 useradd -m "$username" && echo "$username:$password" | chpasswd && passwd -e "$username"
198 usermod -aG sudo "$username"
199
200 touch "/home/$username/.Xauthority"
201 chown "$username:$username" "/home/$username/.Xauthority"
202
203 echo -e "${ANSI_GREEN}Пользователь $username с паролем '${ANSI_YELLOW}$password${ANSI_GREEN}' создан${ANSI_RESET}"
204}
205
206# Настройка истории для пользователя
207configure_user_history() {
208 local username=$1
209 cat >> "/home/$username/.bashrc" << 'EOF'
210export HISTSIZE=10000
211export HISTFILESIZE=50000
212export HISTTIMEFORMAT="%F %T "
213export HISTCONTROL=ignoredups:ignorespace
214shopt -s histappend
215export PROMPT_COMMAND="history -a; history -c; history -r; $PROMPT_COMMAND"
216export TERM=xterm-256color
217EOF
218 chown "$username:$username" "/home/$username/.bashrc"
219}
220
221# Настройка истории для root
222configure_root_history() {
223 cat >> /root/.bashrc << 'EOF'
224export HISTSIZE=10000
225export HISTFILESIZE=50000
226export HISTTIMEFORMAT="%F %T "
227export HISTCONTROL=ignoredups:ignorespace
228shopt -s histappend
229export PROMPT_COMMAND="history -a; history -c; history -r; $PROMPT_COMMAND"
230export TERM=xterm-256color
231EOF
232 source /root/.bashrc
233}
234
235# Очистка документации git
236clean_git_docs() {
237 rm -rf /usr/share/doc/git/RelNotes
238}
239
240# Финальное сообщение
241finish_setup() {
242 echo -e "${ANSI_GREEN}=== Настройка завершена ===${ANSI_RESET}"
243}
244
245# Запись конфигурационного файла
246write_config_file() {
247 local file_path=$1
248 local create_backup=${2:-true} # создать резервную копию? (по умолчанию true)
249
250 echo -e "${ANSI_GREEN}=== Запись конфигурации в $file_path ===${ANSI_RESET}"
251
252 # Создание резервной копии
253 if [ "$create_backup" = true ] && [ -f "$file_path" ]; then
254 local backup_path="${file_path}.backup.$(date +%Y%m%d_%H%M%S)"
255 cp "$file_path" "$backup_path"
256 echo -e "${ANSI_YELLOW}Создана резервная копия: $backup_path${ANSI_RESET}"
257 fi
258
259 # Создание директории, если её нет
260 local dir_path=$(dirname "$file_path")
261 if [ ! -d "$dir_path" ]; then
262 mkdir -p "$dir_path"
263 echo -e "${ANSI_YELLOW}Создана директория: $dir_path${ANSI_RESET}"
264 fi
265
266 # Чтение содержимого из heredoc
267 cat > "$file_path"
268
269 # Установка правильных прав (опционально)
270 chmod 644 "$file_path"
271
272 echo -e "${ANSI_GREEN}Файл успешно записан: $file_path${ANSI_RESET}"
273}
274
275make_motd_generator() {
276 write_config_file "/etc/update-motd.d/99-mymotd-generator" false << 'EOF'
277 #!/bin/bash
278
279 # Text Color Variables http://misc.flogisoft.com/bash/tip_colors_and_formatting
280 tcLtG="\033[00;37m" # LIGHT GRAY
281 tcDkG="\033[01;30m" # DARK GRAY
282 tcLtR="\033[01;31m" # LIGHT RED
283 tcLtGRN="\033[01;32m" # LIGHT GREEN
284 tcLtBL="\033[01;34m" # LIGHT BLUE
285 tcLtP="\033[01;35m" # LIGHT PURPLE
286 tcLtC="\033[01;36m" # LIGHT CYAN
287 tcW="\033[01;37m" # WHITE
288 tcRESET="\033[0m"
289 tcORANGE="\033[38;5;209m"
290
291 # Time of day
292 HOUR=$(date +"%H")
293 if [ $HOUR -lt 12 -a $HOUR -ge 0 ]; then TIME="morning"
294 elif [ $HOUR -lt 17 -a $HOUR -ge 12 ]; then TIME="afternoon"
295 else TIME="evening"
296 fi
297
298 # System uptime
299 uptime=`cat /proc/uptime | cut -f1 -d.`
300 upDays=$((uptime/60/60/24))
301 upHours=$((uptime/60/60%24))
302 upMins=$((uptime/60%60))
303
304 # System + Memory
305 MEMORY_USED=`free -b | grep Mem | awk '{print $3/$2 * 100.0}'`
306 SWAP_USED=`free -b | grep Swap | awk '{print $3/$2 * 100.0}'`
307 NUM_PROCS=`ps aux | wc -l`
308
309 # IP первого сетевого интерфейса
310 LOCAL_IP=$(ip -4 addr show | grep -oP '(?<=inet\s)\d+\.\d+\.\d+\.\d+' | grep -v '^127\.' | head -n 1)
311
312 # Имя сервера
313 HOSTNAME=$(hostname)
314
315 # ОS
316 OS=$(grep PRETTY_NAME /etc/os-release | cut -d= -f2 | tr -d '"')
317
318 # Пользователь
319 USER_NAME=$(whoami)
320 [ "$USER_NAME" = "root" ] && USER_NAME="${tcLtR}${USER_NAME}${tcRESET}"
321
322 # Load average
323 LOADAVG=$(awk '{print $1" "$2" "$3}' /proc/loadavg)
324 SYS_LOADS=`cat /proc/loadavg | awk '{print $1}'`
325
326 # RAM: всего и свободно (в мегабайтах)
327 RAM_TOTAL=$(free -m | awk '/Mem:/ {print $2}')
328 RAM_FREE=$(free -m | awk '/Mem:/ {print $7}')
329 RAM_FREE_PCT=$(( RAM_FREE * 100 / RAM_TOTAL ))
330
331 # Количество CPU
332 CPU_COUNT=$(nproc)
333
334 # Uptime
335 UPTIME=$(uptime -p)
336
337 # HDD: для корневого раздела /
338 DISK_TOTAL_HUMAN=$(df -h / | awk 'NR==2 {print $2}')
339 DISK_FREE_HUMAN=$(df -h / | awk 'NR==2 {print $4}')
340
341 # Используем df без форматирования для процентов
342 DISK_TOTAL=$(df -k / | awk 'NR==2 {print $2}')
343 DISK_FREE=$(df -k / | awk 'NR==2 {print $4}')
344 DISK_FREE_PCT=$(( DISK_FREE * 100 / DISK_TOTAL ))
345
346 echo -e "$tcDkG ==============================================================="
347 echo -e $tcLtG " Good $TIME! $tcORANGE $LOCAL_IP"
348 echo -e $tcDkG "==============================================================="
349 echo -e $tcLtG " - Hostname :$tcW ${HOSTNAME}"
350 echo -e $tcLtG " - IP Address :$tcW ${LOCAL_IP:-N/A}"
351 echo -e $tcLtG " - User :$tcW ${USER_NAME}"
352 echo -e $tcLtG "==============================================================="
353 echo -e $tcLtG " - OS Release :$tcW ${OS}"
354 echo -e $tcLtG " - Kernel : `uname -a | awk '{print $1" "$3" "$12}'`"
355 echo -e $tcLtG " - Users : Currently `users | wc -w` user(s) logged on"
356 echo -e $tcLtG "==============================================================="
357 echo -e $tcLtG " - Server Time : `date`"
358 echo -e $tcLtG " - System load : ${SYS_LOADS} / ${NUM_PROCS} processes running"
359 echo -e $tcLtG " - Load average : ${LOADAVG}"
360 echo -e $tcLtG " - System uptime : ${upDays} days ${upHours} hours ${upMins} minutes"
361 echo -e $tcLtG "==============================================================="
362 echo -e $tcLtG " - CPU : ${CPU_COUNT} CPU"
363 echo -e $tcLtG " - RAM : ${RAM_TOTAL} MB, ${RAM_FREE} MB (${RAM_FREE_PCT}%) free"
364 echo -e $tcLtG " - HDD : ${DISK_TOTAL_HUMAN}, ${DISK_FREE_HUMAN} (${DISK_FREE_PCT}%) free"
365 echo -e $tcLtG " - Swap used % : ${SWAP_USED}"
366 echo -e $tcDkG "==============================================================="
367 echo -e $tcRESET ""
368EOF
369
370 write_config_file "/etc/issue" false << 'EOF'
371 Debian GNU/Linux 12 \n \l
372
373 Local IP: не доступен
374EOF
375}
376
377install_midnight_commanger() {
378 apt-get -y install mc
379
380 write_config_file "/root/.config/mc/ini" false << 'EOF'
381[Midnight-Commander]
382verbose=true
383shell_patterns=true
384auto_save_setup=true
385preallocate_space=false
386auto_menu=false
387use_internal_view=true
388use_internal_edit=false
389clear_before_exec=true
390confirm_delete=true
391confirm_overwrite=true
392confirm_execute=false
393confirm_history_cleanup=true
394confirm_exit=false
395confirm_directory_hotlist_delete=false
396confirm_view_dir=false
397safe_delete=false
398safe_overwrite=false
399use_8th_bit_as_meta=false
400mouse_move_pages_viewer=true
401mouse_close_dialog=false
402fast_refresh=false
403drop_menus=false
404wrap_mode=true
405old_esc_mode=true
406cd_symlinks=true
407show_all_if_ambiguous=false
408use_file_to_guess_type=true
409alternate_plus_minus=false
410only_leading_plus_minus=true
411show_output_starts_shell=false
412xtree_mode=false
413file_op_compute_totals=true
414classic_progressbar=true
415use_netrc=true
416ftpfs_always_use_proxy=false
417ftpfs_use_passive_connections=true
418ftpfs_use_passive_connections_over_proxy=false
419ftpfs_use_unix_list_options=true
420ftpfs_first_cd_then_ls=true
421ignore_ftp_chattr_errors=true
422editor_fill_tabs_with_spaces=false
423editor_return_does_auto_indent=false
424editor_backspace_through_tabs=false
425editor_fake_half_tabs=true
426editor_option_save_position=true
427editor_option_auto_para_formatting=false
428editor_option_typewriter_wrap=false
429editor_edit_confirm_save=true
430editor_syntax_highlighting=true
431editor_persistent_selections=true
432editor_drop_selection_on_copy=true
433editor_cursor_beyond_eol=false
434editor_cursor_after_inserted_block=false
435editor_visible_tabs=true
436editor_visible_spaces=true
437editor_line_state=false
438editor_simple_statusbar=false
439editor_check_new_line=false
440editor_show_right_margin=false
441editor_group_undo=true
442editor_state_full_filename=true
443editor_ask_filename_before_edit=false
444nice_rotating_dash=true
445shadows=true
446mcview_remember_file_position=false
447auto_fill_mkdir_name=true
448copymove_persistent_attr=true
449pause_after_run=1
450mouse_repeat_rate=100
451double_click_speed=250
452old_esc_mode_timeout=1000000
453max_dirt_limit=10
454num_history_items_recorded=60
455vfs_timeout=60
456ftpfs_directory_timeout=900
457ftpfs_retry_seconds=30
458fish_directory_timeout=900
459editor_tab_spacing=8
460editor_word_wrap_line_length=72
461editor_option_save_mode=0
462editor_backup_extension=~
463editor_filesize_threshold=64M
464editor_stop_format_chars=-+*\\,.;:&>
465mcview_eof=
466skin=modarcon16
467
468filepos_max_saved_entries=1024
469
470[Layout]
471output_lines=0
472left_panel_size=86
473top_panel_size=0
474message_visible=false
475keybar_visible=true
476xterm_title=true
477command_prompt=true
478menubar_visible=true
479free_space=true
480horizontal_split=false
481vertical_equal=true
482horizontal_equal=true
483
484[Misc]
485timeformat_recent=%b %e %H:%M
486timeformat_old=%b %e %Y
487ftp_proxy_host=gate
488ftpfs_password=anonymous@
489display_codepage=UTF-8
490source_codepage=Other_8_bit
491autodetect_codeset=
492spell_language=en
493clipboard_store=
494clipboard_paste=
495
496[Colors]
497base_color=
498linux=
499color_terminals=
500
501xterm-256color=
502
503[Panels]
504show_mini_info=true
505kilobyte_si=false
506mix_all_files=false
507show_backups=true
508show_dot_files=true
509fast_reload=false
510fast_reload_msg_shown=false
511mark_moves_down=true
512reverse_files_only=true
513auto_save_setup_panels=false
514navigate_with_arrows=false
515panel_scroll_pages=true
516panel_scroll_center=false
517mouse_move_pages=true
518filetype_mode=true
519permission_mode=false
520torben_fj_mode=false
521quick_search_mode=2
522select_flags=6
523
524[Panelize]
525Find *.orig after patching=find . -name \\*.orig -print
526Find SUID and SGID programs=find . \\( \\( -perm -04000 -a -perm /011 \\) -o \\( -perm -02000 -a -perm /01 \\) \\) -print
527Find rejects after patching=find . -name \\*.rej -print
528Modified git files=git ls-files --modified
529
530EOF
531
532 write_config_file "/root/.config/mc/panels.ini" false << 'EOF'
533[New Left Panel]
534display=listing
535reverse=false
536case_sensitive=true
537exec_first=false
538sort_order=name
539list_mode=full
540brief_cols=2
541user_format=half type name | size | owner
542user_status0=half type name | size | perm
543user_status1=half type name | size | perm
544user_status2=half type name | size | perm
545user_status3=half type name | size | perm
546user_mini_status=false
547filter_flags=7
548list_format=user
549
550[New Right Panel]
551display=listing
552reverse=false
553case_sensitive=true
554exec_first=false
555sort_order=name
556list_mode=full
557brief_cols=2
558user_format=half type name | size | owner
559user_status0=half type name | size | perm
560user_status1=half type name | size | perm
561user_status2=half type name | size | perm
562user_status3=half type name | size | perm
563user_mini_status=false
564filter_flags=7
565list_format=user
566
567[Dirs]
568current_is_left=false
569other_dir=/root
570EOF
571}
572
573# Обновление GPG ключей Debian
574fix_debian_keys() {
575 echo -e "${ANSI_GREEN}=== Исправление GPG-ключей Debian ===${ANSI_RESET}"
576
577 # 1. Полностью удаляем кэш списков пакетов
578 rm -rf /var/lib/apt/lists/*
579
580 # 2. Удаляем старые проблемные ключи из trusted.gpg.d и связки apt-key
581 rm -f /etc/apt/trusted.gpg.d/*54404762BBB6E853* \
582 /etc/apt/trusted.gpg.d/*6ED0E7B82643E131* 2>/dev/null || true
583 apt-key del 54404762BBB6E853 2>/dev/null || true
584 apt-key del 6ED0E7B82643E131 2>/dev/null || true
585
586 # 3. Очищаем кэш apt
587 apt-get clean
588
589 # 4. Принудительно переустанавливаем пакет с ключами (без проверки подписи)
590 apt-get install --reinstall -y --allow-unauthenticated debian-archive-keyring
591
592 # 5. Если прокси-сервер используется, настраиваем переменные окружения для gpg
593 if [ -n "$PROXY_URL" ]; then
594 export http_proxy="$PROXY_URL"
595 export https_proxy="$PROXY_URL"
596 fi
597
598 # 6. Пытаемся получить ключи напрямую с keyserver (альтернативный источник)
599 echo "Получение свежих ключей с keyserver..."
600 gpg --keyserver keyserver.ubuntu.com --recv-keys 54404762BBB6E853 6ED0E7B82643E131 2>/dev/null || \
601 gpg --keyserver pgp.mit.edu --recv-keys 54404762BBB6E853 6ED0E7B82643E131 2>/dev/null || \
602 gpg --keyserver keys.openpgp.org --recv-keys 54404762BBB6E853 6ED0E7B82643E131 2>/dev/null || true
603
604 # 7. Экспортируем ключи в директорию apt (если были получены)
605 gpg --export 54404762BBB6E853 > /etc/apt/trusted.gpg.d/debian-security-automatic.gpg 2>/dev/null || true
606 gpg --export 6ED0E7B82643E131 > /etc/apt/trusted.gpg.d/debian-archive-automatic.gpg 2>/dev/null || true
607
608 # 8. Снова очищаем списки и обновляем
609 rm -rf /var/lib/apt/lists/*
610 apt-get update --allow-insecure-repositories || true
611
612 echo -e "${ANSI_GREEN}Ключи обновлены.${ANSI_RESET}"
613}
614
615# ==================== ОСНОВНАЯ ЛОГИКА ====================
616
617main() {
618 # check_root
619
620 echo -e "${ANSI_GREEN}=== Настройка системы ===${ANSI_RESET}"
621
622 NEW_HOSTNAME=$(ask_hostname)
623
624 ask_create_user
625
626 echo $CREATE_USER
627
628 if [ "$CREATE_USER" = "yes" ]; then
629 NEW_USER=$(ask_username)
630 USER_PASS=$(ask_password "$NEW_USER")
631 create_user "$NEW_USER" "$USER_PASS"
632 configure_user_history "$NEW_USER"
633 fi
634
635 ask_proxy
636 fix_debian_keys
637
638 # Выполнение настроек
639 configure_root_history
640
641 set_hostname "$NEW_HOSTNAME"
642 add_mephi_repository
643 update_system
644 install_packages
645 configure_ssh
646 configure_locale
647 configure_journald
648 clean_git_docs
649
650 make_motd_generator
651
652 install_midnight_commanger
653
654 finish_setup
655}
656
657# Запуск главной функции
658main "$@"