Shell script — различия между версиями

Материал из pNp Wiki
Перейти к: навигация, поиск
(Условия)
Строка 340: Строка 340:
  
 
== Условия ==
 
== Условия ==
 
+
==== if/then ====
 
Общие правила построения условий <code>if/then</code> условий:
 
Общие правила построения условий <code>if/then</code> условий:
  
Строка 357: Строка 357:
 
count="$#"
 
count="$#"
  
if [ -n "$count" ]; then
+
if [ "$count" -gt 1 ]; then
 +
    echo "Number of args greater than one"
 +
fi
  
 
for i in "$@"; do
 
for i in "$@"; do
Строка 365: Строка 367:
 
done
 
done
  
     echo "You do not enter any arguments for script!"
+
exit 0
 +
[root@vm-01 ~]# ./test.sh
 +
[root@vm-01 ~]# ./test.sh 1
 +
Current argument is: 1
 +
Total amount of args: 1
 +
Whole arg string: 1
 +
[root@vm-01 ~]# ./test.sh 1 2
 +
Number of args greater than one
 +
Current argument is: 1
 +
Total amount of args: 2
 +
Whole arg string: 1 2
 +
Current argument is: 2
 +
Total amount of args: 2
 +
Whole arg string: 1 2
 +
[root@vm-01 ~]#
 +
</syntaxhighlight>
 +
 
 +
==== if/then/else ====
 +
Общие правила построения условий <code>if/then/else</code> условий:
 +
 
 +
<syntaxhighlight lang="bash">
 +
if условие; then
 +
  действие
 +
  действие
 +
  ...
 +
else
 +
  действие
 +
  действие
 +
  ...
 +
fi
 +
</syntaxhighlight>
 +
Например:
 +
<syntaxhighlight lang="bash">
 +
[root@vm-01 ~]# cat test.sh
 +
#!/bin/bash
 +
 
 +
count="$#"
 +
 
 +
if [ "$count" == 0 ]; then
 +
     echo "You did not enter any args!"  
 
     exit 1
 
     exit 1
 +
else
 +
 +
for i in "$@"; do
 +
    echo "Current argument is: $i"
 +
    echo "Total amount of args: ${count}"
 +
    echo "Whole arg string: $*"
 +
done
 
fi
 
fi
  
 
exit 0
 
exit 0
 
[root@vm-01 ~]# ./test.sh  
 
[root@vm-01 ~]# ./test.sh  
You do not enter any arguments for script!
+
You did not enter any args!
[root@vm-01 ~]# ./test.sh 1
+
[root@vm-01 ~]# echo $?
 +
1
 +
[root@vm-01 ~]# ./test.sh 0 1
 +
Current argument is: 0
 +
Total amount of args: 2
 +
Whole arg string: 0 1
 
Current argument is: 1
 
Current argument is: 1
Total amount of args: 1
+
Total amount of args: 2
Whole arg string: 1
+
Whole arg string: 0 1
You do not enter any arguments for script!
+
[root@vm-01 ~]# echo $?
 +
0
 
[root@vm-01 ~]#
 
[root@vm-01 ~]#
 
</syntaxhighlight>
 
</syntaxhighlight>
 
  
 
== Ссылки ==
 
== Ссылки ==

Версия 11:48, 4 апреля 2018

Основы создания скриптов на bash

Предварительные требования

  • Виртуальная машина с двумя сетевыми интерфейсами
  • Установленные пакеты: bash, bash-completion, vim-minimal, vim-enhanced

Общая информация

Любой скрипт, начинается с "магической строки" - сигнатуры, которая говорит командному интерпретатору как обрабатывать файл. Для скриптов, такая последовательность носит называние she-bang (сокращение от sharp-bang):

#!

Соответственно, если файл должен обрабатывать bash, то это необходимо указать после "магической строки". То есть, she-bang будет иметь следующий вид:

#!/bin/bash

Если скрипт предназначен для пользователя, то должны быть выставлен необходимый владелец файла, и права на запуск скрипта 755. Поскольку скрипт является выполняемым, то по первым литерам названия, коммандный интерпертатор производит поиск названия в переменной $PATH, соответственно, лучше выбрать имя для скрипта, которое не совпадает с имеющейся утилитой. Для просмотра переменной $PATH надо выполнить следующую команду:

[root@vm-01 ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@vm-01 ~]#

Поскольку текущая директория /root отсутствует в переменной $PATH, то запускать скрипты необходимо следующим образом:

