X-Git-Url: https://git.librecmc.org/?a=blobdiff_plain;f=src%2Fgns%2Fnss%2Fnss_gns_query.c;h=380788d35d0cea3a761cbd87db8781edf050e26d;hb=557f5487b3e82416ff315989528e2ba6714cc650;hp=b8f6c164de7617eb0ff53ba003c208b1e4d2281c;hpb=7e3313ce2b4797bde341340f4402ddc14cb63138;p=oweals%2Fgnunet.git diff --git a/src/gns/nss/nss_gns_query.c b/src/gns/nss/nss_gns_query.c index b8f6c164d..380788d35 100644 --- a/src/gns/nss/nss_gns_query.c +++ b/src/gns/nss/nss_gns_query.c @@ -1,27 +1,46 @@ /* This file is part of GNUnet. - (C) 2012 Christian Grothoff (and other contributing authors) + Copyright (C) 2012 GNUnet e.V. - GNUnet is free software; you can redistribute it and/or modify - it under the terms of the GNU General Public License as published - by the Free Software Foundation; either version 3, or (at your - option) any later version. + GNUnet is free software: you can redistribute it and/or modify it + under the terms of the GNU Affero General Public License as published + by the Free Software Foundation, either version 3 of the License, + or (at your option) any later version. GNUnet 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 - General Public License for more details. + Affero General Public License for more details. - You should have received a copy of the GNU General Public License - along with GNUnet; see the file COPYING. If not, write to the - Free Software Foundation, Inc., 59 Temple Place - Suite 330, - Boston, MA 02111-1307, USA. + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . + + SPDX-License-Identifier: AGPL3.0-or-later */ #include #include #include #include "nss_gns_query.h" #include +#include +#include +#include +#include +#include +#include +#include + + +static void +kwait (pid_t chld) +{ + int ret; + + kill (chld, SIGKILL); + waitpid (chld, + &ret, + 0); +} /** @@ -31,72 +50,113 @@ * @param af address family * @param name the name to resolve * @param u the userdata (result struct) - * @return -1 on error else 0 + * @return -1 on internal error, + * -2 if request is not for GNS, + * -3 on timeout, + * else 0 */ -int gns_resolve_name(int af, const char *name, struct userdata *u) +int +gns_resolve_name (int af, + const char *name, + struct userdata *u) { FILE *p; - char *cmd; char line[128]; + int ret; + int out[2]; + pid_t pid; - if (af == AF_INET6) + if (0 != pipe (out)) + return -1; + pid = fork (); + if (-1 == pid) + return -1; + if (0 == pid) { - if (-1 == asprintf(&cmd, "%s -t AAAA -u %s\n", "gnunet-gns -r", name)) - return -1; + char *argv[] = { + "gnunet-gns", + "-r", + "-t", + (AF_INET6 == af) ? "AAAA" : "A", + "-u", + (char *) name, + NULL + }; + + (void) close (STDOUT_FILENO); + if ( (0 != close (out[0])) || + (STDOUT_FILENO != dup2 (out[1], STDOUT_FILENO)) ) + _exit (1); + (void) execvp ("gnunet-gns", + argv); + _exit (1); } - else - { - if (-1 == asprintf(&cmd, "%s %s\n", "gnunet-gns -r -u", name)) + (void) close (out[1]); + p = fdopen (out[0], "r"); + if (NULL == p) + { + kwait (pid); return -1; - } - - p = popen(cmd,"r"); - - if (p != NULL ) + } + while (NULL != fgets (line, + sizeof (line), + p)) { - while (fgets( line, sizeof(line), p ) != NULL) + if (u->count >= MAX_ENTRIES) + break; + if (line[strlen(line)-1] == '\n') { - - if (u->count >= MAX_ENTRIES) - break; - - if (line[strlen(line)-1] == '\n') + line[strlen(line)-1] = '\0'; + if (AF_INET == af) { - line[strlen(line)-1] = '\0'; - if (af == AF_INET) + if (inet_pton(af, + line, + &u->data.ipv4[u->count])) { - if (inet_pton(af, line, &(u->data.ipv4[u->count]))) - { - u->count++; - u->data_len += sizeof(ipv4_address_t); - } - else - { - fclose (p); - free (cmd); - return -1; - } - } - else if ((af == AF_INET6)) + u->count++; + u->data_len += sizeof(ipv4_address_t); + } + else + { + (void) fclose (p); + kwait (pid); + errno = EINVAL; + return -1; + } + } + else if (AF_INET6 == af) + { + if (inet_pton(af, + line, + &u->data.ipv6[u->count])) + { + u->count++; + u->data_len += sizeof(ipv6_address_t); + } + else { - if (inet_pton(af, line, &(u->data.ipv6[u->count]))) - { - u->count++; - u->data_len += sizeof(ipv6_address_t); - } - else - { - fclose (p); - free (cmd); - return -1; - } - } + (void) fclose (p); + kwait (pid); + errno = EINVAL; + return -1; + } } } - fclose (p); } - free (cmd); - + (void) fclose (p); + waitpid (pid, + &ret, + 0); + if (! WIFEXITED (ret)) + return -1; + if (4 == WEXITSTATUS (ret)) + return -2; /* not for GNS */ + if (3 == ret) + return -3; /* timeout -> not found */ + if ( (2 == WEXITSTATUS (ret)) || + (1 == WEXITSTATUS (ret)) ) + return -2; /* launch failure -> service unavailable */ return 0; - } + +/* end of nss_gns_query.c */