Home | History | Annotate | Line # | Download | only in ypxfr
ypxfr.c revision 1.17
      1 /*	$NetBSD: ypxfr.c,v 1.17 2009/10/20 00:51:15 snj Exp $	*/
      2 
      3 /*
      4  * Copyright (c) 1994 Mats O Jansson <moj (at) stacken.kth.se>
      5  * All rights reserved.
      6  *
      7  * Redistribution and use in source and binary forms, with or without
      8  * modification, are permitted provided that the following conditions
      9  * are met:
     10  * 1. Redistributions of source code must retain the above copyright
     11  *    notice, this list of conditions and the following disclaimer.
     12  * 2. Redistributions in binary form must reproduce the above copyright
     13  *    notice, this list of conditions and the following disclaimer in the
     14  *    documentation and/or other materials provided with the distribution.
     15  *
     16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
     17  * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
     18  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
     20  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     26  * SUCH DAMAGE.
     27  */
     28 
     29 #include <sys/cdefs.h>
     30 #ifndef lint
     31 __RCSID("$NetBSD: ypxfr.c,v 1.17 2009/10/20 00:51:15 snj Exp $");
     32 #endif
     33 
     34 #include <sys/param.h>
     35 #include <sys/types.h>
     36 #include <sys/stat.h>
     37 #include <sys/socket.h>
     38 
     39 #include <netinet/in.h>
     40 #include <arpa/inet.h>
     41 
     42 #include <err.h>
     43 #include <netdb.h>
     44 #include <string.h>
     45 #include <stdio.h>
     46 #include <stdlib.h>
     47 #include <syslog.h>
     48 #include <unistd.h>
     49 
     50 #include <rpc/rpc.h>
     51 #include <rpc/xdr.h>
     52 #include <rpcsvc/yp_prot.h>
     53 #include <rpcsvc/ypclnt.h>
     54 
     55 #include "yplib_host.h"
     56 #include "ypdb.h"
     57 #include "ypdef.h"
     58 
     59 DBM	*db;
     60 
     61 static	int ypxfr_foreach(int, char *, int, char *, int, char *);
     62 
     63 int	main(int, char *[]);
     64 int	get_local_ordernum(char *, char *, u_int *);
     65 int	get_remote_ordernum(CLIENT *, char *, char *, u_int, u_int *);
     66 void	get_map(CLIENT *, char *, char *, struct ypall_callback *);
     67 DBM	*create_db(char *, char *, char *, size_t);
     68 int	install_db(char *, char *, char *);
     69 int	unlink_db(char *, char *, char *);
     70 int	add_order(DBM *, u_int);
     71 int	add_master(CLIENT *, char *, char *, DBM *);
     72 int	add_interdomain(CLIENT *, char *, char *, DBM *);
     73 int	add_secure(CLIENT *, char *, char *, DBM *);
     74 int	send_clear(CLIENT *);
     75 int	send_reply(CLIENT *, int, int);
     76 
     77 int
     78 main(int argc, char **argv)
     79 {
     80 	int need_usage = 0, cflag = 0, fflag = 0, Cflag = 0;
     81 	int ch;
     82 	char *domain;
     83 	char *host = NULL;
     84 	char *srcdomain = NULL;
     85 	char *tid = NULL;
     86 	char *prog = NULL;
     87 	char *ipadd = NULL;
     88 	char *port = NULL;
     89 	char *map = NULL;
     90 	u_int ordernum, new_ordernum;
     91 	struct ypall_callback callback;
     92 	CLIENT *client;
     93 	char temp_map[MAXPATHLEN];
     94 	int status, xfr_status;
     95 
     96 	status = YPPUSH_SUCC;
     97 	client = NULL;
     98 
     99 	if (yp_get_default_domain(&domain))
    100 		errx(1, "can't get YP domain name");
    101 
    102 	while ((ch = getopt(argc, argv, "cd:fh:s:C:")) != -1) {
    103 		switch (ch) {
    104 		case 'c':
    105 			cflag = 1;
    106 			break;
    107 
    108 		case 'd':
    109 			domain = optarg;
    110 			break;
    111 
    112 		case 'f':
    113 			fflag = 1;
    114 			break;
    115 
    116 		case 'h':
    117 			host = optarg;
    118 			break;
    119 
    120 		case 's':
    121 			srcdomain = optarg;
    122 			break;
    123 
    124 		case 'C':
    125 			if (optind + 3 >= argc) {
    126 				need_usage = 1;
    127 				optind = argc;
    128 				break;
    129 			}
    130 			Cflag = 1;
    131 			tid = optarg;
    132 			prog = argv[optind++];
    133 			ipadd = argv[optind++];
    134 			port = argv[optind++];
    135 			break;
    136 
    137 		default:
    138 			need_usage = 1;
    139 		}
    140 	}
    141 	argc -= optind; argv += optind;
    142 
    143 	if (argc != 1)
    144 		need_usage = 1;
    145 
    146 	map = argv[0];
    147 
    148 	if (need_usage) {
    149 		status = YPPUSH_BADARGS;
    150 		fprintf(stderr, "usage: %s [-cf] [-d domain] [-h host] %s\n",
    151 		   getprogname(),
    152 		   "[-s domain] [-C tid prog ipadd port] mapname");
    153 		goto punt;
    154 	}
    155 
    156 #ifdef DEBUG
    157 	openlog("ypxfr", LOG_PID, LOG_DAEMON);
    158 
    159 	syslog(LOG_DEBUG, "ypxfr: Arguments:");
    160 	syslog(LOG_DEBUG, "YP clear to local: %s", (cflag) ? "no" : "yes");
    161 	syslog(LOG_DEBUG, "   Force transfer: %s", (fflag) ? "yes" : "no");
    162 	syslog(LOG_DEBUG, "           domain: %s", domain);
    163 	syslog(LOG_DEBUG, "             host: %s", host);
    164 	syslog(LOG_DEBUG, "    source domain: %s", srcdomain);
    165 	syslog(LOG_DEBUG, "          transid: %s", tid);
    166 	syslog(LOG_DEBUG, "             prog: %s", prog);
    167 	syslog(LOG_DEBUG, "             port: %s", port);
    168 	syslog(LOG_DEBUG, "            ipadd: %s", ipadd);
    169 	syslog(LOG_DEBUG, "              map: %s", map);
    170 #endif
    171 
    172 	if (fflag != 0)
    173 		ordernum = 0;
    174 	else {
    175 		status = get_local_ordernum(domain, map, &ordernum);
    176 		if (status < 0)
    177 			goto punt;
    178 	}
    179 
    180 #ifdef DEBUG
    181         syslog(LOG_DEBUG, "Get Master");
    182 #endif
    183 
    184 	if (host == NULL) {
    185 		if (srcdomain == NULL)
    186 			status = yp_master(domain, map, &host);
    187 	        else
    188 			status = yp_master(srcdomain, map, &host);
    189 
    190 		if (status == 0)
    191 			status = YPPUSH_SUCC;
    192 		else {
    193 			status = -status;
    194 			goto punt;
    195 		}
    196 	}
    197 
    198 #ifdef DEBUG
    199         syslog(LOG_DEBUG, "Connect host: %s", host);
    200 #endif
    201 
    202 	client = yp_bind_host(host, YPPROG, YPVERS, 0, 1);
    203 
    204 	status = get_remote_ordernum(client, domain, map, ordernum,
    205 	    &new_ordernum);
    206 
    207 
    208 	if (status == YPPUSH_SUCC) {
    209 		/* Create temporary db */
    210 		db = create_db(domain, map, temp_map, sizeof(temp_map));
    211 		if (db == NULL)
    212 			status = YPPUSH_DBM;
    213 
    214 	  	/* Add ORDER */
    215 		if (status > 0)
    216 			status = add_order(db, new_ordernum);
    217 
    218 		/* Add MASTER */
    219 		if (status > 0)
    220 			status = add_master(client, domain, map, db);
    221 
    222 	        /* Add INTERDOMAIN */
    223 		if (status > 0)
    224 			status = add_interdomain(client, domain, map, db);
    225 
    226 	        /* Add SECURE */
    227 		if (status > 0)
    228 			status = add_secure(client, domain, map, db);
    229 
    230 		if (status > 0) {
    231 			callback.foreach = ypxfr_foreach;
    232 			get_map(client, domain, map, &callback);
    233 		}
    234 
    235 		/* Close db */
    236 		if (db != NULL)
    237 			ypdb_close(db);
    238 
    239 		/* Rename db */
    240 		if (status > 0)
    241 			status = install_db(domain, map, temp_map);
    242 		else
    243 			status = unlink_db(domain, map, temp_map);
    244 	}
    245 
    246  punt:
    247 	xfr_status = status;
    248 
    249 	if (client != NULL)
    250 		clnt_destroy(client);
    251 
    252 	/* YP_CLEAR */
    253 	if (!cflag) {
    254 		client = yp_bind_local(YPPROG, YPVERS);
    255 		status = send_clear(client);
    256 		clnt_destroy(client);
    257 	}
    258 
    259 	if (Cflag > 0) {
    260 		/* Send Response */
    261 		client = yp_bind_host(ipadd, atoi(prog), 1, atoi(port), 0);
    262 		status = send_reply(client, xfr_status, atoi(tid));
    263 		clnt_destroy(client);
    264 	}
    265 
    266 	exit (0);
    267 }
    268 
    269 static int
    270 ypxfr_foreach(int status, char *keystr, int keylen, char *valstr,
    271 	      int vallen, char *data)
    272 {
    273 	datum key, val;
    274 
    275 	if (status == YP_NOMORE)
    276 		return (0);
    277 
    278 	keystr[keylen] = '\0';
    279 	valstr[vallen] = '\0';
    280 
    281 	key.dptr = keystr;
    282 	key.dsize = strlen(keystr);
    283 
    284 	val.dptr = valstr;
    285 	val.dsize = strlen(valstr);
    286 
    287 	ypdb_store(db, key, val, YPDB_INSERT);
    288 
    289 	return (0);
    290 }
    291 
    292 int
    293 get_local_ordernum(char *domain, char *map, u_int *lordernum)
    294 {
    295 	char map_path[1024];
    296 	char order_key[] = YP_LAST_KEY;
    297 	char order[MAX_LAST_LEN+1];
    298 	struct stat finfo;
    299 	DBM *ldb;
    300 	datum k, v;
    301 	unsigned int status;
    302 
    303 	status = YPPUSH_SUCC;
    304 
    305 	snprintf(map_path, sizeof(map_path), "%s/%s", YP_DB_PATH, domain);
    306 
    307 	/* Make sure we serve the domain. */
    308 	if ((stat(map_path, &finfo)) != 0 ||
    309 	    (S_ISDIR(finfo.st_mode) == 0)) {
    310 		warnx("domain `%s' not found locally", domain);
    311 		status = YPPUSH_NODOM;
    312 		goto out;
    313 	}
    314 
    315 	/* Make sure we serve the map. */
    316 	snprintf(map_path, sizeof(map_path), "%s/%s/%s%s",
    317 	    YP_DB_PATH, domain, map, YPDB_SUFFIX);
    318 	if (stat(map_path, &finfo) != 0) {
    319 		status = YPPUSH_NOMAP;
    320 		goto out;
    321 	}
    322 
    323 	/* Open the map file. */
    324 	snprintf(map_path, sizeof(map_path), "%s/%s/%s",
    325 	    YP_DB_PATH, domain, map);
    326 	ldb = ypdb_open(map_path);
    327 	if (ldb == NULL) {
    328 		status = YPPUSH_DBM;
    329 		goto out;
    330 	}
    331 
    332 	k.dptr = (char *)&order_key;
    333 	k.dsize = YP_LAST_LEN;
    334 
    335 	v = ypdb_fetch(ldb, k);
    336 
    337 	if (v.dptr == NULL)
    338 		*lordernum = 0;
    339 	else {
    340 		strncpy(order, v.dptr, v.dsize);
    341 		order[v.dsize] = '\0';
    342 		*lordernum = (u_int)atoi((char *)&order);
    343 	}
    344 	ypdb_close(ldb);
    345 
    346  out:
    347 	if ((status == YPPUSH_NOMAP) || (status == YPPUSH_DBM)) {
    348 		*lordernum = 0;
    349 		status = YPPUSH_SUCC;
    350 	}
    351 
    352 	return (status);
    353 }
    354 
    355 int
    356 get_remote_ordernum(CLIENT *client, char *domain, char *map,
    357 		    u_int lordernum, u_int *rordernum)
    358 {
    359 	int status;
    360 
    361 	status = yp_order_host(client, domain, map, (int *)rordernum);
    362 
    363 	if (status == 0) {
    364 		if (*rordernum <= lordernum)
    365 			status = YPPUSH_AGE;
    366 		else
    367 			status = YPPUSH_SUCC;
    368 	}
    369 
    370 	return status;
    371 }
    372 
    373 void
    374 get_map(CLIENT *client, char *domain, char *map,
    375 	struct ypall_callback *incallback)
    376 {
    377 
    378 	(void)yp_all_host(client, domain, map, incallback);
    379 }
    380 
    381 DBM *
    382 create_db(char *domain, char *map, char *db_temp, size_t db_temp_len)
    383 {
    384 	static const char template[] = "ypdbXXXXXX";
    385 	DBM *ldb;
    386 
    387 	snprintf(db_temp, db_temp_len, "%s/%s/%s",
    388 	    YP_DB_PATH, domain, template);
    389 
    390 	ldb = ypdb_mktemp(db_temp);
    391 
    392 	return ldb;
    393 }
    394 
    395 int
    396 install_db(char *domain, char *map, char *db_temp)
    397 {
    398 	char db_name[MAXPATHLEN];
    399 
    400 	snprintf(db_name, sizeof(db_name), "%s/%s/%s%s",
    401 	    YP_DB_PATH, domain, map, YPDB_SUFFIX);
    402 
    403 	if (rename(db_temp, db_name)) {
    404 		warn("can't rename `%s' -> `%s'", db_temp, db_name);
    405 		return YPPUSH_YPERR;
    406 	}
    407 
    408 	return YPPUSH_SUCC;
    409 }
    410 
    411 int
    412 unlink_db(char *domain, char *map, char *db_temp)
    413 {
    414 
    415 	if (unlink(db_temp)) {
    416 		warn("can't unlink `%s'", db_temp);
    417 		return YPPUSH_YPERR;
    418 	}
    419 
    420 	return YPPUSH_SUCC;
    421 }
    422 
    423 int
    424 add_order(DBM *ldb, u_int ordernum)
    425 {
    426 	char datestr[11];
    427 	datum key, val;
    428 	char keystr[] = YP_LAST_KEY;
    429 	int status;
    430 
    431 	snprintf(datestr, sizeof(datestr), "%010d", ordernum);
    432 
    433 	key.dptr = keystr;
    434 	key.dsize = strlen(keystr);
    435 
    436 	val.dptr = datestr;
    437 	val.dsize = strlen(datestr);
    438 
    439 	status = ypdb_store(ldb, key, val, YPDB_INSERT);
    440 	if(status >= 0)
    441 		status = YPPUSH_SUCC;
    442 	else
    443 		status = YPPUSH_DBM;
    444 
    445 	return (status);
    446 }
    447 
    448 int
    449 add_master(CLIENT *client, char *domain, char *map, DBM *ldb)
    450 {
    451 	char keystr[] = YP_MASTER_KEY;
    452 	char *master;
    453 	int status;
    454 	datum key, val;
    455 
    456 	master = NULL;
    457 
    458 	/* Get MASTER */
    459 	status = yp_master_host(client, domain, map, &master);
    460 
    461 	if (master != NULL) {
    462 		key.dptr = keystr;
    463 		key.dsize = strlen(keystr);
    464 
    465 		val.dptr = master;
    466 		val.dsize = strlen(master);
    467 
    468 		status = ypdb_store(ldb, key, val, YPDB_INSERT);
    469 		if (status >= 0)
    470 			status = YPPUSH_SUCC;
    471 		else
    472 			status = YPPUSH_DBM;
    473 	}
    474 
    475 	return status;
    476 }
    477 
    478 int
    479 add_interdomain(CLIENT *client, char *domain, char *map, DBM *ldb)
    480 {
    481 	char keystr[] = YP_INTERDOMAIN_KEY;
    482 	char *value;
    483 	int vallen;
    484 	int status;
    485 	datum k, v;
    486 
    487 	/* Get INTERDOMAIN */
    488 	k.dptr = keystr;
    489 	k.dsize = strlen(keystr);
    490 
    491 	status = yp_match_host(client, domain, map,
    492 	    k.dptr, k.dsize, &value, &vallen);
    493 
    494 	if (status == 0 && value) {
    495 		v.dptr = value;
    496 		v.dsize = vallen;
    497 
    498 		if (v.dptr != NULL) {
    499 			status = ypdb_store(ldb, k, v, YPDB_INSERT);
    500 			if (status >= 0)
    501 				status = YPPUSH_SUCC;
    502 			else
    503 				status = YPPUSH_DBM;
    504 		}
    505 	}
    506 
    507 	return status;
    508 }
    509 
    510 int
    511 add_secure(CLIENT *client, char *domain, char *map, DBM *ldb)
    512 {
    513 	char keystr[] = YP_SECURE_KEY;
    514 	char *value;
    515 	int vallen;
    516 	int status;
    517 	datum k, v;
    518 
    519 	/* Get SECURE */
    520 	k.dptr = keystr;
    521 	k.dsize = strlen(keystr);
    522 
    523 	status = yp_match_host(client, domain, map,
    524 	    k.dptr, k.dsize, &value, &vallen);
    525 
    526 	if (status == 0 && value != 0) {
    527 		v.dptr = value;
    528 		v.dsize = vallen;
    529 
    530 		if (v.dptr != NULL) {
    531 			status = ypdb_store(ldb, k, v, YPDB_INSERT);
    532 			if (status >= 0)
    533 				status = YPPUSH_SUCC;
    534 			else
    535 				status = YPPUSH_DBM;
    536 		}
    537 	}
    538 
    539 	return status;
    540 }
    541 
    542 int
    543 send_clear(CLIENT *client)
    544 {
    545 	struct timeval tv;
    546 	int r;
    547 	int status;
    548 
    549 	status = YPPUSH_SUCC;
    550 
    551 	tv.tv_sec = 10;
    552 	tv.tv_usec = 0;
    553 
    554 	/* Send CLEAR */
    555 	r = clnt_call(client, YPPROC_CLEAR, xdr_void, 0, xdr_void, 0, tv);
    556 	if (r != RPC_SUCCESS) {
    557 		clnt_perror(client, "yp_clear: clnt_call");
    558 		status = YPPUSH_RPC;
    559 	}
    560 
    561 	return status;
    562 }
    563 
    564 int
    565 send_reply(CLIENT *client, int status, int tid)
    566 {
    567 	struct timeval tv;
    568 	struct ypresp_xfr resp;
    569 	int r;
    570 
    571 	tv.tv_sec = 10;
    572 	tv.tv_usec = 0;
    573 
    574 	resp.transid = tid;
    575 	resp.xfrstat = status;
    576 
    577 	/* Send XFRRESP */
    578 	r = clnt_call(client, YPPUSHPROC_XFRRESP, xdr_ypresp_xfr, &resp,
    579 	    xdr_void, 0, tv);
    580 	if (r != RPC_SUCCESS) {
    581 		clnt_perror(client, "yppushresp_xdr: clnt_call");
    582 		status = YPPUSH_RPC;
    583 	}
    584 
    585 	return status;
    586 }
    587