run-parts(8) 实用程序的目的

run-parts(8) 实用程序的目的

至少在基于 Debian 的系统下,有一个run-parts实用程序工具各种脚本中使用的包。例如在 /etc/X11/Xsession.它将运行目录中找到的所有可执行文件。为什么需要运行部件,而可以find-perm选项或test实用程序一起使用?另外,什么才算run-parts可执行文件?看起来它不只是检查文件权限:

# run-parts --list --lsbsysinit /etc/X11/Xsession.d | tail -1
/etc/X11/Xsession.d/90x11-common_ssh-agent
# ls -l /etc/X11/Xsession.d/90x11-common_ssh-agent
-rw-r--r-- 1 root root 629 2010-11-02 23:17 /etc/X11/Xsession.d/90x11-common_ssh-agent
# head /etc/X11/Xsession.d/90x11-common_ssh-agent
# $Id: 90x11-common_ssh-agent 305 2005-07-03 18:51:43Z dnusinow $

# This file is sourced by Xsession(5), not executed.

STARTSSH=
SSHAGENT=/usr/bin/ssh-agent
SSHAGENTARGS=

if has_option use-ssh-agent; then
  if [ -x "$SSHAGENT" ] && [ -z "$SSH_AUTH_SOCK" ] \
# 

答案1

您可以使用find代替run-parts,没有办法表明哪个更好。但我认为使用run-parts更短(更少的打字)并使你的脚本更易于维护。一个例子是/etc/crontab

# /etc/crontab: system-wide crontab
# Unlike any other crontab you don't have to run the `crontab'
# command to install the new version when you edit this file
# and files in /etc/cron.d. These files also have username fields,
# that none of the other crontabs do.

SHELL=/bin/sh
PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin

# m h dom mon dow user  command
17 *    * * *   root    cd / && run-parts --report /etc/cron.hourly
25 6    * * *   root    test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.daily )
47 6    * * 7   root    test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.weekly )
52 6    1 * *   root    test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.monthly )

debianutils对于第二个问题,你可以在源码中找到答案。在文件 中run-parts.c,第 198 行:

/* Execute a file */                                                            
void run_part(char *progname)                                                   
{ 
     ....
     args[0] = progname;                                                         
     execv(progname, args);                                                      
     error("failed to exec %s: %s", progname, strerror(errno));                  
     exit(1);
     ....
}

你可以看到run-parts使用execv系统调用。因此,如果您的文件不是二进制可执行文件或Interpreter script,execv则无法运行该文件。

笔记

  • 什么是Interpreter script

来自man execve, 部分Interpreter scripts:

Interpreter scripts
       An  interpreter  script  is  a  text  file  that has execute permission
       enabled and whose first line is of the form:

           #! interpreter [optional-arg]

       The interpreter must be a valid pathname for an executable which is not
       itself  a  script.   If  the filename argument of execve() specifies an
       interpreter script, then interpreter will be invoked with the following
       arguments:

           interpreter [optional-arg] filename arg...

       where arg...  is the series of words pointed to by the argv argument of
       execve().

       For portable use, optional-arg should either be absent, or be specified
       as  a  single word (i.e., it should not contain white space); see NOTES
       below.
  • 可以查看debianutils源代码这里

答案2

运行的文件run-parts有相当详细的记录。除了它们是可执行的之外,以下手册页片段还解释了这些要求:

If  neither  the --lsbsysinit option nor the --regex option is given then the
names must consist entirely of ASCII upper- and lower-case letters, ASCII digits,
ASCII underscores, and ASCII minus-hyphens.

If the --lsbsysinit option is given, then the names must not end in .dpkg-old  or
.dpkg-dist or .dpkg-new or .dpkg-tmp, and must belong to one or more  of  the 
following  namespaces:  the  LANANA-assigned  namespace  (^[a-z0-9]+$); the LSB
hierarchical and reserved namespaces (^_?([a-z0-9_.]+-)+[a-z0-9]+$); and the
Debian cron script namespace (^[a-zA-Z0-9_-]+$)

使用该--lsbsysinit选项对于运行脚本特别有用,/etc因为许多脚本将在各自的包中作为配置文件列出。创建具有扩展名的文件的常见情况dpkg-*是对已安装的版本进行更改并dpkg尝试安装新版本时。dpkg通常会将用户未选择的版本存储在同一目录中。使用run-parts是一种很好的标准方法,可以确保这些扩展或任何其他不应该运行的扩展都不会运行。它减少了出现错误的可能性,因为开发人员忘记在他们的脚本中包含其中之一。

即使没有--lsbsysinit,它仍然是一个有用的命令,可以减少需要编写的代码量并提高在整个系统中使用时的可靠性。虽然不完全是,因为用 afind ... -executable -exec {} ;或类似的东西替换更容易。

答案3

Why is run-parts needed while one could use find 
with -perm option or test utility?

这只是设计使然。正如您在手册页上看到的,run-parts还有许多选项。所有的东西都可以用 shell 编写。

[编辑]

它也可能是用 C 语言实现的,以防止黑客攻击。

答案4

直接进入问题标题(来自man run-parts):

run scripts or programs in a directory

    run-parts  runs  all  the  executable files named within constraints described
    below, found in directory directory.  Other files and directories are silently ignored.

当谈到cron作业时,它提供了一种按特定计划从目录执行一组脚本的简单方法。如果您要使用单个命令手动运行一组脚本,则同样适用:

ivanleoncz@ilex: /tmp $ ll scripts/
total 12
-rwxrw-r-- 1 ivanleoncz ivanleoncz 27 Jun 15 18:29 script_1*
-rwxrw-r-- 1 ivanleoncz ivanleoncz 27 Jun 15 18:29 script_2*
-rwxrw-r-- 1 ivanleoncz ivanleoncz 27 Jun 15 18:30 script_3*

ivanleoncz@ilex: /tmp $ cat scripts/script_1
#!/bin/bash
echo "I'm $0."

ivanleoncz@ilex: /tmp $ cat scripts/script_2
#!/bin/bash
echo "I'm $0."

ivanleoncz@ilex: /tmp $ cat scripts/script_3
#!/bin/bash
echo "I'm $0."

ivanleoncz@ilex: /tmp $ run-parts scripts
I'm scripts/script_1.
I'm scripts/script_2.
I'm scripts/script_3.

ivanleoncz@ilex: /tmp $ run-parts --report scripts
scripts/script_1:
I'm scripts/script_1.
scripts/script_2:
I'm scripts/script_2.
scripts/script_3:
I'm scripts/script_3.

脚本需要以下条件(一些在 shell 脚本中很常见):

  1. UGO(用户|组|其他)的执行权限,取决于用户适合的位置。
  2. Shebang 位于脚本顶部,告诉执行时将使用哪个 shell。
  3. 必须没有文件扩展名。这很奇怪,但是是的,这是必需的。

相关内容