123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276 |
- /* Enumerate /etc/hosts with a long line (bug 18991).
- Copyright (C) 2018-2019 Free Software Foundation, Inc.
- This file is part of the GNU C Library.
- The GNU C Library is free software; you can redistribute it and/or
- modify it under the terms of the GNU Lesser General Public
- License as published by the Free Software Foundation; either
- version 2.1 of the License, or (at your option) any later version.
- The GNU C Library is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- Lesser General Public License for more details.
- You should have received a copy of the GNU Lesser General Public
- License along with the GNU C Library; if not, see
- <http://www.gnu.org/licenses/>. */
- #include <dlfcn.h>
- #include <errno.h>
- #include <gnu/lib-names.h>
- #include <netdb.h>
- #include <nss.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <support/check.h>
- #include <support/check_nss.h>
- #include <support/namespace.h>
- #include <support/support.h>
- #include <support/test-driver.h>
- #include <support/xmemstream.h>
- #include <support/xstdio.h>
- #include <support/xunistd.h>
- struct support_chroot *chroot_env;
- /* Number of alias names in the long line. This is varied to catch
- different cases where the ERANGE handling can go wrong (line buffer
- length, alias buffer). */
- static int name_count;
- /* Write /etc/hosts, from outside of the chroot. */
- static void
- write_hosts (void)
- {
- FILE *fp = xfopen (chroot_env->path_hosts, "w");
- fputs ("127.0.0.1 localhost localhost.localdomain\n", fp);
- fputs ("192.0.2.2 host2.example.com\n", fp);
- fputs ("192.0.2.1", fp);
- for (int i = 0; i < name_count; ++i)
- fprintf (fp, " host%d.example.com", i);
- fputs ("\n192.0.2.80 www.example.com\n"
- "192.0.2.5 host5.example.com\n"
- "192.0.2.81 www1.example.com\n", fp);
- xfclose (fp);
- }
- const char *host1_expected =
- "name: localhost\n"
- "alias: localhost.localdomain\n"
- "address: 127.0.0.1\n";
- const char *host2_expected =
- "name: host2.example.com\n"
- "address: 192.0.2.2\n";
- const char *host4_expected =
- "name: www.example.com\n"
- "address: 192.0.2.80\n";
- const char *host5_expected =
- "name: host5.example.com\n"
- "address: 192.0.2.5\n";
- const char *host6_expected =
- "name: www1.example.com\n"
- "address: 192.0.2.81\n";
- static void
- prepare (int argc, char **argv)
- {
- chroot_env = support_chroot_create
- ((struct support_chroot_configuration)
- {
- .resolv_conf = "",
- .hosts = "", /* Filled in by write_hosts. */
- .host_conf = "multi on\n",
- });
- }
- /* If -1, no sethostent call. Otherwise, pass do_stayopen as the
- sethostent argument. */
- static int do_stayopen;
- /* If non-zero, perform an endostent call. */
- static int do_endent;
- static void
- subprocess_getent (void *closure)
- {
- xchroot (chroot_env->path_chroot);
- errno = 0;
- if (do_stayopen >= 0)
- sethostent (do_stayopen);
- TEST_VERIFY (errno == 0);
- int i = 0;
- while (true)
- {
- struct xmemstream expected;
- xopen_memstream (&expected);
- switch (++i)
- {
- case 1:
- fputs (host1_expected, expected.out);
- break;
- case 2:
- fputs (host2_expected, expected.out);
- break;
- case 3:
- fputs ("name: host0.example.com\n", expected.out);
- for (int j = 1; j < name_count; ++j)
- fprintf (expected.out, "alias: host%d.example.com\n", j);
- fputs ("address: 192.0.2.1\n", expected.out);
- break;
- case 4:
- fputs (host4_expected, expected.out);
- break;
- case 5:
- fputs (host5_expected, expected.out);
- break;
- case 6:
- fputs (host6_expected, expected.out);
- break;
- default:
- fprintf (expected.out, "*** unexpected host %d ***\n", i);
- break;
- }
- xfclose_memstream (&expected);
- char *context = xasprintf ("do_stayopen=%d host=%d", do_stayopen, i);
- errno = 0;
- struct hostent *e = gethostent ();
- if (e == NULL)
- {
- TEST_VERIFY (errno == 0);
- break;
- }
- check_hostent (context, e, expected.buffer);
- free (context);
- free (expected.buffer);
- }
- errno = 0;
- if (do_endent)
- endhostent ();
- TEST_VERIFY (errno == 0);
- /* Exercise process termination. */
- exit (0);
- }
- /* getaddrinfo test. To be run from a subprocess. */
- static void
- test_gai (int family)
- {
- struct addrinfo hints =
- {
- .ai_family = family,
- .ai_protocol = IPPROTO_TCP,
- .ai_socktype = SOCK_STREAM,
- };
- struct addrinfo *ai;
- int ret = getaddrinfo ("host2.example.com", "80", &hints, &ai);
- check_addrinfo ("host2.example.com", ai, ret,
- "address: STREAM/TCP 192.0.2.2 80\n"
- "address: STREAM/TCP 192.0.2.1 80\n");
- ret = getaddrinfo ("host5.example.com", "80", &hints, &ai);
- check_addrinfo ("host5.example.com", ai, ret,
- "address: STREAM/TCP 192.0.2.1 80\n"
- "address: STREAM/TCP 192.0.2.5 80\n");
- ret = getaddrinfo ("www.example.com", "80", &hints, &ai);
- check_addrinfo ("www.example.com", ai, ret,
- "address: STREAM/TCP 192.0.2.80 80\n");
- ret = getaddrinfo ("www1.example.com", "80", &hints, &ai);
- check_addrinfo ("www1.example.com", ai, ret,
- "address: STREAM/TCP 192.0.2.81 80\n");
- }
- /* Subprocess routine for gethostbyname/getaddrinfo testing. */
- static void
- subprocess_gethost (void *closure)
- {
- xchroot (chroot_env->path_chroot);
- /* This tests enlarging the read buffer in the multi case. */
- struct xmemstream expected;
- xopen_memstream (&expected);
- fputs ("name: host2.example.com\n", expected.out);
- for (int j = 1; j < name_count; ++j)
- /* NB: host2 is duplicated in the alias list. */
- fprintf (expected.out, "alias: host%d.example.com\n", j);
- fputs ("alias: host0.example.com\n"
- "address: 192.0.2.2\n"
- "address: 192.0.2.1\n",
- expected.out);
- xfclose_memstream (&expected);
- check_hostent ("host2.example.com",
- gethostbyname ("host2.example.com"),
- expected.buffer);
- free (expected.buffer);
- /* Similarly, but with a different order in the /etc/hosts file. */
- xopen_memstream (&expected);
- fputs ("name: host0.example.com\n", expected.out);
- for (int j = 1; j < name_count; ++j)
- fprintf (expected.out, "alias: host%d.example.com\n", j);
- /* NB: host5 is duplicated in the alias list. */
- fputs ("alias: host5.example.com\n"
- "address: 192.0.2.1\n"
- "address: 192.0.2.5\n",
- expected.out);
- xfclose_memstream (&expected);
- check_hostent ("host5.example.com",
- gethostbyname ("host5.example.com"),
- expected.buffer);
- free (expected.buffer);
- check_hostent ("www.example.com",
- gethostbyname ("www.example.com"),
- host4_expected);
- check_hostent ("www1.example.com",
- gethostbyname ("www1.example.com"),
- host6_expected);
- test_gai (AF_INET);
- test_gai (AF_UNSPEC);
- }
- static int
- do_test (void)
- {
- support_become_root ();
- if (!support_can_chroot ())
- return EXIT_UNSUPPORTED;
- __nss_configure_lookup ("hosts", "files");
- if (dlopen (LIBNSS_FILES_SO, RTLD_LAZY) == NULL)
- FAIL_EXIT1 ("could not load " LIBNSS_DNS_SO ": %s", dlerror ());
- /* Each name takes about 20 bytes, so this covers a wide range of
- buffer sizes, from less than 1000 bytes to about 18000 bytes. */
- for (name_count = 40; name_count <= 850; ++name_count)
- {
- write_hosts ();
- for (do_stayopen = -1; do_stayopen < 2; ++do_stayopen)
- for (do_endent = 0; do_endent < 2; ++do_endent)
- {
- if (test_verbose > 0)
- printf ("info: name_count=%d do_stayopen=%d do_endent=%d\n",
- name_count, do_stayopen, do_endent);
- support_isolate_in_subprocess (subprocess_getent, NULL);
- }
- support_isolate_in_subprocess (subprocess_gethost, NULL);
- }
- support_chroot_free (chroot_env);
- return 0;
- }
- #define PREPARE prepare
- #include <support/test-driver.c>
|