专家们,
我有一个通过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 等待CLOSE_WRITE文件(大多数Linux机器上为UTMPX_FILE)上的/var/run/utmp事件,并在每个事件之后重新读取记录。这样,该服务将在大多数时间阻止inotify文件描述符上的read()(不浪费任何CPU时间),并且几乎立即对任何登录/注销事件使用react。

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

10-10 00:17