dnsdebug.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <bio.h>
  4. #include <ctype.h>
  5. #include <ip.h>
  6. #include <ndb.h>
  7. #include "dns.h"
  8. enum {
  9. Maxrequest= 128,
  10. };
  11. Cfg cfg;
  12. static char *servername;
  13. static RR *serverrr;
  14. static RR *serveraddrs;
  15. char *dbfile;
  16. int debug;
  17. uchar ipaddr[IPaddrlen]; /* my ip address */
  18. char *logfile = "dnsdebug";
  19. int maxage = 60*60;
  20. char mntpt[Maxpath];
  21. int needrefresh;
  22. ulong now;
  23. int testing;
  24. char *trace;
  25. int traceactivity;
  26. char *zonerefreshprogram;
  27. void docmd(int, char**);
  28. void doquery(char*, char*);
  29. void preloadserveraddrs(void);
  30. int prettyrrfmt(Fmt*);
  31. int setserver(char*);
  32. void squirrelserveraddrs(void);
  33. void
  34. usage(void)
  35. {
  36. fprint(2, "%s: [-rx] [-f db-file]\n", argv0);
  37. exits("usage");
  38. }
  39. void
  40. main(int argc, char *argv[])
  41. {
  42. int n;
  43. Biobuf in;
  44. char *p;
  45. char *f[4];
  46. strcpy(mntpt, "/net");
  47. cfg.inside = 1;
  48. ARGBEGIN{
  49. case 'f':
  50. dbfile = EARGF(usage());
  51. break;
  52. case 'r':
  53. cfg.resolver = 1;
  54. break;
  55. case 'x':
  56. dbfile = "/lib/ndb/external";
  57. strcpy(mntpt, "/net.alt");
  58. break;
  59. default:
  60. usage();
  61. }ARGEND
  62. now = time(nil);
  63. dninit();
  64. fmtinstall('R', prettyrrfmt);
  65. if(myipaddr(ipaddr, mntpt) < 0)
  66. sysfatal("can't read my ip address");
  67. opendatabase();
  68. if(cfg.resolver)
  69. squirrelserveraddrs();
  70. debug = 1;
  71. if(argc > 0){
  72. docmd(argc, argv);
  73. exits(0);
  74. }
  75. Binit(&in, 0, OREAD);
  76. for(print("> "); p = Brdline(&in, '\n'); print("> ")){
  77. p[Blinelen(&in)-1] = 0;
  78. n = tokenize(p, f, 3);
  79. if(n>=1) {
  80. dnpurge(); /* flush the cache */
  81. docmd(n, f);
  82. }
  83. }
  84. exits(0);
  85. }
  86. static char*
  87. longtime(long t)
  88. {
  89. int d, h, m, n;
  90. static char x[128];
  91. for(d = 0; t >= 24*60*60; t -= 24*60*60)
  92. d++;
  93. for(h = 0; t >= 60*60; t -= 60*60)
  94. h++;
  95. for(m = 0; t >= 60; t -= 60)
  96. m++;
  97. n = 0;
  98. if(d)
  99. n += sprint(x, "%d day ", d);
  100. if(h)
  101. n += sprint(x+n, "%d hr ", h);
  102. if(m)
  103. n += sprint(x+n, "%d min ", m);
  104. if(t || n == 0)
  105. sprint(x+n, "%ld sec", t);
  106. return x;
  107. }
  108. int
  109. prettyrrfmt(Fmt *f)
  110. {
  111. RR *rp;
  112. char buf[3*Domlen];
  113. char *p, *e;
  114. Txt *t;
  115. rp = va_arg(f->args, RR*);
  116. if(rp == 0){
  117. strcpy(buf, "<null>");
  118. goto out;
  119. }
  120. p = buf;
  121. e = buf + sizeof(buf);
  122. p = seprint(p, e, "%-32.32s %-15.15s %-5.5s", rp->owner->name,
  123. longtime(rp->db? rp->ttl: (rp->ttl - now)),
  124. rrname(rp->type, buf, sizeof buf));
  125. if(rp->negative){
  126. seprint(p, e, "negative rcode %d\n", rp->negrcode);
  127. goto out;
  128. }
  129. switch(rp->type){
  130. case Thinfo:
  131. seprint(p, e, "\t%s %s", rp->cpu->name, rp->os->name);
  132. break;
  133. case Tcname:
  134. case Tmb:
  135. case Tmd:
  136. case Tmf:
  137. case Tns:
  138. seprint(p, e, "\t%s", (rp->host? rp->host->name: ""));
  139. break;
  140. case Tmg:
  141. case Tmr:
  142. seprint(p, e, "\t%s", (rp->mb? rp->mb->name: ""));
  143. break;
  144. case Tminfo:
  145. seprint(p, e, "\t%s %s", (rp->mb? rp->mb->name: ""),
  146. (rp->rmb? rp->rmb->name: ""));
  147. break;
  148. case Tmx:
  149. seprint(p, e, "\t%lud %s", rp->pref,
  150. (rp->host? rp->host->name: ""));
  151. break;
  152. case Ta:
  153. case Taaaa:
  154. seprint(p, e, "\t%s", (rp->ip? rp->ip->name: ""));
  155. break;
  156. case Tptr:
  157. seprint(p, e, "\t%s", (rp->ptr? rp->ptr->name: ""));
  158. break;
  159. case Tsoa:
  160. seprint(p, e, "\t%s %s %lud %lud %lud %lud %lud",
  161. rp->host->name, rp->rmb->name, rp->soa->serial,
  162. rp->soa->refresh, rp->soa->retry,
  163. rp->soa->expire, rp->soa->minttl);
  164. break;
  165. case Tsrv:
  166. seprint(p, e, "\t%ud %ud %ud %s",
  167. rp->srv->pri, rp->srv->weight, rp->srv->port,
  168. rp->srv->target->name);
  169. break;
  170. case Tnull:
  171. seprint(p, e, "\t%.*H", rp->null->dlen, rp->null->data);
  172. break;
  173. case Ttxt:
  174. p = seprint(p, e, "\t");
  175. for(t = rp->txt; t != nil; t = t->next)
  176. p = seprint(p, e, "%s", t->p);
  177. break;
  178. case Trp:
  179. seprint(p, e, "\t%s %s", rp->rmb->name, rp->rp->name);
  180. break;
  181. case Tkey:
  182. seprint(p, e, "\t%d %d %d", rp->key->flags, rp->key->proto,
  183. rp->key->alg);
  184. break;
  185. case Tsig:
  186. seprint(p, e, "\t%d %d %d %lud %lud %lud %d %s",
  187. rp->sig->type, rp->sig->alg, rp->sig->labels,
  188. rp->sig->ttl, rp->sig->exp, rp->sig->incep,
  189. rp->sig->tag, rp->sig->signer->name);
  190. break;
  191. case Tcert:
  192. seprint(p, e, "\t%d %d %d",
  193. rp->sig->type, rp->sig->tag, rp->sig->alg);
  194. break;
  195. }
  196. out:
  197. return fmtstrcpy(f, buf);
  198. }
  199. void
  200. logsection(char *flag, RR *rp)
  201. {
  202. if(rp == nil)
  203. return;
  204. print("\t%s%R\n", flag, rp);
  205. for(rp = rp->next; rp != nil; rp = rp->next)
  206. print("\t %R\n", rp);
  207. }
  208. void
  209. logreply(int id, uchar *addr, DNSmsg *mp)
  210. {
  211. RR *rp;
  212. char buf[12], resp[32];
  213. switch(mp->flags & Rmask){
  214. case Rok:
  215. strcpy(resp, "OK");
  216. break;
  217. case Rformat:
  218. strcpy(resp, "Format error");
  219. break;
  220. case Rserver:
  221. strcpy(resp, "Server failed");
  222. break;
  223. case Rname:
  224. strcpy(resp, "Nonexistent");
  225. break;
  226. case Runimplimented:
  227. strcpy(resp, "Unimplemented");
  228. break;
  229. case Rrefused:
  230. strcpy(resp, "Refused");
  231. break;
  232. default:
  233. sprint(resp, "%d", mp->flags & Rmask);
  234. break;
  235. }
  236. print("%d: rcvd %s from %I (%s%s%s%s%s)\n", id, resp, addr,
  237. mp->flags & Fauth? "authoritative": "",
  238. mp->flags & Ftrunc? " truncated": "",
  239. mp->flags & Frecurse? " recurse": "",
  240. mp->flags & Fcanrec? " can_recurse": "",
  241. (mp->flags & (Fauth|Rmask)) == (Fauth|Rname)? " nx": "");
  242. for(rp = mp->qd; rp != nil; rp = rp->next)
  243. print("\tQ: %s %s\n", rp->owner->name,
  244. rrname(rp->type, buf, sizeof buf));
  245. logsection("Ans: ", mp->an);
  246. logsection("Auth: ", mp->ns);
  247. logsection("Hint: ", mp->ar);
  248. }
  249. void
  250. logsend(int id, int subid, uchar *addr, char *sname, char *rname, int type)
  251. {
  252. char buf[12];
  253. print("%d.%d: sending to %I/%s %s %s\n", id, subid,
  254. addr, sname, rname, rrname(type, buf, sizeof buf));
  255. }
  256. RR*
  257. getdnsservers(int class)
  258. {
  259. RR *rr;
  260. if(servername == nil)
  261. return dnsservers(class);
  262. rr = rralloc(Tns);
  263. rr->owner = dnlookup("local#dns#servers", class, 1);
  264. rr->host = dnlookup(servername, class, 1);
  265. return rr;
  266. }
  267. void
  268. squirrelserveraddrs(void)
  269. {
  270. RR *rr, *rp, **l;
  271. Request req;
  272. /* look up the resolver address first */
  273. cfg.resolver = 0;
  274. debug = 0;
  275. if(serveraddrs)
  276. rrfreelist(serveraddrs);
  277. serveraddrs = nil;
  278. rr = getdnsservers(Cin);
  279. l = &serveraddrs;
  280. for(rp = rr; rp != nil; rp = rp->next){
  281. if(strcmp(ipattr(rp->host->name), "ip") == 0){
  282. *l = rralloc(Ta);
  283. (*l)->owner = rp->host;
  284. (*l)->ip = rp->host;
  285. l = &(*l)->next;
  286. continue;
  287. }
  288. req.isslave = 1;
  289. req.aborttime = now + Maxreqtm;
  290. *l = dnresolve(rp->host->name, Cin, Ta, &req, 0, 0, Recurse, 0, 0);
  291. while(*l != nil)
  292. l = &(*l)->next;
  293. }
  294. cfg.resolver = 1;
  295. debug = 1;
  296. }
  297. void
  298. preloadserveraddrs(void)
  299. {
  300. RR *rp, **l, *first;
  301. l = &first;
  302. for(rp = serveraddrs; rp != nil; rp = rp->next){
  303. rrcopy(rp, l);
  304. rrattach(first, 1);
  305. }
  306. }
  307. int
  308. setserver(char *server)
  309. {
  310. if(servername != nil){
  311. free(servername);
  312. servername = nil;
  313. cfg.resolver = 0;
  314. }
  315. if(server == nil || *server == 0)
  316. return 0;
  317. servername = strdup(server);
  318. squirrelserveraddrs();
  319. if(serveraddrs == nil){
  320. print("can't resolve %s\n", servername);
  321. cfg.resolver = 0;
  322. } else
  323. cfg.resolver = 1;
  324. return cfg.resolver? 0: -1;
  325. }
  326. void
  327. doquery(char *name, char *tstr)
  328. {
  329. int len, type, rooted;
  330. char *p, *np;
  331. char buf[1024];
  332. RR *rr, *rp;
  333. Request req;
  334. if(cfg.resolver)
  335. preloadserveraddrs();
  336. /* default to an "ip" request if alpha, "ptr" if numeric */
  337. if(tstr == nil || *tstr == 0)
  338. if(strcmp(ipattr(name), "ip") == 0)
  339. tstr = "ptr";
  340. else
  341. tstr = "ip";
  342. /* if name end in '.', remove it */
  343. len = strlen(name);
  344. if(len > 0 && name[len-1] == '.'){
  345. rooted = 1;
  346. name[len-1] = 0;
  347. } else
  348. rooted = 0;
  349. /* inverse queries may need to be permuted */
  350. strncpy(buf, name, sizeof buf);
  351. if(strcmp("ptr", tstr) == 0
  352. && strstr(name, "IN-ADDR") == 0
  353. && strstr(name, "in-addr") == 0){
  354. for(p = name; *p; p++)
  355. ;
  356. *p = '.';
  357. np = buf;
  358. len = 0;
  359. while(p >= name){
  360. len++;
  361. p--;
  362. if(*p == '.'){
  363. memmove(np, p+1, len);
  364. np += len;
  365. len = 0;
  366. }
  367. }
  368. memmove(np, p+1, len);
  369. np += len;
  370. strcpy(np, "in-addr.arpa");
  371. }
  372. /* look it up */
  373. type = rrtype(tstr);
  374. if(type < 0){
  375. print("!unknown type %s\n", tstr);
  376. return;
  377. }
  378. memset(&req, 0, sizeof req);
  379. getactivity(&req, 0);
  380. req.isslave = 1;
  381. req.aborttime = now + Maxreqtm;
  382. rr = dnresolve(buf, Cin, type, &req, 0, 0, Recurse, rooted, 0);
  383. if(rr){
  384. print("----------------------------\n");
  385. for(rp = rr; rp; rp = rp->next)
  386. print("answer %R\n", rp);
  387. print("----------------------------\n");
  388. }
  389. rrfreelist(rr);
  390. putactivity(0);
  391. }
  392. void
  393. docmd(int n, char **f)
  394. {
  395. int tmpsrv;
  396. char *name, *type;
  397. name = type = nil;
  398. tmpsrv = 0;
  399. if(*f[0] == '@') {
  400. if(setserver(f[0]+1) < 0)
  401. return;
  402. switch(n){
  403. case 3:
  404. type = f[2];
  405. /* fall through */
  406. case 2:
  407. name = f[1];
  408. tmpsrv = 1;
  409. break;
  410. }
  411. } else
  412. switch(n){
  413. case 2:
  414. type = f[1];
  415. /* fall through */
  416. case 1:
  417. name = f[0];
  418. break;
  419. }
  420. if(name == nil)
  421. return;
  422. doquery(name, type);
  423. if(tmpsrv)
  424. setserver("");
  425. }