AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 541417
Accepted
Anthony Geoghegan
Anthony Geoghegan
Asked: 2019-09-13 06:16:11 +0800 CST2019-09-13 06:16:11 +0800 CST 2019-09-13 06:16:11 +0800 CST

获取与用户 ID 关联的用户名的 POSIX 兼容方式

  • 772

我经常想要获取与用户 ID 相关联的登录名,并且因为它已被证明是一个常见的用例,所以我决定编写一个 shell 函数来执行此操作。虽然我主要使用 GNU/Linux 发行版,但我尝试将我的脚本编写为尽可能可移植,并检查我所做的是否与 POSIX 兼容。

解析/etc/passwd

我尝试的第一种方法是解析/etc/passwd(使用awk)。

awk -v uid="$uid" -F: '$3 == uid {print $1}' /etc/passwd

但是,这种方法的问题是登录可能不是本地的,例如,用户身份验证可能通过 NIS 或 LDAP。

使用getent命令

使用getent passwd比解析更便携,/etc/passwd因为这也查询非本地 NIS 或 LDAP 数据库。

  • getent - 维基百科
  • getent(1) - Linux 手册页
getent passwd "$uid" | cut -d: -f1

不幸的是,getentPOSIX 似乎没有指定该实用程序。

使用id命令

id是用于获取有关用户身份的数据的 POSIX 标准化实用程序。

BSD 和 GNU 实现接受用户 ID 作为操作数:

  • id(1) - OpenBSD 手册页
  • id(1) – FreeBSD 手册页
  • GNU Coreutils:id 调用

这意味着它可用于打印与用户 ID 关联的登录名:

id -nu "$uid"

但是,在 POSIX 中没有指定提供用户 ID 作为操作数;它只描述了使用登录名作为操作数。

结合以上所有

我考虑将上述三种方法组合成如下内容:

get_username(){
    uid="$1"
    # First try using getent
    getent passwd "$uid" | cut -d: -f1 ||
        # Next try using the UID as an operand to id.
        id -nu "$uid" ||
        # As a last resort, parse `/etc/passwd`.
        awk -v uid="$uid" -F: '$3 == uid {print $1}' /etc/passwd
}

然而,这很笨重、不优雅,而且——更重要的是——不够健壮;如果用户 ID 无效或不存在,它将以非零状态退出。在我编写一个更长更笨重的 shell 脚本来分析和存储每个命令调用的退出状态之前,我想我会在这里问:

是否有一种更优雅、更便携(POSIX 兼容)的方式来获取与用户 ID 关联的登录名?

users
  • 4 4 个回答
  • 4984 Views