[root@vm-01 ~]# ls -lahi test.sh 
33969143 -rwxr-xr-x. 1 root root 34 Mar 26 10:38 test.sh
[root@vm-01 ~]# ./test.sh 
Hello, world!
[root@vm-01 ~]#

Вывод на экран

Скрипт test.sh использует комманду echo для отображения текстовой строки. По-умолчанию отображение производится на стандартный поток вывода (STDOUT):

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

echo "Hello, world!"
[root@vm-01 ~]#

Хорошим тоном считается выводить ошибки на поток ошибок (STDERR):

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

echo "Hello, world!"
echo "Something has happend! Can not proceed!" >&2
[root@vm-01 ~]#

Запустим скрипт таким образом, что бы поток ошибок направлялся в файл:

[root@vm-01 ~]# ./test.sh 2>test.log
Hello, world!
[root@vm-01 ~]# cat test.log 
Something has happend! Can not proceed!
[root@vm-01 ~]#

При выводе на экран, следует учесть, что некоторые символы могут иметь специальное значение, в следствие чего, использование таких символов может привести к результатам отличным от ожидаемого. Например, если необходимо узнать содержимое PATH, то следует написать так:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

echo "$PATH"
[root@vm-01 ~]# ./test.sh 
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@vm-01 ~]#

