ypxfr_misc.c revision 90297
1/*
2 * Copyright (c) 1995
3 *	Bill Paul <wpaul@ctr.columbia.edu>.  All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 *    notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 *    notice, this list of conditions and the following disclaimer in the
12 *    documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 *    must display the following acknowledgement:
15 *	This product includes software developed by Bill Paul.
16 * 4. Neither the name of the author nor the names of any co-contributors
17 *    may be used to endorse or promote products derived from this software
18 *    without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY Bill Paul AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED.  IN NO EVENT SHALL Bill Paul OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 * SUCH DAMAGE.
31 */
32
33#ifndef lint
34static const char rcsid[] =
35  "$FreeBSD: head/libexec/ypxfr/ypxfr_misc.c 90297 2002-02-06 13:30:31Z des $";
36#endif /* not lint */
37
38#include <stdio.h>
39#include <string.h>
40#include <stdlib.h>
41#include <unistd.h>
42#include <sys/param.h>
43#include <rpc/rpc.h>
44#include <rpcsvc/yp.h>
45struct dom_binding {};
46#include <rpcsvc/ypclnt.h>
47#include "ypxfr_extern.h"
48
49char *ypxfrerr_string(code)
50	ypxfrstat code;
51{
52	switch (code) {
53	case YPXFR_SUCC:
54		return ("Map successfully transferred");
55		break;
56	case YPXFR_AGE:
57		return ("Master's version not newer");
58		break;
59	case YPXFR_NOMAP:
60		return ("No such map in server's domain");
61		break;
62	case YPXFR_NODOM:
63		return ("Domain not supported by server");
64		break;
65	case YPXFR_RSRC:
66		return ("Local resource allocation failure");
67		break;
68	case YPXFR_RPC:
69		return ("RPC failure talking to server");
70		break;
71	case YPXFR_MADDR:
72		return ("Could not get master server address");
73		break;
74	case YPXFR_YPERR:
75		return ("NIS server/map database error");
76		break;
77	case YPXFR_BADARGS:
78		return ("Request arguments bad");
79		break;
80	case YPXFR_DBM:
81		return ("Local database operation failed");
82		break;
83	case YPXFR_FILE:
84		return ("Local file I/O operation failed");
85		break;
86	case YPXFR_SKEW:
87		return ("Map version skew during transfer");
88		break;
89	case YPXFR_CLEAR:
90		return ("Couldn't send \"clear\" request to local ypserv");
91		break;
92	case YPXFR_FORCE:
93		return ("No local order number in map -- use -f flag");
94		break;
95	case YPXFR_XFRERR:
96		return ("General ypxfr error");
97		break;
98	case YPXFR_REFUSED:
99		return ("Transfer request refused by ypserv");
100		break;
101	default:
102		return ("Unknown error code");
103		break;
104	}
105}
106
107/*
108 * These are wrappers for the usual yp_master() and yp_order() functions.
109 * They can use either local yplib functions (the real yp_master() and
110 * yp_order()) or do direct RPCs to a specified server. The latter is
111 * necessary if ypxfr is run on a machine that isn't configured as an
112 * NIS client (this can happen very easily: a given machine need not be
113 * an NIS client in order to be an NIS server).
114 */
115
116/*
117 * Careful: yp_master() returns a pointer to a dynamically allocated
118 * buffer. Calling ypproc_master_2() ourselves also returns a pointer
119 * to dynamically allocated memory, though this time it's memory
120 * allocated by the XDR routines. We have to rememver to free() or
121 * xdr_free() the memory as required to avoid leaking memory.
122 */
123char *ypxfr_get_master(domain,map,source,yplib)
124	char *domain;
125	char *map;
126	char *source;
127	const int yplib;
128{
129	static char mastername[MAXPATHLEN + 2];
130
131	bzero((char *)&mastername, sizeof(mastername));
132
133	if (yplib) {
134		int res;
135		char *master;
136		if ((res = yp_master(domain, map, &master))) {
137			switch (res) {
138			case YPERR_DOMAIN:
139				yp_errno = YPXFR_NODOM;
140				break;
141			case YPERR_MAP:
142				yp_errno = YPXFR_NOMAP;
143				break;
144			case YPERR_YPERR:
145			default:
146				yp_errno = YPXFR_YPERR;
147				break;
148			}
149			return(NULL);
150		} else {
151			snprintf(mastername, sizeof(mastername), "%s", master);
152			free(master);
153			return((char *)&mastername);
154		}
155	} else {
156		CLIENT *clnt;
157		ypresp_master *resp;
158		ypreq_nokey req;
159
160		if ((clnt = clnt_create(source,YPPROG,YPVERS,"udp")) == NULL) {
161			yp_error("%s",clnt_spcreateerror("failed to \
162create udp handle to ypserv"));
163			yp_errno = YPXFR_RPC;
164			return(NULL);
165		}
166
167		req.map = map;
168		req.domain = domain;
169		if ((resp = ypproc_master_2(&req, clnt)) == NULL) {
170			yp_error("%s",clnt_sperror(clnt,"YPPROC_MASTER \
171failed"));
172			clnt_destroy(clnt);
173			yp_errno = YPXFR_RPC;
174			return(NULL);
175		}
176		clnt_destroy(clnt);
177		if (resp->stat != YP_TRUE) {
178			switch (resp->stat) {
179			case YP_NODOM:
180				yp_errno = YPXFR_NODOM;
181				break;
182			case YP_NOMAP:
183				yp_errno = YPXFR_NOMAP;
184				break;
185			case YP_YPERR:
186			default:
187				yp_errno = YPXFR_YPERR;
188				break;
189			}
190			return(NULL);
191		}
192		snprintf(mastername, sizeof(mastername), "%s", resp->peer);
193/*		xdr_free(xdr_ypresp_master, (char *)&resp); */
194		return((char *)&mastername);
195	}
196}
197
198unsigned long ypxfr_get_order(domain, map, source, yplib)
199	char *domain;
200	char *map;
201	char *source;
202	const int yplib;
203{
204	if (yplib) {
205		unsigned long order;
206		int res;
207		if ((res = yp_order(domain, map, (int *)&order))) {
208			switch (res) {
209			case YPERR_DOMAIN:
210				yp_errno = YPXFR_NODOM;
211				break;
212			case YPERR_MAP:
213				yp_errno = YPXFR_NOMAP;
214				break;
215			case YPERR_YPERR:
216			default:
217				yp_errno = YPXFR_YPERR;
218				break;
219			}
220			return(0);
221		} else
222			return(order);
223	} else {
224		CLIENT *clnt;
225		ypresp_order *resp;
226		ypreq_nokey req;
227
228		if ((clnt = clnt_create(source,YPPROG,YPVERS,"udp")) == NULL) {
229			yp_error("%s",clnt_spcreateerror("couldn't create \
230udp handle to ypserv"));
231			yp_errno = YPXFR_RPC;
232			return(0);
233		}
234		req.map = map;
235		req.domain = domain;
236		if ((resp = ypproc_order_2(&req, clnt)) == NULL) {
237			yp_error("%s", clnt_sperror(clnt, "YPPROC_ORDER \
238failed"));
239			clnt_destroy(clnt);
240			yp_errno = YPXFR_RPC;
241			return(0);
242		}
243		clnt_destroy(clnt);
244		if (resp->stat != YP_TRUE) {
245			switch (resp->stat) {
246			case YP_NODOM:
247				yp_errno = YPXFR_NODOM;
248				break;
249			case YP_NOMAP:
250				yp_errno = YPXFR_NOMAP;
251				break;
252			case YP_YPERR:
253			default:
254				yp_errno = YPXFR_YPERR;
255				break;
256			}
257			return(0);
258		}
259		return(resp->ordernum);
260	}
261}
262
263int ypxfr_match(server, domain, map, key, keylen)
264	char *server;
265	char *domain;
266	char *map;
267	char *key;
268	unsigned long keylen;
269{
270	ypreq_key ypkey;
271	ypresp_val *ypval;
272	CLIENT *clnt;
273	static char buf[YPMAXRECORD + 2];
274
275	bzero((char *)buf, sizeof(buf));
276
277	if ((clnt = clnt_create(server, YPPROG,YPVERS,"udp")) == NULL) {
278		yp_error("failed to create UDP handle: %s",
279					clnt_spcreateerror(server));
280		return(0);
281	}
282
283	ypkey.domain = domain;
284	ypkey.map = map;
285	ypkey.key.keydat_len = keylen;
286	ypkey.key.keydat_val = key;
287
288	if ((ypval = ypproc_match_2(&ypkey, clnt)) == NULL) {
289		clnt_destroy(clnt);
290		yp_error("%s: %s", server,
291				clnt_sperror(clnt,"YPPROC_MATCH failed"));
292		return(0);
293	}
294
295	clnt_destroy(clnt);
296
297	if (ypval->stat != YP_TRUE) {
298		xdr_free(xdr_ypresp_val, (char *)ypval);
299		return(0);
300	}
301
302	xdr_free(xdr_ypresp_val, (char *)ypval);
303
304	return(1);
305}
306