root/libexec/ypxfr/ypxfr_misc.c
/*-
 * SPDX-License-Identifier: BSD-4-Clause
 *
 * Copyright (c) 1995
 *      Bill Paul <wpaul@ctr.columbia.edu>.  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.
 */

#include <sys/cdefs.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/param.h>
#include <rpc/rpc.h>
#include <rpcsvc/yp.h>
#include <rpcsvc/ypclnt.h>
#include "ypxfr_extern.h"

const char *
ypxfrerr_string(ypxfrstat code)
{
        switch (code) {
        case YPXFR_SUCC:
                return ("Map successfully transferred");
                break;
        case YPXFR_AGE:
                return ("Master's version not newer");
                break;
        case YPXFR_NOMAP:
                return ("No such map in server's domain");
                break;
        case YPXFR_NODOM:
                return ("Domain not supported by server");
                break;
        case YPXFR_RSRC:
                return ("Local resource allocation failure");
                break;
        case YPXFR_RPC:
                return ("RPC failure talking to server");
                break;
        case YPXFR_MADDR:
                return ("Could not get master server address");
                break;
        case YPXFR_YPERR:
                return ("NIS server/map database error");
                break;
        case YPXFR_BADARGS:
                return ("Request arguments bad");
                break;
        case YPXFR_DBM:
                return ("Local database operation failed");
                break;
        case YPXFR_FILE:
                return ("Local file I/O operation failed");
                break;
        case YPXFR_SKEW:
                return ("Map version skew during transfer");
                break;
        case YPXFR_CLEAR:
                return ("Couldn't send \"clear\" request to local ypserv");
                break;
        case YPXFR_FORCE:
                return ("No local order number in map -- use -f flag");
                break;
        case YPXFR_XFRERR:
                return ("General ypxfr error");
                break;
        case YPXFR_REFUSED:
                return ("Transfer request refused by ypserv");
                break;
        default:
                return ("Unknown error code");
                break;
        }
}

/*
 * These are wrappers for the usual yp_master() and yp_order() functions.
 * They can use either local yplib functions (the real yp_master() and
 * yp_order()) or do direct RPCs to a specified server. The latter is
 * necessary if ypxfr is run on a machine that isn't configured as an
 * NIS client (this can happen very easily: a given machine need not be
 * an NIS client in order to be an NIS server).
 */

/*
 * Careful: yp_master() returns a pointer to a dynamically allocated
 * buffer. Calling ypproc_master_2() ourselves also returns a pointer
 * to dynamically allocated memory, though this time it's memory
 * allocated by the XDR routines. We have to rememver to free() or
 * xdr_free() the memory as required to avoid leaking memory.
 */
char *
ypxfr_get_master(char *domain, char *map, char *source, const int yplib)
{
        static char mastername[MAXPATHLEN + 2];

        bzero((char *)&mastername, sizeof(mastername));

        if (yplib) {
                int res;
                char *master;
                if ((res = yp_master(domain, map, &master))) {
                        switch (res) {
                        case YPERR_DOMAIN:
                                yp_errno = (enum ypstat)YPXFR_NODOM;
                                break;
                        case YPERR_MAP:
                                yp_errno = (enum ypstat)YPXFR_NOMAP;
                                break;
                        case YPERR_YPERR:
                        default:
                                yp_errno = (enum ypstat)YPXFR_YPERR;
                                break;
                        }
                        return(NULL);
                } else {
                        snprintf(mastername, sizeof(mastername), "%s", master);
                        free(master);
                        return((char *)&mastername);
                }
        } else {
                CLIENT *clnt;
                ypresp_master *resp;
                ypreq_nokey req;

                if ((clnt = clnt_create(source,YPPROG,YPVERS,"udp")) == NULL) {
                        yp_error("%s",clnt_spcreateerror("failed to \
create udp handle to ypserv"));
                        yp_errno = (enum ypstat)YPXFR_RPC;
                        return(NULL);
                }

                req.map = map;
                req.domain = domain;
                if ((resp = ypproc_master_2(&req, clnt)) == NULL) {
                        yp_error("%s",clnt_sperror(clnt,"YPPROC_MASTER \
failed"));
                        clnt_destroy(clnt);
                        yp_errno = (enum ypstat)YPXFR_RPC;
                        return(NULL);
                }
                clnt_destroy(clnt);
                if (resp->stat != YP_TRUE) {
                        switch (resp->stat) {
                        case YP_NODOM:
                                yp_errno = (enum ypstat)YPXFR_NODOM;
                                break;
                        case YP_NOMAP:
                                yp_errno = (enum ypstat)YPXFR_NOMAP;
                                break;
                        case YP_YPERR:
                        default:
                                yp_errno = (enum ypstat)YPXFR_YPERR;
                                break;
                        }
                        return(NULL);
                }
                snprintf(mastername, sizeof(mastername), "%s", resp->peer);
/*              xdr_free(xdr_ypresp_master, (char *)&resp); */
                return((char *)&mastername);
        }
}