Если необходимо вывести именно строку $PATH, то следует использовать символы экранирования (двойные кавычки ", одинарные кавычки ', либо обратный слэш \):

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

echo "\$PATH"
echo '$PATH'

[root@vm-01 ~]# ./test.sh 
$PATH
$PATH
[root@vm-01 ~]#

Следует заметить, что двойные кавчки экранируют не все спецсимволы, поэтому если необходимо вывести какую-либо информацию как текст, то следует использовать одинарные кавычки, которые интерпретируют любую включенную в них информацию как текст! Обратный слэш (\) экранирует только символ следующий за ним!

Использование переменных

Переменная - это область памяти, которая хранит некое значение. К этой области памяти можно обращаться по имени. Название переменной не может начинаться с цифры! Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

one=1
echo "This is number $one"
 
[root@vm-01 ~]# ./test.sh 
This is number 1
[root@vm-01 ~]#

Хорошим тоном считается заключать значения, присваиваемые переменными в двойные кавычки, к примеру, потому, что в значении могут содержаться пробелы, а пробелы интерпретируются bash'ем как разделитель слова. Так же, хорошим тоном считается заключение переменных в фигурные скобки, так как переменные могут конкатенироваться непосредственно друг за другом, поэтому, во избежании получения неожиданных результатов, переменные следует записывать как ${one}. Пример конкатенцации переменных:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

one="1"
two="2"
echo "This is number twelve: ${one}${two}"

[root@vm-01 ~]# ./test.sh 
This is number twelve: 12
[root@vm-01 ~]#

Расширения командного интерпретатора

Кроме переменных, bash позволяет использовать подстановку команд и арифметические операции

Подстановка команд

Подстановка команд осуществляется путем помещения команды в обратные кавычки `, либо путем указания переменной в круглых скобках (). Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash
cli="`which bash`"
curl="$(which curl)"
echo "My bash is here: ${cli} and curl is here: ${curl}" 
[root@vm-01 ~]# ./test.sh 
My bash is here: /usr/bin/bash and curl is here: /usr/bin/curl
[root@vm-01 ~]#

Второй способ предпочтительнее, так как он поддерживает вложенность команд. Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash
count="$($(which ls) -l *.conf | $(which wc) -l)"
echo "Amount of conf files in /root directory: ${count}"

[root@vm-01 ~]# ./test.sh 
Amount of conf files in /root directory: 4
[root@vm-01 ~]#

Арифметика

Арифметические операции осуществляются путем помещения команды в квадратные скобки ([]), которым предшествует знак $. Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash
count1=$[ 3 * 3 ]
count2=$[2+2]
echo "${count1}"
echo "${count2}"

[root@vm-01 ~]# ./test.sh 
9
4
[root@vm-01 ~]#

Часто применяемые арифметические действия и их значения

Оператор Значение
<переменная>++ постинкремент переменной
<переменная>-- постдекремент переменной
++<переменная> преинкремент переменной
--<переменная> предекремент переменной
- вычитание, унарный минус
+ сложение, унарный плюс
** возведение в степень
* умножение
/ деление
 % остаток от деления

Если в арифметическом выражении есть несколько действий, то bash выполняет такие операции в соответствии с правилами математики. Таблица приоритета арифметических действий:

Оператор Значение
<переменная>++, <переменная>-- постинкремент и постинкремент переменных
++<переменная>, --<переменная> преинкремент и предекремент переменных
-, + унарный минус и плюс
** возведение в степень
*, /, % умножение, деление, остаток от деления
+, - сложение, вычитание

Циклы bash

Для повторяемых действий в bash имеются циклы for, while и until

Использование цикла for

Синтаксис цикла выглядит следующим образом:

for переменная in список; do команды; done

Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

count=0

for i in $(ls /root/*.conf); do 
    count=$[count+1]
    echo "${count}"
done
[root@vm-01 ~]# ./test.sh 
1
2
3
4
[root@vm-01 ~]

Специальные переменные bash

Переменная Значение
$? код завершения программы (как правило: 0 - успешное завершение, все отилчное от 0 - ошибка)
$0 имя самой программы
$1..$9 аргументы скрипта
$@ список аргументов ($1,$2,$3..)
$* список аргументов, рассматриваемых как единый аргумент ("$1 $2 $3..")
$# количество переданных аргументов
[root@vm-01 ~]# cat test.sh 
#!/bin/bash

count="$#"

for i in "$@"; do
    echo "Current argument is: $i"
    echo "Total amount of args: ${count}"
    echo "Whole arg string: $*"
done
[root@vm-01 ~]# ./test.sh 1 2 3 4 5 6 7 8 9
Current argument is: 1
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 2
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 3
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 4
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 5
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 6
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 7
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 8
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 9
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
[root@vm-01 ~]# echo $?
0
[root@vm-01 ~]#

Сравнения

Для сравнения переменных в bash, используется либо команда test, либо внутренние команды [[]].

Числовые сравнения

Оператор Значение Пример
-eq равно [ "$a" -eq "$b" ]
-ne неравно [ "$a" -ne "$b" ]
-gt больше чем [ "$a" -gt "$b" ]
-ge больше или равно [ "$a" -ge "$b" ]
-lt меньше чем [ "$a" -lt "$b" ]
-le меньше или равно [ "$a" -le "$b" ]

Строковые сравнения

Оператор Значение Пример
= равно [ "$a" = "$b" ]
== равно [ "$a" == "$b" ]
 != неравно [ "$a" != "$b" ]

Логические операторы AND и OR

Оператор Значение
&& Следующая команда выполняется только, если результат предыдущий был успешен (истина)
|| Следующая команда выполняется, если результат предыдущий был неуспешен (ложь)

Условия

if/then

Общие правила построения условий if/then условий:

if условие; then
   действие
   действие
   ...
fi

Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

count="$#"

if [ "$count" -gt 1 ]; then
    echo "Number of args greater than one"
fi

for i in "$@"; do
    echo "Current argument is: $i"
    echo "Total amount of args: ${count}"
    echo "Whole arg string: $*"
done

exit 0
[root@vm-01 ~]# ./test.sh
[root@vm-01 ~]# ./test.sh 1
Current argument is: 1
Total amount of args: 1
Whole arg string: 1
[root@vm-01 ~]# ./test.sh 1 2
Number of args greater than one
Current argument is: 1
Total amount of args: 2
Whole arg string: 1 2
Current argument is: 2
Total amount of args: 2
Whole arg string: 1 2
[root@vm-01 ~]#

if/then/else

Общие правила построения условий if/then/else условий:

if условие; then
   действие
   действие
   ...
else 
   действие
   действие
   ...
fi

Например:

[root@vm-01 ~]# cat test.sh 
#!/bin/bash

count="$#"

if [ "$count" == 0 ]; then
    echo "You did not enter any args!" 
    exit 1
else

for i in "$@"; do
    echo "Current argument is: $i"
    echo "Total amount of args: ${count}"
    echo "Whole arg string: $*"
done
fi

exit 0
[root@vm-01 ~]# ./test.sh 
You did not enter any args!
[root@vm-01 ~]# echo $?
1
[root@vm-01 ~]# ./test.sh 0 1
Current argument is: 0
Total amount of args: 2
Whole arg string: 0 1
Current argument is: 1
Total amount of args: 2
Whole arg string: 0 1
[root@vm-01 ~]# echo $?
0
[root@vm-01 ~]#

Ссылки

Интерпретатор командного языка shell
Программирование на языке командного интерпретатора - shell
Bash
Advanced Bash-Scripting Guide