add support for search domains to dns resolver
authorRich Felker <dalias@aerifal.cx>
Fri, 29 Jan 2016 00:50:48 +0000 (19:50 -0500)
committerRich Felker <dalias@aerifal.cx>
Fri, 29 Jan 2016 01:29:55 +0000 (20:29 -0500)
search is only performed if the search or domain keyword is used in
resolv.conf and the queried name has fewer than ndots dots. there is
no default domain and names with >=ndots dots are never subjected to
search; failure in the root scope is final.

the (non-POSIX) res_search API presently does not honor search. this
may be added at some point in the future if needed.

resolv.conf is now parsed twice, at two different layers of the code
involved. this will be fixed in a subsequent patch.

src/network/lookup_name.c

index fb7b5c12d25321109f031a6cf88be5f26ad0c4ad..09734b50ddd02236942c79e0d6c7b6addfee5b7c 100644 (file)
@@ -157,6 +157,46 @@ static int name_from_dns(struct address buf[static MAXADDRS], char canon[static
        return EAI_FAIL;
 }
 
+static int name_from_dns_search(struct address buf[static MAXADDRS], char canon[static 256], const char *name, int family)
+{
+       char search[256];
+       struct resolvconf conf;
+       size_t l, dots;
+       char *p, *z;
+
+       if (__get_resolv_conf(&conf, search, sizeof search) < 0) return -1;
+
+       /* Count dots, suppress search when >=ndots or name ends in
+        * a dot, which is an explicit request for global scope. */
+       for (dots=l=0; name[l]; l++) if (name[l]=='.') dots++;
+       if (dots >= conf.ndots || name[l-1]=='.') *search = 0;
+
+       /* This can never happen; the caller already checked length. */
+       if (l >= 256) return EAI_NONAME;
+
+       /* Name with search domain appended is setup in canon[]. This both
+        * provides the desired default canonical name (if the requested
+        * name is not a CNAME record) and serves as a buffer for passing
+        * the full requested name to name_from_dns. */
+       memcpy(canon, name, l);
+       canon[l] = '.';
+
+       for (p=search; *p; p=z) {
+               for (; isspace(*p); p++);
+               for (z=p; *z && !isspace(*z); z++);
+               if (z==p) break;
+               if (z-p < 256 - l - 1) {
+                       memcpy(canon+l+1, p, z-p);
+                       canon[z-p+1+l] = 0;
+                       int cnt = name_from_dns(buf, canon, canon, family);
+                       if (cnt) return cnt;
+               }
+       }
+
+       canon[l] = 0;
+       return name_from_dns(buf, canon, name, family);
+}
+
 static const struct policy {
        unsigned char addr[16];
        unsigned char len, mask;
@@ -257,7 +297,7 @@ int __lookup_name(struct address buf[static MAXADDRS], char canon[static 256], c
        if (!cnt) cnt = name_from_numeric(buf, name, family);
        if (!cnt && !(flags & AI_NUMERICHOST)) {
                cnt = name_from_hosts(buf, canon, name, family);
-               if (!cnt) cnt = name_from_dns(buf, canon, name, family);
+               if (!cnt) cnt = name_from_dns_search(buf, canon, name, family);
        }
        if (cnt<=0) return cnt ? cnt : EAI_NONAME;