unsigned long
ypxfr_get_order(char *domain, char *map, char *source, const int yplib)
{
        if (yplib) {
                unsigned int order;
                int res;
                if ((res = yp_order(domain, map, &order))) {
                        switch (res) {
                        case YPERR_DOMAIN:
                                yp_errno = (enum ypstat)YPXFR_NODOM;
                                break;
                        case YPERR_MAP:
                                yp_errno = (enum ypstat)YPXFR_NOMAP;
                                break;
                        case YPERR_YPERR:
                        default:
                                yp_errno = (enum ypstat)YPXFR_YPERR;
                                break;
                        }
                        return(0);
                } else
                        return(order);
        } else {
                CLIENT *clnt;
                ypresp_order *resp;
                ypreq_nokey req;

                if ((clnt = clnt_create(source,YPPROG,YPVERS,"udp")) == NULL) {
                        yp_error("%s",clnt_spcreateerror("couldn't create \
udp handle to ypserv"));
                        yp_errno = (enum ypstat)YPXFR_RPC;
                        return(0);
                }
                req.map = map;
                req.domain = domain;
                if ((resp = ypproc_order_2(&req, clnt)) == NULL) {
                        yp_error("%s", clnt_sperror(clnt, "YPPROC_ORDER \
failed"));
                        clnt_destroy(clnt);
                        yp_errno = (enum ypstat)YPXFR_RPC;
                        return(0);
                }
                clnt_destroy(clnt);
                if (resp->stat != YP_TRUE) {
                        switch (resp->stat) {
                        case YP_NODOM:
                                yp_errno = (enum ypstat)YPXFR_NODOM;
                                break;
                        case YP_NOMAP:
                                yp_errno = (enum ypstat)YPXFR_NOMAP;
                                break;
                        case YP_YPERR:
                        default:
                                yp_errno = (enum ypstat)YPXFR_YPERR;
                                break;
                        }
                        return(0);
                }
                return(resp->ordernum);
        }
}

int
ypxfr_match(char *server, char *domain, char *map, char *key,
    unsigned long keylen)
{
        ypreq_key ypkey;
        ypresp_val *ypval;
        CLIENT *clnt;
        static char buf[YPMAXRECORD + 2];

        bzero(buf, sizeof(buf));

        if ((clnt = clnt_create(server, YPPROG,YPVERS,"udp")) == NULL) {
                yp_error("failed to create UDP handle: %s",
                                        clnt_spcreateerror(server));
                return(0);
        }

        ypkey.domain = domain;
        ypkey.map = map;
        ypkey.key.keydat_len = keylen;
        ypkey.key.keydat_val = key;

        if ((ypval = ypproc_match_2(&ypkey, clnt)) == NULL) {
                clnt_destroy(clnt);
                yp_error("%s: %s", server,
                                clnt_sperror(clnt,"YPPROC_MATCH failed"));
                return(0);
        }

        clnt_destroy(clnt);

        if (ypval->stat != YP_TRUE) {
                xdr_free((xdrproc_t)xdr_ypresp_val, ypval);
                return(0);
        }

        xdr_free((xdrproc_t)xdr_ypresp_val, ypval);

        return(1);
}