cddb.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <bio.h>
  4. #include <ctype.h>
  5. char *server = "freedb.freedb.org";
  6. int debug;
  7. #define DPRINT if(debug)fprint
  8. int tflag;
  9. int Tflag;
  10. typedef struct Track Track;
  11. struct Track {
  12. int n;
  13. char *title;
  14. };
  15. enum {
  16. MTRACK = 64,
  17. };
  18. typedef struct Toc Toc;
  19. struct Toc {
  20. ulong diskid;
  21. int ntrack;
  22. char *title;
  23. Track track[MTRACK];
  24. };
  25. void*
  26. emalloc(uint n)
  27. {
  28. void *p;
  29. p = malloc(n);
  30. if(p == nil)
  31. sysfatal("can't malloc: %r");
  32. memset(p, 0, n);
  33. return p;
  34. }
  35. char*
  36. estrdup(char *s)
  37. {
  38. char *t;
  39. t = emalloc(strlen(s)+1);
  40. strcpy(t, s);
  41. return t;
  42. }
  43. static void
  44. dumpcddb(Toc *t)
  45. {
  46. int i, n, s;
  47. print("title %s\n", t->title);
  48. for(i=0; i<t->ntrack; i++){
  49. if(tflag){
  50. n = t->track[i+1].n;
  51. if(i == t->ntrack-1)
  52. n *= 75;
  53. s = (n - t->track[i].n)/75;
  54. print("%d\t%s\t%d:%2.2d\n", i+1, t->track[i].title, s/60, s%60);
  55. }
  56. else
  57. print("%d\t%s\n", i+1, t->track[i].title);
  58. }
  59. if(Tflag){
  60. s = t->track[i].n;
  61. print("Total time: %d:%2.2d\n", s/60, s%60);
  62. }
  63. }
  64. char*
  65. append(char *a, char *b)
  66. {
  67. char *c;
  68. c = emalloc(strlen(a)+strlen(b)+1);
  69. strcpy(c, a);
  70. strcat(c, b);
  71. return c;
  72. }
  73. static int
  74. cddbfilltoc(Toc *t)
  75. {
  76. int fd;
  77. int i;
  78. char *p, *q;
  79. Biobuf bin;
  80. char *f[10];
  81. int nf;
  82. char *id, *categ;
  83. fd = dial(netmkaddr(server, "tcp", "888"), 0, 0, 0);
  84. if(fd < 0) {
  85. fprint(2, "cannot dial: %r\n");
  86. return -1;
  87. }
  88. Binit(&bin, fd, OREAD);
  89. if((p=Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2) {
  90. died:
  91. close(fd);
  92. Bterm(&bin);
  93. fprint(2, "error talking to server\n");
  94. if(p) {
  95. p[Blinelen(&bin)-1] = 0;
  96. fprint(2, "server says: %s\n", p);
  97. }
  98. return -1;
  99. }
  100. fprint(fd, "cddb hello gre plan9 9cd 1.0\r\n");
  101. if((p = Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2)
  102. goto died;
  103. fprint(fd, "cddb query %8.8lux %d", t->diskid, t->ntrack);
  104. DPRINT(2, "cddb query %8.8lux %d", t->diskid, t->ntrack);
  105. for(i=0; i<t->ntrack; i++) {
  106. fprint(fd, " %d", t->track[i].n);
  107. DPRINT(2, " %d", t->track[i].n);
  108. }
  109. fprint(fd, " %d\r\n", t->track[t->ntrack].n);
  110. DPRINT(2, " %d\r\n", t->track[t->ntrack].n);
  111. if((p = Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2)
  112. goto died;
  113. p[Blinelen(&bin)-1] = 0;
  114. DPRINT(2, "cddb: %s\n", p);
  115. nf = tokenize(p, f, nelem(f));
  116. if(nf < 1)
  117. goto died;
  118. switch(atoi(f[0])) {
  119. case 200: /* exact match */
  120. if(nf < 3)
  121. goto died;
  122. categ = f[1];
  123. id = f[2];
  124. break;
  125. case 211: /* close matches */
  126. if((p = Brdline(&bin, '\n')) == nil)
  127. goto died;
  128. if(p[0] == '.') /* no close matches? */
  129. goto died;
  130. p[Blinelen(&bin)-1] = '\0';
  131. /* accept first match */
  132. nf = tokenize(p, f, nelem(f));
  133. if(nf < 2)
  134. goto died;
  135. categ = f[0];
  136. id = f[1];
  137. /* snarf rest of buffer */
  138. while(p[0] != '.') {
  139. if((p = Brdline(&bin, '\n')) == nil)
  140. goto died;
  141. p[Blinelen(&bin)-1] = '\0';
  142. DPRINT(2, "cddb: %s\n", p);
  143. }
  144. break;
  145. case 202: /* no match */
  146. default:
  147. goto died;
  148. }
  149. t->title = "";
  150. for(i=0; i<t->ntrack; i++)
  151. t->track[i].title = "";
  152. /* fetch results for this cd */
  153. fprint(fd, "cddb read %s %s\r\n", categ, id);
  154. do {
  155. if((p = Brdline(&bin, '\n')) == nil)
  156. goto died;
  157. q = p+Blinelen(&bin)-1;
  158. while(isspace(*q))
  159. *q-- = 0;
  160. DPRINT(2, "cddb %s\n", p);
  161. if(strncmp(p, "DTITLE=", 7) == 0)
  162. t->title = append(t->title, p+7);
  163. else if(strncmp(p, "TTITLE", 6) == 0 && isdigit(p[6])) {
  164. i = atoi(p+6);
  165. if(i < t->ntrack) {
  166. p += 6;
  167. while(isdigit(*p))
  168. p++;
  169. if(*p == '=')
  170. p++;
  171. t->track[i].title = append(t->track[i].title, estrdup(p));
  172. }
  173. }
  174. } while(*p != '.');
  175. fprint(fd, "quit\r\n");
  176. close(fd);
  177. Bterm(&bin);
  178. return 0;
  179. }
  180. void
  181. usage(void)
  182. {
  183. fprint(2, "usage: aux/cddb [-DTt] [-s server] query diskid n ...\n");
  184. exits("usage");
  185. }
  186. void
  187. main(int argc, char **argv)
  188. {
  189. int i;
  190. Toc toc;
  191. ARGBEGIN{
  192. case 'D':
  193. debug = 1;
  194. break;
  195. case 's':
  196. server = EARGF(usage());
  197. break;
  198. case 'T':
  199. Tflag = 1;
  200. /*FALLTHROUGH*/
  201. case 't':
  202. tflag = 1;
  203. break;
  204. }ARGEND
  205. if(argc < 3 || strcmp(argv[0], "query") != 0)
  206. usage();
  207. toc.diskid = strtoul(argv[1], 0, 16);
  208. toc.ntrack = atoi(argv[2]);
  209. if(argc != 3+toc.ntrack+1)
  210. sysfatal("argument count does not match given ntrack");
  211. for(i=0; i<=toc.ntrack; i++)
  212. toc.track[i].n = atoi(argv[3+i]);
  213. if(cddbfilltoc(&toc) < 0)
  214. exits("whoops");
  215. dumpcddb(&toc);
  216. exits(nil);
  217. }