曲径通幽论坛

 找回密码
 立即注册
搜索
查看: 4322|回复: 0
打印 上一主题 下一主题

[系统应用] resolveip 代码解读

[复制链接]

4918

主题

5880

帖子

3万

积分

GROAD

曲径通幽,安觅芳踪。

Rank: 6Rank: 6

积分
34395
跳转到指定楼层
楼主
发表于 2015-3-30 17:43:56 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
在看到一篇帖子里,有谈到要 opkg install resolveip 时,想要看看 resolveip 是做什么的。于是在 https://dev.openwrt.org/browser/ ... ev=27666&order=name 里看到其源码,贴在下边:
[C] 纯文本查看 复制代码
/*
 * Based on code found at [url=https://dev.openwrt.org/ticket/4876]https://dev.openwrt.org/ticket/4876[/url] .
 * Extended by Jo-Philipp Wich <[url=mailto:jow@openwrt.org]jow@openwrt.org[/url]> for use in OpenWrt.
 *
 * You may use this program under the terms of the GPLv2 license.
 */

#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>


static void abort_query(int sig)
{
        exit(1);
}

static void show_usage(void)
{
        printf("Usage:\n");
        printf("        resolveip -h\n");
        printf("        resolveip [-t timeout] hostname\n");
        printf("        resolveip -4 [-t timeout] hostname\n");
        printf("        resolveip -6 [-t timeout] hostname\n");
        exit(255);
}

int main(int argc, char **argv)
{
        int timeout = 3;
        char opt;
    char ipaddr[INET6_ADDRSTRLEN];
    void *addr;
    struct addrinfo *res, *rp;
        struct sigaction sa = {        .sa_handler = &abort_query };
    struct addrinfo hints = {
            .ai_family   = AF_UNSPEC,
            .ai_socktype = SOCK_STREAM,
            .ai_protocol = IPPROTO_TCP,
            .ai_flags    = 0
    };

        while ((opt = getopt(argc, argv, "46t:h")) > -1)
        {
                switch (opt)
                {
                        case '4':
                                hints.ai_family = AF_INET;
                                break;

                        case '6':
                                hints.ai_family = AF_INET6;
                                break;

                        case 't':
                                timeout = atoi(optarg);
                                if (timeout <= 0)
                                        show_usage();
                                break;

                        case 'h':
                                show_usage();
                                break;
                }
        }

        if (!argv[optind])
                show_usage();

        sigaction(SIGALRM, &sa, NULL);
        alarm(timeout);

        if (getaddrinfo(argv[optind], NULL, &hints, &res))
                exit(2);

    for (rp = res; rp != NULL; rp = rp->ai_next)
    {
                addr = (rp->ai_family == AF_INET)
                        ? (void *)&((struct sockaddr_in *)rp->ai_addr)->sin_addr
                        : (void *)&((struct sockaddr_in6 *)rp->ai_addr)->sin6_addr
                ;

                if (!inet_ntop(rp->ai_family, addr, ipaddr, INET6_ADDRSTRLEN - 1))
                        exit(3);

                printf("%s\n", ipaddr);
        }

        freeaddrinfo(res);
        exit(0);
}

从代码中可以看到,程序有 4 个选项参数可用,-h 查看帮助信息;-t 指定查询的超时时间(可选项,不指定则默认为 3s);-4 要求解析 hostname 的 IPv4 地址;-6 要求解析 hostname 的 IPv6 地址。


while 循环用来解析命令行参数。

在 switch 中,case 't' 时,atoi(optarg) 中的 optarg 代表的是 t 后面的时间参数,这是自动获得的;因为在用 getopt() 处理时,第 3 个参数中 t 的后面有个冒号,因此要求必须接个参数。

sigaction() 是安装一个信号处理函数,这里指定 SIGALRM 信号。也就是说,当指定时间超时时(查询不到结果,然后由 alarm() 函数将信号发送),就会触发 abort_query() 函数做一个简单的退出。

getaddrinfo() 函数正是用来获得解析结果的,其中第 3 个参数 &hints ,它在所指向的结构中填入期望返回的信息类型的暗示,比如这里是希望返回 TCP 信息。


在使用 getaddrinfo() 返回的是一个地址结构链表,这是由于一个域名可能会绑定多个 IP,因此有多种地址信息,每一个信息都存储在一个 addrinfo 的结构中,所以在程序底下使用一个 for 来遍历整个链表,并将每个链表的中的 IP 地址信息打印出来。


最后使用 freeaddrinfo() 来释放相应的链表空间。


由上可见,resolveip 这个程序是一个很小的 DNS 查询工具,主要用在脚本里 。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|曲径通幽 ( 琼ICP备11001422号-1|公安备案:46900502000207 )

GMT+8, 2025-5-3 12:52 , Processed in 0.077285 second(s), 23 queries .

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表