c - 来自终端的 SIGHUP

标签 c linux process ssh signals

专家们, 我有一个通过 ssh 连接到服务器的客户端(它分配了一个 tty)。我有一个在服务器上运行的进程 A。现在,无论何时客户端断开连接,我都需要 A 知道消失的 tty。

我在想,既然 SSHD 知道 session 即将结束(在超时或简单退出之后),它可以生成一个信号来处理 A。

有没有其他方法可以让 A 获取有关消失的 tty 的信息,就像在 SIGHUP 上监听 tty 一样?我正在 Linux 上用 C 编写代码。

感谢您的帮助。

最佳答案

POSIX.1 提供了一个工具,utmpx ,其中列出了当前登录的用户、他们的终端和其他信息。在 Linux 中,这与 utmp 相同;见man 5 utmp了解更多信息。

OpenSSH 确实维护 utmp 记录。

这是一个简单的例子,它列出了当前从远程机器登录的所有用户、他们使用的终端以及用户拥有的初始进程组:

#define  _POSIX_C_SOURCE 200809L
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <utmpx.h>

int main(void)
{
    struct utmpx *entry;

    setutxent();
    while ((entry = getutxent()))
        if (entry->ut_type == USER_PROCESS && entry->ut_host[0] != '\0')
            printf("%s is logged in on /dev/%s from %s and owns process group %d\n", 
                   entry->ut_user, entry->ut_line, entry->ut_host,
                   (int)getpgid(entry->ut_pid));

    return 0;
}

在您的情况下,我希望进程 A 维护远程连接用户的列表,并定期执行与上述类似的循环以更新已知条目的状态并添加新条目;并删除不再可见的条目。

然后新条目匹配“登录”事件,不再看到“注销”事件的条目(并在循环后删除),所有其他事件都是“仍然登录”的用户。

就使用的 CPU 时间和使用的 I/O 而言,上面的循环非常轻量级。 utmp 记录(在大多数 Linux 机器中为 /var/run/utmp)是二进制形式,如果经常访问,通常在页面缓存中。条目相对较小,即使在有很多用户的服务器上,读取的文件大小也远低于 1 兆字节。不过,我不会在一个紧密的循环中这样做。

就我个人而言,我会使用 inotify等待 UTMPX_FILE 文件(在大多数 Linux 机器上为 /var/run/utmp)上的 CLOSE_WRITE 事件,并在每个事件后重新读取记录.这样,服务将在大部分时间阻塞在 inotify 文件描述符的 read() 上(不会浪费任何 CPU 时间),并且几乎立即对任何登录/注销事件使用react。

关于c - 来自终端的 SIGHUP,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21033297/

相关文章:

rust - 如何在 Rust 中 fork/exec 转发所有参数的进程?

c - printf 返回 c 但出现错误是什么

python - 使用 swig 进行 python 列表输入和输出

linux - 如何检测哪个用户已从一台服务器登录到另一台服务器(仅限 root 访问)

linux - 将文件从一个位置复制到另一位置

ruby - 进程的 pid、ppid、uid、euid、gid 和 egid 之间有什么区别?

c - MPI:程序工作取决于进程数

c - 如何从内核内置模块发送信号/中断到可加载内核模块?

java - 默认: classes in this makefile?是什么意思

这段 C 代码可以创建僵尸进程吗?