Shell编程

参考:Shell教程|菜鸟教程

shell 环境

#!/bin/bash:脚本第一行,告诉系统使用什么解释器来执行,即使用哪一个 shell

注释

  • 单行注释用#
  • 多行注释用 :<< 开头,! 包裹注释内容
# 这是一个单行注释
:<<!
这是多行注释
这是多行注释
这是多行注释
!

变量

# 变量 = 号两侧不能有空格,大小写敏感
my_name="lanhuli"
echo ${my_name}
unset my_name  # 删除变量

双引号和单引号的区别

  • 单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
  • 单引号字符串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
  • 双引号里可以有变量
  • 双引号里可以出现转义字符

只读变量

myUrl="https://www.google.com"  
readonly myUrl  
myUrl="https://www.runoob.com"
# 执行报错,NAME: This variable is read only.

字符串变量

#!/bin/bash
name="Shell"
url="http://1.1.1.1/"
# 拼接字符串
str1="${url}${name}"
str2=${url}${name}  # 中间不能有空格
str3="${url}:${name}"  # 中间可以出现其它字符,包括空格
######### 输出 ########
http://1.1.1.1/Shell
http://1.1.1.1/Shell
http://1.1.1.1/:Shell

获取字符串长度

string="abcdefg"
echo ${#string}  # 变量为字符串时,${#string} 等价于 ${#string[0]}
echo ${#string[0]}
echo ${string:1:4}  # 字符串第 2 个字符开始截取 4 个字符

整型变量

运算符

关系运算符

关系运算符只支持数字,不支持字符串,除非字符串的值是数字。

假定变量 a 为 10,变量 b 为 20:

字符串运算符

假定变量 a 为 "abc",变量 b 为 "efg":

数组

# 创建数组
my_array=(A B "C" D)
echo "第一个元素为: ${my_array[0]}"  
echo "第二个元素为: ${my_array[1]}"  
echo "第三个元素为: ${my_array[2]}"  
echo "第四个元素为: ${my_array[3]}"

关联数组

declare -A array_name  # 创建关联数组
declare -A site  
site["google"]="www.google.com"  
site["runoob"]="www.runoob.com"  
site["taobao"]="www.taobao.com"  
  
echo ${site["runoob"]}

获取数组中所有的元素

declare -A site  
site["google"]="www.google.com"  
site["runoob"]="www.runoob.com"  
site["taobao"]="www.taobao.com"  
  
echo "数组的键为: ${!site[*]}"  
echo "数组的键为: ${!site[@]}"

获取数组的长度

my_array[0]=A  
my_array[1]=B  
my_array[2]=C  
my_array[3]=D  
  
echo "数组元素个数为: ${#my_array[*]}"  
echo "数组元素个数为: ${#my_array[@]}"

流程控制

if分支语句

a=10  
b=20  
if [ $a == $b ]  
then  
	echo "a 等于 b"
else
	echo "a 不等于 b"
fi

注意:如果 else 分支没有语句执行,就不要写这个 else。

多分支语句

a=10  
b=20  
if [ $a == $b ]  
then  
   echo "a 等于 b"  
elif [ $a -gt $b ]  
then  
   echo "a 大于 b"  
elif [ $a -lt $b ]  
then  
   echo "a 小于 b"  
else  
   echo "没有符合的条件"  
fi 

for 循环

while 循环

#!/bin/bash
num=2
while ((num<100)) #数值与运算符可以没有空格,变量的使用时也可以不使用$num
do
	echo "$num" 
	((num=num*2))
done

sh 脚本接收参数

  • $0:脚本自身的名称(如 11.sh
  • $1:第一个参数,$2 是第二个参数,依此类推
  • $#:参数的个数
  • $@:所有参数的列表
#!/bin/bash
# 检查参数数量
if [ $# -eq 0 ]; then
    echo "Usage: $0 {start|stop}"
    exit 1
fi

# 判断第一个参数
if [ "$1" = "start" ]; then
    echo "执行启动操作..."
    # 启动命令
elif [ "$1" = "stop" ]; then
    echo "执行停止操作..."
    # 停止命令
else
    echo "错误:无效参数 '$1'"
    echo "可用参数: start, stop"
    exit 1
fi

shell 脚本例子

安装 docker 脚本

从自己的服务器下载 docker 离线安装包

#!/bin/bash

url="url"
docker_version="docker-23.0.6.tgz"
docker_compose_version="docker-compose-2.17.0"

# 判断网络环境,优先内网
echo "正在检测网络环境,请稍等..."
ping 192.168.0.211 -c 3 &> /dev/null
if [ $? -eq 0 ]
then
        echo "检测到网络环境可通内网"
        url="http://192.168.0.211/"
else
        # 内网不通,检测外网
        ping oss.lanhuli.top -c 3 &> /dev/null
        if [ $? -eq 0 ]
        then
                echo "检测到网络环境可通外网"
                url="http://oss.lanhuli.top/"
        fi
fi

# 安装docker
echo "正在下载docker安装包,请稍等..."
wget ${url}${docker_version} &> /dev/null
if [ $? -eq 0 ]
then
        echo ${docker_version}"安装包下载成功"
else
        echo ${docker_version}"安装包下载失败"
fi

tar -xf ${docker_version}
chmod 755 ./docker/*
cp ./docker/* /usr/bin/
rm -rf docker
rm -rf ${docker_version}
echo "docker安装完毕"
echo "docker版本:"`docker -v`


# 安装docker-compose
echo "正在下载docker-compose"
wget ${url}${docker_compose_version} &> /dev/null
if [ $? -eq 0 ]
then
        echo ${docker_compose_version}"下载成功"
else
        echo ${docker_compose_version}"下载失败"
fi

chmod 755 ${docker_compose_version}
mv ${docker_compose_version} /usr/local/bin/docker-compose
echo "docker-compose安装完毕"
echo "docker-compose版本:"`docker-compose -v`

# 将 docker 注册成系统服务
touch /etc/systemd/system/docker.service
cat > /etc/systemd/system/docker.service<< EOF
[Unit]
Description=Docker Application Container Engine
Documentation=https://docs.docker.com
After=network-online.target firewalld.service
Wants=network-online.target

[Service]
Type=notify
ExecStart=/usr/bin/dockerd
ExecReload=/bin/kill -s HUP $MAINPID
LimitNOFILE=infinity
LimitNPROC=infinity
TimeoutStartSec=0
Delegate=yes
KillMode=process
Restart=on-failure
StartLimitBurst=3
StartLimitInterval=60s

[Install]
WantedBy=multi-user.target
EOF

chmod +x /etc/systemd/system/docker.service
systemctl daemon-reload
echo "注册完毕,可以使用systemctl命令启动docker"

安装 frp 脚本

#!/bin/bash
# wget http://oss.lanhuli.top/frp_0.60.0_linux_amd64.tar.gz
wget http://192.168.0.211/frp_0.60.0_linux_amd64.tar.gz
tar -xzf frp_0.60.0_linux_amd64.tar.gz

cat > frp_0.60.0_linux_amd64/frps.toml << EOF
bindPort = 16669
auth.token = "xujiajun#7732."
EOF

mv frp_0.60.0_linux_amd64 /opt/frp_0.60.0_linux_amd64
rm frp_0.60.0_linux_amd64.tar.gz

# 注册成系统服务
touch /etc/systemd/system/frps.service
cat > /etc/systemd/system/frps.service<< EOF
[Unit]
# 服务名称,可自定义
Description = frp server
After = network.target syslog.target
Wants = network.target

[Service]
Type = simple
# 启动 frps 的命令,需修改为您的 frps 的安装路径
ExecStart = /opt/frp_0.60.0_linux_amd64/frps -c /opt/frp_0.60.0_linux_amd64/frps.toml

[Install]
WantedBy = multi-user.target
EOF

chmod +x /etc/systemd/system/frps.service
systemctl daemon-reload

监控服务是否挂掉并自动重新启动

脚本主要负责查询服务是否运行,没有运行的话就启动服务。然后将这个脚本通过 crontab 定时任务每分钟运行一次

#!/bin/bash

# 检测服务是否运行
result=`systemctl status nginx | grep "active (running)"`
#echo ${result}

if [[ "$result" != "" ]]
then
	echo "nginx已启动"
else
	echo "nginx未启动,立即启动nginx"
	systemctl restart nginx
fi

上述脚本逻辑:通过 systemctl 命令查看 nginx 的运行状态,如果是启动状态,就会有关键词 active (running) 在其中,result 就不为空,通过这个来判断 nginx 是启动还是关闭状态。如果 result 为空,说明 nginx 停了,可以使用 systemctl start nginx 命令重新启动 nginx。

把脚本通过 crontab 定时任务每隔 1 分钟运行检查一次,就可以达到目的。输入 crontab -e 命令,在文件末尾添加 */1 * * * * /root/start_nginx.sh 然后保存

注意:不一定非要使用 systemctl status nginx | grep "active (running)" 进行判断,也可以使用 ps 命令查看是否存在进程。

#!/bin/bash

# 检测frpc是否运行
if [[ `systemctl status frps | grep "active (running)"` = "" ]];then systemctl start frps;fi

# 需要将 frps 注册为 systemd 服务

PostgreSQL SQL 注入练习靶场

#!/bin/bash
# 检查参数数量
if [ $# -eq 0 ]; then
    echo "Usage: $0 {start|stop|down}"
    exit 1
fi

# 判断第一个参数
if [ "$1" = "start" ]; then
    echo "执行启动操作..."
    # 检查环境是否已经创建
    result1=$(docker ps -a --format "table {{.Names}}" | grep -w pgsql-injection-pgsql)
    result2=$(docker ps -a --format "table {{.Names}}" | grep -w pgsql-injection-php-apache)
    #echo ${result1}

    if [[ "$result1" != "" && "$result2" != "" ]]; then
        # 环境已创建,直接启动即可
        docker start pgsql-injection-pgsql pgsql-injection-php-apache
    else
        # 创建靶场环境文件夹
        mkdir -p /opt/sql-injection-pgsql
        # 下载靶场代码文件
        wget -O sql-injection-pgsql.php http://oss.lanhuli.top/bachang/sql-injection-pgsql.php
        mv sql-injection-pgsql.php /opt/sql-injection-pgsql/index.php
        # 创建单独的docker网络
        docker network create pgsql-injection
        # 启动容器pgsql
        docker run -d --name pgsql-injection-pgsql -e POSTGRES_PASSWORD=admin8848 -e POSTGRES_USER=postgres --restart=always --network pgsql-injection --network-alias pgsql-injection-pgsql registry.cn-hangzhou.aliyuncs.com/lanhuli/postgres:14.17
        # 启动容器php-apache
        docker run -d --name pgsql-injection-php-apache -p 17287:80 -v /opt/sql-injection-pgsql:/var/www/html --restart=always --network pgsql-injection --network-alias pgsql-injection-php-apache registry.cn-hangzhou.aliyuncs.com/lanhuli/php:8.0-apache
    fi
elif [ "$1" = "stop" ]; then
    echo "执行停止操作..."
    docker stop pgsql-injection-pgsql pgsql-injection-php-apache
elif [ "$1" = "down" ]; then
    echo "执行销毁操作..."
    docker stop pgsql-injection-pgsql pgsql-injection-php-apache
    docker rm pgsql-injection-pgsql pgsql-injection-php-apache
    docker network rm pgsql-injection
    rm -rf /opt/sql-injection-pgsql
else
    echo "错误:无效参数 '$1'"
    echo "可用参数: start, stop,down"
    exit 1
fi

其它小知识

关于双小括号使用

参考: https://blog.csdn.net/u011479200/article/details/79603385

a=10
b=20
if (( $a == $b ))
then
   echo "a 等于 b"
elif (( $a > $b ))
then
   echo "a 大于 b"
elif (( $a < $b ))
then
   echo "a 小于 b"
else
   echo "没有符合的条件"
fi

[]

a=10
b=20
if [ $a == $b ]
then
   echo "a 等于 b"
elif [ $a -gt $b ]
then
   echo "a 大于 b"
elif [ $a -lt $b ]
then
   echo "a 小于 b"
else
   echo "没有符合的条件"
fi

关于 == 、=和-eq

在 shell 中,=和 == 运算符都可以用于判断两个字符串、两个字符串变量是否相同,== 支持模式匹配,而= 不支持模式匹配。使用 -eq 来判断两个整数是否相等。

判断命令是否执行成功

# shell中使用符号“$?”来显示上一条命令执行的返回值,如果为0则代表执行成功,其他表示失败。
ping 192.168.0.110 -c 3 &>/dev/null
if [ $? -eq 0 ]
then
	echo "能ping通"
else
	echo "不能ping通"
fi