0

问题: xinetd(扩展 Internet 服务守护程序)将输入映射到服务的标准输入,并将输出映射到服务的标准输出。

换句话说,假设一个用 C 编写的自定义 TCP 服务,进入端口X的数据被映射到服务的标准输入,而服务的标准输出被映射到来自端口Y的数据。

同样,假设一个用 C 编写的自定义 TCP 服务,该服务有没有办法确定连接客户端的 IP 地址?

Web Research: 截至发布此问题时,Stack Exchange(或其他地方)没有其他问题专门处理用 C 编写的 xinetd TCP 服务,试图确定连接客户端的 IP 地址。

有类似的问题:

但没有人回答这篇文章中详述的具体问题。

后续注意事项: xinetd 获取与 TCP 端口X关联的套接字描述符并将其映射到服务的标准输入。

认识到这一事实可以进行更好的网络搜索,从而得到以下答案:

4

1 回答 1

4

是的,getpeername(2)在其 stdin (0) 或 stdout (1) 文件描述符上。

示例:当从xinetdor运行时inetd,这将打印连接到其标准输入的客户端的地址:

#define _DEFAULT_SOURCE
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/in.h>
#include <stdio.h>
#include <err.h>
#include <netdb.h>

int main(void){
        union {
                struct sockaddr a;
                struct sockaddr_in in;
                struct sockaddr_in6 in6;
        } na;
        char host[NI_MAXHOST], port[NI_MAXSERV]; int e;
        socklen_t nl = sizeof na;
        if(getpeername(0, &na.a, &nl)) err(1, "getpeername");
        if(e = getnameinfo(&na.a, nl, host, sizeof host, port, sizeof port,
                        NI_NUMERICHOST|NI_NUMERICSERV))
                errx(1, "getnameinfo: %s", gai_strerror(e));
        switch(na.a.sa_family){
        case AF_INET:
                errx(0, "connection from %s:%s", host, port);
        case AF_INET6:
                errx(0, "connection from [%s]:%s", host, port);
        default:
                errx(0, "connection from unknown address family %d",
                        na.a.sa_family);
        }
}
于 2019-04-19T20:44:23.553 回答