dnsdebug.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  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. vlong nowns;
  24. int testing;
  25. char *trace;
  26. int traceactivity;
  27. char *zonerefreshprogram;
  28. void docmd(int, char**);
  29. void doquery(char*, char*);
  30. void preloadserveraddrs(void);
  31. int prettyrrfmt(Fmt*);
  32. int setserver(char*);
  33. void squirrelserveraddrs(void);
  34. void
  35. usage(void)
  36. {
  37. fprint(2, "%s: [-rx] [-f db-file]\n", argv0);
  38. exits("usage");
  39. }
  40. void
  41. main(int argc, char *argv[])
  42. {
  43. int n;
  44. Biobuf in;
  45. char *p;
  46. char *f[4];
  47. strcpy(mntpt, "/net");
  48. cfg.inside = 1;
  49. ARGBEGIN{
  50. case 'f':
  51. dbfile = EARGF(usage());
  52. break;
  53. case 'r':
  54. cfg.resolver = 1;
  55. break;
  56. case 'x':
  57. dbfile = "/lib/ndb/external";
  58. strcpy(mntpt, "/net.alt");
  59. break;
  60. default:
  61. usage();
  62. }ARGEND
  63. now = time(nil);
  64. dninit();
  65. fmtinstall('R', prettyrrfmt);
  66. if(myipaddr(ipaddr, mntpt) < 0)
  67. sysfatal("can't read my ip address");
  68. opendatabase();
  69. if(cfg.resolver)
  70. squirrelserveraddrs();
  71. debug = 1;
  72. if(argc > 0){
  73. docmd(argc, argv);
  74. exits(0);
  75. }
  76. Binit(&in, 0, OREAD);
  77. for(print("> "); p = Brdline(&in, '\n'); print("> ")){
  78. p[Blinelen(&in)-1] = 0;
  79. n = tokenize(p, f, 3);
  80. if(n>=1) {
  81. dnpurge(); /* flush the cache */
  82. docmd(n, f);
  83. }
  84. }
  85. exits(0);
  86. }
  87. static char*
  88. longtime(long t)
  89. {
  90. int d, h, m, n;
  91. static char x[128];
  92. for(d = 0; t >= 24*60*60; t -= 24*60*60)
  93. d++;
  94. for(h = 0; t >= 60*60; t -= 60*60)
  95. h++;
  96. for(m = 0; t >= 60; t -= 60)
  97. m++;
  98. n = 0;
  99. if(d)
  100. n += sprint(x, "%d day ", d);
  101. if(h)
  102. n += sprint(x+n, "%d hr ", h);
  103. if(m)
  104. n += sprint(x+n, "%d min ", m);
  105. if(t || n == 0)
  106. sprint(x+n, "%ld sec", t);
  107. return x;
  108. }
  109. int
  110. prettyrrfmt(Fmt *f)
  111. {
  112. RR *rp;
  113. char buf[3*Domlen];
  114. char *p, *e;
  115. Txt *t;
  116. rp = va_arg(f->args, RR*);
  117. if(rp == 0){
  118. strcpy(buf, "<null>");
  119. goto out;
  120. }
  121. p = buf;
  122. e = buf + sizeof(buf);
  123. p = seprint(p, e, "%-32.32s %-15.15s %-5.5s", rp->owner->name,
  124. longtime(rp->db? rp->ttl: (rp->ttl - now)),
  125. rrname(rp->type, buf, sizeof buf));
  126. if(rp->negative){
  127. seprint(p, e, "negative rcode %d\n", rp->negrcode);
  128. goto out;
  129. }
  130. switch(rp->type){
  131. case Thinfo:
  132. seprint(p, e, "\t%s %s", rp->cpu->name, rp->os->name);
  133. break;
  134. case Tcname:
  135. case Tmb:
  136. case Tmd:
  137. case Tmf:
  138. case Tns:
  139. seprint(p, e, "\t%s", (rp->host? rp->host->name: ""));
  140. break;
  141. case Tmg:
  142. case Tmr:
  143. seprint(p, e, "\t%s", (rp->mb? rp->mb->name: ""));
  144. break;
  145. case Tminfo:
  146. seprint(p, e, "\t%s %s", (rp->mb? rp->mb->name: ""),
  147. (rp->rmb? rp->rmb->name: ""));
  148. break;
  149. case Tmx:
  150. seprint(p, e, "\t%lud %s", rp->pref,
  151. (rp->host? rp->host->name: ""));
  152. break;
  153. case Ta:
  154. case Taaaa:
  155. seprint(p, e, "\t%s", (rp->ip? rp->ip->name: ""));
  156. break;
  157. case Tptr:
  158. seprint(p, e, "\t%s", (rp->ptr? rp->ptr->name: ""));
  159. break;
  160. case Tsoa:
  161. seprint(p, e, "\t%s %s %lud %lud %lud %lud %lud",
  162. rp->host->name, rp->rmb->name, rp->soa->serial,
  163. rp->soa->refresh, rp->soa->retry,
  164. rp->soa->expire, rp->soa->minttl);
  165. break;
  166. case Tsrv:
  167. seprint(p, e, "\t%ud %ud %ud %s",
  168. rp->srv->pri, rp->srv->weight, rp->port, rp->host->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, Authoritative);
  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 && cistrstr(name, ".arpa") == nil){
  352. /* TODO: reversing v6 addrs is harder */
  353. for(p = name; *p; p++)
  354. ;
  355. *p = '.';
  356. np = buf;
  357. len = 0;
  358. while(p >= name){
  359. len++;
  360. p--;
  361. if(*p == '.'){
  362. memmove(np, p+1, len);
  363. np += len;
  364. len = 0;
  365. }
  366. }
  367. memmove(np, p+1, len);
  368. np += len;
  369. strcpy(np, "in-addr.arpa"); /* TODO: ip6.arpa for v6 */
  370. }
  371. /* look it up */
  372. type = rrtype(tstr);
  373. if(type < 0){
  374. print("!unknown type %s\n", tstr);
  375. return;
  376. }
  377. memset(&req, 0, sizeof req);
  378. getactivity(&req, 0);
  379. req.isslave = 1;
  380. req.aborttime = now + Maxreqtm;
  381. rr = dnresolve(buf, Cin, type, &req, 0, 0, Recurse, rooted, 0);
  382. if(rr){
  383. print("----------------------------\n");
  384. for(rp = rr; rp; rp = rp->next)
  385. print("answer %R\n", rp);
  386. print("----------------------------\n");
  387. }
  388. rrfreelist(rr);
  389. putactivity(0);
  390. }
  391. void
  392. docmd(int n, char **f)
  393. {
  394. int tmpsrv;
  395. char *name, *type;
  396. name = type = nil;
  397. tmpsrv = 0;
  398. if(*f[0] == '@') {
  399. if(setserver(f[0]+1) < 0)
  400. return;
  401. switch(n){
  402. case 3:
  403. type = f[2];
  404. /* fall through */
  405. case 2:
  406. name = f[1];
  407. tmpsrv = 1;
  408. break;
  409. }
  410. } else
  411. switch(n){
  412. case 2:
  413. type = f[1];
  414. /* fall through */
  415. case 1:
  416. name = f[0];
  417. break;
  418. }
  419. if(name == nil)
  420. return;
  421. doquery(name, type);
  422. if(tmpsrv)
  423. setserver("");
  424. }