4 个回答

  • Voted
  1. Best Answer
    cas
    2019-09-13T06:34:10+08:002019-09-13T06:34:10+08:00

    一种常见的方法是测试您想要的程序是否存在并且是否可以从您的PATH. 例如:

    get_username(){
      uid="$1"
    
      # First try using getent
      if command -v getent > /dev/null 2>&1; then 
        getent passwd "$uid" | cut -d: -f1
    
      # Next try using the UID as an operand to id.
      elif command -v id > /dev/null 2>&1 && \
           id -nu "$uid" > /dev/null 2>&1; then
        id -nu "$uid"
    
      # Next try perl - perl's getpwuid just calls the system's C library getpwuid
      elif command -v perl >/dev/null 2>&1; then
        perl -e '@u=getpwuid($ARGV[0]);
                 if ($u[0]) {print $u[0]} else {exit 2}' "$uid"
    
      # As a last resort, parse `/etc/passwd`.
      else
          awk -v uid="$uid" -F: '
             BEGIN {ec=2};
             $3 == uid {print $1; ec=0; exit 0};
             END {exit ec}' /etc/passwd
      fi
    }
    

    因为 POSIXid不支持 UID 参数,所以 eliffor 子句id不仅要测试是否id在 PATH 中,还要测试它是否会无错误地运行。这意味着它可能会运行id两次,幸运的是这不会对性能产生明显影响。也有可能同时运行id和awk,但对性能的影响同样可以忽略不计。

    顺便说一句,使用这种方法,无需存储输出。只有其中一个会运行,因此只有一个会打印输出以供函数返回。

    • 15
  2. Gilles 'SO- stop being evil'
    2019-09-13T13:01:06+08:002019-09-13T13:01:06+08:00

    POSIX 中除了id. 尝试id并回退到解析/etc/passwd可能与实践中一样可移植。

    BusyBoxid不接受用户 ID,但带有 BusyBox 的系统通常是自治的嵌入式系统,解析/etc/passwd就足够了。

    如果您遇到不接受用户 ID 的非 GNU 系统id,您也可以尝试getpwuid通过 Perl 调用,如果它可用的话:

    username=$(perl -e 'print((getpwuid($ARGV[0]))[0])) 2>/dev/null
    if [ -n "$username" ]; then echo "$username"; return; fi
    

    或 Python:

    if python -c 'import pwd, sys; print(pwd.getpwuid(int(sys.argv[1]))).pw_name' 2>/dev/null; then return; fi
    
    • 7
  3. Anthony Geoghegan
    2019-09-14T04:57:47+08:002019-09-14T04:57:47+08:00

    POSIX 指定getpwuid作为标准 C 函数在用户数据库中搜索用户 ID,从而允许将 ID 转换为登录名。我下载了 GNU coreutils的源代码,并且可以看到在他们的实用程序实现中使用了这个函数,例如id和ls.

    作为一个学习练习,我编写了这个简单粗暴的 C 程序来简单地充当这个函数的包装器。请记住,自大学以来(多年前)我就没有用 C 编程过,我不打算在生产中使用它,但我想我会把它贴在这里作为概念证明(如果有人想编辑它,随意):

    #include <stdio.h>
    #include <stdlib.h>  /* atoi */
    #include <pwd.h>
    
    int main( int argc, char *argv[] ) {
        uid_t uid;
        if ( argc >= 2 ) {
            /* NB: atoi returns 0 (super-user ID) if argument is not a number) */
            uid = atoi(argv[1]);
        }
        /* Ignore any other arguments after the first one. */
        else {
            fprintf(stderr, "One numeric argument must be supplied.\n");
            return 1;
        }
    
        struct passwd *pwd;
        pwd = getpwuid(uid);
        if (pwd) {
            printf("The login name for %d is: %s\n", uid, pwd->pw_name);
            return 0;
        }
        else {
            fprintf(stderr, "Invalid user ID: %d\n", uid);
            return 1;
        }
    }
    

    我没有机会使用 NIS/LDAP 对其进行测试,但我注意到如果同一用户中有多个条目/etc/passwd,它会忽略除第一个之外的所有条目。

    示例用法:

    $ ./get_user ""
    The login name for 0 is: root
    
    $ ./get_user 99
    Invalid user ID: 99
    
    • 7
  4. R.. GitHub STOP HELPING ICE
    2019-09-13T15:43:48+08:002019-09-13T15:43:48+08:00

    一般来说,我建议不要这样做。从用户名到 uid 的映射不是一对一的,并且您可以从 uid 转换回来以获取用户名的编码假设会破坏事情。例如,我经常通过使容器中的passwd和group文件将所有用户和组名映射到 id 0 来运行完全无 root 的用户命名空间容器;这允许安装软件包工作而不会chown失败。但是,如果某些东西试图将 0 转换回 uid 并且没有得到它所期望的,它将无缘无故地中断。因此,在本例中,您应该转换为 uid 并在该空间中进行比较,而不是转换回和比较用户名。

    如果您确实需要执行此操作,如果您是 root,则可以通过制作一个临时文件,chown将其写入 uid,然后使用ls读取并解析所有者的名称来进行半便携操作。但是我只会使用一种众所周知的方法,这种方法不是标准化的,而是“在实践中可移植的”,就像您已经找到的一种方法一样。

    但同样,不要这样做。有时很难做的事情是向您发送消息。

    • 3

相关问题

  • 用户`ldl`存在于root组中,但是当我使用`groups root`时检查它,不显示`ldl`用户

  • 如何在 Samba Active Directory 中导出完整的用户列表?

  • 我可以在不删除主目录的情况下对用户进行 userdel 吗?

  • 删除用户时会删除用户的主要组吗?

  • 即使我在文件上设置了全局读取权限,也得到“权限被拒绝”

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve