xref: /freebsd/libexec/ypxfr/ypxfr_misc.c (revision 4cf49a43559ed9fdad601bdcccd2c55963008675)
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
34 static const char rcsid[] =
35   "$FreeBSD$";
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>
45 struct dom_binding {};
46 #include <rpcsvc/ypclnt.h>
47 #include "ypxfr_extern.h"
48 
49 char *ypxfrerr_string(code)
50 	ypxfrstat code;
51 {
52 	switch(code) {
53 	case YPXFR_SUCC:
54 		return ("Map successfully transfered");
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  */
123 char *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 \
162 create 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 \
171 failed"));
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 
198 unsigned 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 \
230 udp 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 \
238 failed"));
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 
263 int 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