/*      $NetBSD: revnetgroup.c,v 1.15 2011/08/30 21:10:29 joerg Exp $ */

/*
* Copyright (c) 1995
*      Bill Paul <[email protected]>.  All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
*    notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
*    notice, this list of conditions and the following disclaimer in the
*    documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
*    must display the following acknowledgement:
*      This product includes software developed by Bill Paul.
* 4. Neither the name of the author nor the names of any co-contributors
*    may be used to endorse or promote products derived from this software
*    without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY Bill Paul AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED.  IN NO EVENT SHALL Bill Paul OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
* reverse netgroup map generator program
*
* Written by Bill Paul <[email protected]>
* Center for Telecommunications Research
* Columbia University, New York City
*
*/

#include <sys/cdefs.h>
#ifndef lint
__RCSID("$NetBSD: revnetgroup.c,v 1.15 2011/08/30 21:10:29 joerg Exp $");
#endif

#include <ctype.h>
#include <err.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#include "hash.h"
#include "protos.h"

__dead static void      usage(void);



/* Default location of netgroup file. */
const char *netgroup = "/etc/netgroup";

/* Stored hash table version of 'forward' netgroup database. */
struct group_entry *gtable[TABLESIZE];

/*
* Stored hash table of 'reverse' netgroup member database
* which we will construct.
*/
struct member_entry *mtable[TABLESIZE];

static void
usage(void)
{

       fprintf (stderr,"usage: %s -u|-h [-f netgroup file]\n", getprogname());
       exit(1);
}

int
main(int argc, char *argv[])
{
       struct group_entry *gcur;
       struct member_entry *mcur;
       FILE    *fp;
       char    *line, *p, *host, *user, *domain;
       int      ch, i;
       size_t   len;
       char    *key;
       int      hosts = -1;

       if (argc < 2)
               usage();

       while ((ch = getopt(argc, argv, "uhf:")) != -1) {
               switch(ch) {
               case 'u':
                       if (hosts != -1) {
                               warnx("please use only one of -u or -h");
                               usage();
                       }
                       hosts = 0;
                       break;
               case 'h':
                       if (hosts != -1) {
                               warnx("please use only one of -u or -h");
                               usage();
                       }
                       hosts = 1;
                       break;
               case 'f':
                       netgroup = optarg;
                       break;
               default:
                       usage();
                       break;
               }
       }

       if (hosts == -1)
               usage();

       if (strcmp(netgroup, "-")) {
               if ((fp = fopen(netgroup, "r")) == NULL) {
                       err(1, "%s", netgroup);
               }
       } else {
               fp = stdin;
       }

       /* Stuff all the netgroup names and members into a hash table. */
       for (;
           (line = fparseln(fp, &len, NULL, NULL, FPARSELN_UNESCALL));
           free(line)) {
               if (len == 0)
                       continue;
               p = line;

               for (key = p; *p && isspace((unsigned char)*p) == 0; p++)
                       ;
               while (*p && isspace((unsigned char)*p))
                       *p++ = '\0';
               store(gtable, key, p);
       }

       fclose(fp);

       /*
        * Find all members of each netgroup and keep track of which
        * group they belong to.
        */
       for (i = 0; i < TABLESIZE; i++) {
               gcur = gtable[i];
               while (gcur) {
                       rng_setnetgrent(gcur->key);
                       while (rng_getnetgrent(&host, &user, &domain) != 0) {
                               if (hosts) {
                                       if (!(host && !strcmp(host,"-"))) {
                                               mstore(mtable,
                                                      host ? host : "*",
                                                      gcur->key,
                                                      domain ? domain : "*");
                                       }
                               } else {
                                       if (!(user && !strcmp(user,"-"))) {
                                               mstore(mtable,
                                                      user ? user : "*",
                                                      gcur->key,
                                                      domain ? domain : "*");
                                       }
                               }
                       }
                       gcur = gcur->next;
               }
       }

       /* Release resources used by the netgroup parser code. */
       rng_endnetgrent();

       /* Spew out the results. */
       for (i = 0; i < TABLESIZE; i++) {
               mcur = mtable[i];
               while (mcur) {
                       struct grouplist *tmp;
                       printf ("%s.%s\t", mcur->key, mcur->domain);
                       tmp = mcur->groups;
                       while (tmp) {
                               printf ("%s", tmp->groupname);
                               tmp = tmp->next;
                               if (tmp)
                                       printf(",");
                       }
                       mcur = mcur->next;
                       printf ("\n");
               }
       }

       /* Let the OS free all our resources. */
       exit(0);
}