/* * This file is part of the UCB release of Plan 9. It is subject to the license * terms in the LICENSE file found in the top-level directory of this * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No * part of the UCB release of Plan 9, including this file, may be copied, * modified, propagated, or distributed except according to the terms contained * in the LICENSE file. */ /* Portions of this file are Copyright (C) 2015-2018 Giacomo Tesio * See /doc/license/gpl-2.0.txt for details about the licensing. */ /* Portions of this file are Copyright (C) 9front's team. * See /doc/license/9front-mit for details about the licensing. * See http://code.9front.org/hg/plan9front/ for a list of authors. */ #include #include #include #include #include typedef struct x { Ndbtuple *t; Ndbtuple *it; Ndbtuple *nt; } X; X x[4096]; int nx; char *domname; int domnamlen; char* upper(char *x) { char *p; int c; for(p = x; c = *p; p++) *p = toupper(c); return x; } void printArecord(int fd, X *p) { Ndbtuple *nt; char *c; char *dom = 0; char *curdom = 0; int i, cdlen = 0; int mxweight = 0; if(p->nt) { return; } for(nt=p->t; nt; nt = nt->entry) { /* we are only going to handle things in the specified domain */ c = strchr(nt->val, '.'); if (c==0 || strcmp(++c, domname)!=0) continue; i = c - nt->val - 1; if(strcmp(nt->attr, "dom") == 0) { curdom = nt->val; cdlen = i; if (dom == 0) { dom = curdom; fprint(fd, "%-.*s%.*s IN A %s\n", i, nt->val, 15-i, " ", p->it->val); } else fprint(fd, "%-.*s%.*s IN CNAME %s.\n", i, nt->val, 15-i, " ", dom); } else if(strcmp(nt->attr, "mx") == 0) { if (curdom != 0) fprint(fd, "%-.*s%.*s MX %d %s.\n", cdlen, curdom, 15-cdlen, " ", mxweight++, nt->val); } } } void printentry(int fd, X *p) { Ndbtuple *nt; if(p->nt) return; fprint(fd, "%s ", p->it->val); for(nt = p->t; nt; nt = nt->entry) if(strcmp(nt->attr, "dom") == 0) fprint(fd, " %s", nt->val); for(nt = p->t; nt; nt = nt->entry) if(strcmp(nt->attr, "sys") == 0) fprint(fd, " %s", nt->val); fprint(fd, "\n"); } void printsys(int fd, X *p) { Ndbtuple *nt; for(nt = p->t; nt; nt = nt->entry) if(strcmp(nt->attr, "dom") == 0) fprint(fd, "%s\n", nt->val); } void printtxt(int fd, X *p) { int i; Ndbtuple *nt; if(p->nt){ for(;;){ i = strlen(p->it->val); if(strcmp(p->it->val+i-2, ".0") == 0) p->it->val[i-2] = 0; else break; } fprint(fd, "\nNET : %s : %s\n", p->it->val, upper(p->nt->val)); return; } fprint(fd, "HOST : %s :", p->it->val); i = 0; for(nt = p->t; nt; nt = nt->entry) if(strcmp(nt->attr, "dom") == 0){ if(i++ == 0) fprint(fd, " %s", upper(nt->val)); else fprint(fd, ", %s", upper(nt->val)); } fprint(fd, "\n"); } void parse(char *file) { int i; Ndb *db; Ndbtuple *t, *nt, *tt, *ipnett; char *p; db = ndbopen(file); if(db == 0) exits("no database"); while(t = ndbparse(db)){ for(nt = t; nt; nt = nt->entry){ if(strcmp(nt->attr, "ip") == 0) break; if(strcmp(nt->attr, "flavor") == 0 && strcmp(nt->val, "console") == 0) return; } if(nt == 0){ ndbfree(t); continue; } /* dump anything not on our nets */ ipnett = 0; for(tt = t; tt; tt = tt->entry){ if(strcmp(tt->attr, "ipnet") == 0){ ipnett = tt; break; } if(strcmp(tt->attr, "dom") == 0){ i = strlen(tt->val); p = tt->val+i-domnamlen; if(p >= tt->val && strcmp(p, domname) == 0) break; } } if(tt == 0){ ndbfree(t); continue; } for(; nt; nt = nt->entry){ if(strcmp(nt->attr, "ip") != 0) continue; x[nx].it = nt; x[nx].nt = ipnett; x[nx++].t = t; } } } void main(int argc, char *argv[]) { int i, fd; char fn[128]; if (argc>1) domname = argv[1]; else domname = smprint("ndb-dump.%ld", time(0)); domnamlen = strlen(domname); if(argc > 2){ for(i = 2; i < argc; i++) parse(argv[i]); } else { parse("/cfg/ndb/local"); parse("/cfg/ndb/friends"); } sprint(fn, "/cfg/ndb/db.%-.24s", domname); fd = ocreate(fn, OWRITE, 0664); if(fd < 0){ fprint(2, "can't create %s: %r\n", fn); exits("boom"); } fprint(fd, "; This file is generated automatically, do not edit!\n"); for(i = 0; i < nx; i++) printArecord(fd, &x[i]); close(fd); sprint(fn, "/cfg/ndb/equiv.%-.21s", domname); fd = ocreate(fn, OWRITE, 0664); if(fd < 0){ fprint(2, "can't create %s: %r\n", fn); exits("boom"); } for(i = 0; i < nx; i++) printsys(fd, &x[i]); close(fd); sprint(fn, "/cfg/ndb/txt.%-.23s", domname); fd = ocreate(fn, OWRITE, 0664); if(fd < 0){ fprint(2, "can't create %s: %r\n", fn); exits("boom"); } for(i = 0; i < nx; i++) printtxt(fd, &x[i]); close(fd); exits(0); }