0

我遇到了以下脚本,有些东西我无法完全理解

    #!/bin/sh /etc/rc.common
# Copyright (C) 2006-2011 OpenWrt.org

START=50

start() {
        mkdir -m 0755 -p /var/run/vsftpd
        service_start /usr/sbin/vsftpd
}

stop() {
        service_stop /usr/sbin/vsftpd
}

'/etc/rc.common' 在这里如何使用?

这是 rc.common 的内容

#!/bin/sh
# Copyright (C) 2006-2011 OpenWrt.org

. $IPKG_INSTROOT/lib/functions.sh
. $IPKG_INSTROOT/lib/functions/service.sh

initscript=$1
action=${2:-help}
shift 2

start() {
        return 0
}

stop() {
        return 0
}

reload() {
        return 1
}

restart() {
        trap '' TERM
        stop "$@"
        start "$@"
}

boot() {
        start "$@"
}

shutdown() {
        stop
}

disable() {
        name="$(basename "${initscript}")"
        rm -f "$IPKG_INSTROOT"/etc/rc.d/S??$name
        rm -f "$IPKG_INSTROOT"/etc/rc.d/K??$name
}

enable() {
        name="$(basename "${initscript}")"
        disable
        [ -n "$START" -o -n "$STOP" ] || {
                echo "/etc/init.d/$name does not have a START or STOP value"
                return 1
        }
        [ "$START" ] && ln -s "../init.d/$name" "$IPKG_INSTROOT/etc/rc.d/S${START}${name##S[0-9][0-9]}"
        [ "$STOP"  ] && ln -s "../init.d/$name" "$IPKG_INSTROOT/etc/rc.d/K${STOP}${name##K[0-9][0-9]}"
}

enabled() {
        name="$(basename "${initscript}")"
        [ -x "$IPKG_INSTROOT/etc/rc.d/S${START}${name##S[0-9][0-9]}" ]
}

depends() {
        return 0
}

help() {
        cat <<EOF
Syntax: $initscript [command]

Available commands:
        start   Start the service
        stop    Stop the service
        restart Restart the service
        reload  Reload configuration files (or restart if that fails)
        enable  Enable service autostart
        disable Disable service autostart
$EXTRA_HELP
EOF
}

. "$initscript"

ALL_COMMANDS="start stop reload restart boot shutdown enable disable enabled depends ${EXTRA_COMMANDS}"
list_contains ALL_COMMANDS "$action" || action=help
[ "$action" = "reload" ] && action='eval reload "$@" || restart "$@" && :'
$action "$@"

希望你们中的一些人能对此有所了解。谢谢!

PS:我不太明白的另一件事是如何通过简单地将函数名称附加到启动脚本的完整路径来调用脚本中的函数。例如,'/etc/init.d/vsftpd test' 将在 /etc/init.d/vsftpd 或 /etc/rc.common 中执行一个名为 'test' 的函数。但如果 'test' 函数在启动脚本和/etc/rc.common,前者中的函数会运行,而rc.common中的函数不会。

还有,为什么不

'[ "$action" = "reload" ] && action='eval reload "$@" || restart "$@" && :'

简单地写成

'[ "$action" = "reload" ] && action='eval reload "$@" || restart "$@"'

谢谢!

4

1 回答 1

2

execve(2)一个相当流行的 Linux 系统上:

解释器脚本

解释器脚本是一个启用了执行权限的文本文件,其第一行的格式为:

#! interpreter [optional-arg]

解释器必须是本身不是脚本的可执行文件的有效路径名。如果 execve() 的文件名参数指定了解释器脚本,那么解释器将使用以下参数调用:

interpreter [optional-arg] filename arg...

其中 arg... 是 的 argv 参数指向的一系列单词 execve()

对于便携式使用, optional-arg 应该不存在,或者指定为单个单词(即,它不应包含空格);[...]

我没有在野外看到很多使用这个#!/bin/sh filename成语的脚本。我发现它的使用令人困惑。

也许一个简单的测试可以说明。鉴于 test_interpreter.sh 中解释器行的详细信息,这些文件应该存在于 /tmp/test 中,这在这种情况下很重要。

“#!”中命名的脚本 line (rc_interpreter_line) 安排在最初调用的脚本中运行命令,我通过 sourcing_script 变量和 shift 命令执行此操作。您在问题中引用的代码是这个链接的一个更复杂的版本。如果没有这种链接,运行的只是解释器行中命名的文件。

rc_interpreter_line 的内容

echo '===='
echo $0 "$@"

TESTVAR=set

sourcing_script=$1
shift

. "$sourcing_script" "$@"

echo '===='

test_interpreter.sh 的内容

#!/bin/sh /tmp/test/rc_interpreter_line

echo '-----'
echo "My file name is test_interpreter.sh, but \$0 is $0"

echo Command line: $0 "$@"

echo "TESTVAR is '$TESTVAR'"
echo '-----'

exit 0

权限:

sh-4.2$ ls -l
total 8
-rw-r--r-- 1 dev dev 104 Aug 24 13:36 rc_interpreter_line
-rwxr-xr-x 1 dev dev 191 Aug 24 13:36 test_interpreter.sh

样品运行。先直接运行test_interpreter.sh。

sh-4.2$ ./test_interpreter.sh -opt arg1 arg2
====
/tmp/test/rc_interpreter_line ./test_interpreter.sh -opt arg1 arg2
-----
My file name is test_interpreter.sh, but $0 is /tmp/test/rc_interpreter_line
Command line: /tmp/test/rc_interpreter_line -opt arg1 arg2
TESTVAR is 'set'
-----

第二个更清楚地调用 shell。没有execve(2)行为触发,因此 shell 的运行只是运行 test_interpreter.sh 中的命令,将第一行视为注释。

sh-4.2$ sh test_interpreter.sh -opt arg1 arg2
-----
My file name is test_interpreter.sh, but $0 is test_interpreter.sh
Command line: test_interpreter.sh -opt arg1 arg2
TESTVAR is ''
-----

但我个人的偏好是完全避免使用这个成语。在脚本的早期简单地使用命令对我来说要清楚得多,例如:

. /etc/rc.common

...而不是依靠“有趣的”'#!' 行,这样做会在使用时产生不同的./my_script行为sh my_script

于 2013-08-24T19:11:05.550 回答