OpenSolaris_b135/cmd/ypcmd/revnetgroup/revnetgroup.c

Compare this file to the similar file:
Show the results in this format:

/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License, Version 1.0 only
 * (the "License").  You may not use this file except in compliance
 * with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright (c) 1994, by Sun Microsystems, Inc.
 * All rights reserved.
 */
							    
/*
 * For SUNWnskit - version 1.1
 *
 * Based on:
 *	#pragma ident	"%Z%%M%	%I%	%E% SMI"	(SMI4.1 1.6)
 */

#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <ctype.h>
#include <pwd.h>
#include <rpcsvc/ypclnt.h>
#include "util.h"
#include "table.h"
#include "getgroup.h"

#define	MAXDOMAINLEN 256
#define	MAXGROUPLEN 1024

/*
 * Reverse the netgroup file. A flag of "-u" means reverse by username,
 * one of "-h" means reverse by hostname. Each line in the output file
 * will begin with a key formed by concatenating the host or user name
 * with the domain name. The key will be followed by a tab, then the
 * comma-separated, newline-terminated list of groups to which the
 * user or host belongs.
 *
 * Exception: Groups to which everyone belongs (universal groups) will
 * not be included in the list.  The universal groups will be listed under
 * the special name "*".
 *
 * Thus to find out all the groups that user "foo" of domain "bar" is in,
 * lookup the groups under  foo.bar, foo.*, *.bar and *.*.
 *
 */



/* Stores a list of strings */
typedef struct stringnode *stringlist;
struct stringnode {
    char *str;
    stringlist next;
};
typedef struct stringnode stringnode;



/* Stores a list of (name,list-of-groups) */
typedef struct groupentrynode *groupentrylist;
struct groupentrynode {
    char *name;
    stringlist groups;
    groupentrylist next;
};
typedef struct groupentrynode groupentrynode;

stringtable ngtable;

static groupentrylist grouptable[TABLESIZE];

static char *nextgroup(void);
static void storegroup(char *group, struct grouplist *glist, int byuser);
static void enter(char *name, char *group);
static void appendgroup(groupentrylist grlist, char *group);
static groupentrylist newentry(char *name, char *group);
static void loadtable(FILE *nf);
static void dumptable(void);

int
main(argc, argv)
    int argc;
    char *argv[];
{
	char *group;
	struct grouplist *glist;
	int byuser;

	loadtable(stdin);
	if (argc == 2 && argv[1][0] == '-' &&
			(argv[1][1] == 'u' || argv[1][1] == 'h')) {
		byuser = (argv[1][1] == 'u');
	} else {
		(void) fprintf(stderr,
				"usage: %s -h (by host), %s -u (by user)\n",
				argv[0], argv[0]);
		exit(1);
	}

	while (group = nextgroup()) {
		glist = my_getgroup(group);
		storegroup(group, glist, byuser);
	}
	dumptable();

	return (0);
}

/*
 *	Get the next netgroup from /etc/netgroup
 */
static char *
nextgroup(void)
{
	static int index = -1;
	static tablelist cur = NULL;
	char *group;

	while (cur == NULL) {
		if (++index == TABLESIZE) {
			return (NULL);
		}
		cur = ngtable[index];
	}
	group = cur->key;
	cur = cur->next;
	return (group);
}



/*
 * Dump out all of the stored info into a file
 */
static void
dumptable(void)
{
	int i;
	groupentrylist entry;
	stringlist groups;

	for (i = 0; i < TABLESIZE; i++) {
		if (entry = grouptable[i]) {
			while (entry) {
				fputs(entry->name, stdout);
				putc('\t', stdout);
				for (groups = entry->groups; groups;
						groups = groups->next) {
					fputs(groups->str, stdout);
					if (groups->next) {
						putc(',', stdout);
					}
				}
				putc('\n', stdout);
				entry = entry->next;
			}
		}
	}
}




/*
 *	Add a netgroup to a user's list of netgroups
 */
static void
storegroup(char *group, struct grouplist *glist, int byuser)
{
	char *name;	/* username or hostname */
	char *domain;
	char *key;
	static char *universal = "*";

	for (; glist; glist = glist->gl_nxt) {
		name = byuser ? glist->gl_name : glist->gl_machine;
		if (!name) {
		    name = universal;
		} else if (!isalnum(*name) && *name != '_') {
		    continue;
		}
		domain = glist->gl_domain;
		if (!domain) {
		    domain = universal;
		}
		key = malloc((unsigned) (strlen(name)+strlen(domain)+2));
		(void) sprintf(key, "%s.%s", name, domain);
		enter(key, group);
	}
}



static groupentrylist
newentry(char *name, char *group)
{
	groupentrylist new;

	new = MALLOC(groupentrynode);

	STRCPY(new->name, name);

	new->groups = MALLOC(stringnode);
	new->groups->str = group;
	new->groups->next = NULL;

	new->next = NULL;
	return (new);
}

static void
appendgroup(groupentrylist grlist, char *group)
{
	stringlist cur, prev;

	for (cur = grlist->groups; cur; prev = cur, cur = cur->next) {
		if (strcmp(group, cur->str) == 0) {
		    return;
		}
	}
	prev->next = MALLOC(stringnode);
	cur = prev->next;
	cur->str = group;
	cur->next = NULL;
}

static void
enter(char *name, char *group)
{
	int key;
	groupentrylist gel;
	groupentrylist gelprev;

	key = tablekey(name);
	if (grouptable[key] == NULL) {
		grouptable[key] = newentry(name, group);
	} else {
		gel = grouptable[key];
		while (gel && strcmp(gel->name, name)) {
		    gelprev = gel;
		    gel = gel->next;
		}
		if (gel) {
		    appendgroup(gel, group);
		} else {
		    gelprev->next = newentry(name, group);
		}
	}
}

/*
 * Load up a hash table with the info in /etc/netgroup
 */
static void
loadtable(FILE *nf)
{
	char buf[MAXGROUPLEN];
	char *p;
	char *group;
	char *line;

	while (getline(buf, MAXGROUPLEN, nf)) {
		for (p = buf; *p && isspace((int)*p); p++)
			;	/* skip leading blanks */
		for (; *p && *p != '#' && *p != ' ' && *p != '\t'; p++)
			;
		if (*p == EOS || *p == '#')
			continue;
		*p++ = EOS;

		while (*p == ' ' || *p == '\t') {
			p++;
		}
		if (*p == EOS || *p == '#')
			continue;

		STRCPY(group, buf);
		STRCPY(line, p);
		store(ngtable, group, line);
	}
}