cddb.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  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, "%s: %s: cannot dial: %r\n", argv0, server);
  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, "%s: error talking to cddb server %s\n",
  94. argv0, server);
  95. if(p) {
  96. p[Blinelen(&bin)-1] = 0;
  97. fprint(2, "%s: server says: %s\n", argv0, p);
  98. }
  99. return -1;
  100. }
  101. fprint(fd, "cddb hello gre plan9 9cd 1.0\r\n");
  102. if((p = Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2)
  103. goto died;
  104. /*
  105. * Protocol level 6 is the same as level 5 except that
  106. * the character set is now UTF-8 instead of ISO-8859-1.
  107. */
  108. fprint(fd, "proto 6\r\n");
  109. DPRINT(2, "proto 6\r\n");
  110. if((p = Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2)
  111. goto died;
  112. p[Blinelen(&bin)-1] = 0;
  113. DPRINT(2, "cddb: %s\n", p);
  114. fprint(fd, "cddb query %8.8lux %d", t->diskid, t->ntrack);
  115. DPRINT(2, "cddb query %8.8lux %d", t->diskid, t->ntrack);
  116. for(i=0; i<t->ntrack; i++) {
  117. fprint(fd, " %d", t->track[i].n);
  118. DPRINT(2, " %d", t->track[i].n);
  119. }
  120. fprint(fd, " %d\r\n", t->track[t->ntrack].n);
  121. DPRINT(2, " %d\r\n", t->track[t->ntrack].n);
  122. if((p = Brdline(&bin, '\n')) == nil || atoi(p)/100 != 2)
  123. goto died;
  124. p[Blinelen(&bin)-1] = 0;
  125. DPRINT(2, "cddb: %s\n", p);
  126. nf = tokenize(p, f, nelem(f));
  127. if(nf < 1)
  128. goto died;
  129. switch(atoi(f[0])) {
  130. case 200: /* exact match */
  131. if(nf < 3)
  132. goto died;
  133. categ = f[1];
  134. id = f[2];
  135. break;
  136. case 211: /* close matches */
  137. if((p = Brdline(&bin, '\n')) == nil)
  138. goto died;
  139. if(p[0] == '.') /* no close matches? */
  140. goto died;
  141. p[Blinelen(&bin)-1] = '\0';
  142. /* accept first match */
  143. nf = tokenize(p, f, nelem(f));
  144. if(nf < 2)
  145. goto died;
  146. categ = f[0];
  147. id = f[1];
  148. /* snarf rest of buffer */
  149. while(p[0] != '.') {
  150. if((p = Brdline(&bin, '\n')) == nil)
  151. goto died;
  152. p[Blinelen(&bin)-1] = '\0';
  153. DPRINT(2, "cddb: %s\n", p);
  154. }
  155. break;
  156. case 202: /* no match */
  157. default:
  158. goto died;
  159. }
  160. t->title = "";
  161. for(i=0; i<t->ntrack; i++)
  162. t->track[i].title = "";
  163. /* fetch results for this cd */
  164. fprint(fd, "cddb read %s %s\r\n", categ, id);
  165. do {
  166. if((p = Brdline(&bin, '\n')) == nil)
  167. goto died;
  168. q = p+Blinelen(&bin)-1;
  169. while(isspace(*q))
  170. *q-- = 0;
  171. DPRINT(2, "cddb %s\n", p);
  172. if(strncmp(p, "DTITLE=", 7) == 0)
  173. t->title = append(t->title, p+7);
  174. else if(strncmp(p, "TTITLE", 6) == 0 && isdigit(p[6])) {
  175. i = atoi(p+6);
  176. if(i < t->ntrack) {
  177. p += 6;
  178. while(isdigit(*p))
  179. p++;
  180. if(*p == '=')
  181. p++;
  182. t->track[i].title = append(t->track[i].title, estrdup(p));
  183. }
  184. }
  185. } while(*p != '.');
  186. fprint(fd, "quit\r\n");
  187. close(fd);
  188. Bterm(&bin);
  189. return 0;
  190. }
  191. void
  192. usage(void)
  193. {
  194. fprint(2, "usage: aux/cddb [-DTt] [-s server] query diskid n ...\n");
  195. exits("usage");
  196. }
  197. void
  198. main(int argc, char **argv)
  199. {
  200. int i;
  201. Toc toc;
  202. ARGBEGIN{
  203. case 'D':
  204. debug = 1;
  205. break;
  206. case 's':
  207. server = EARGF(usage());
  208. break;
  209. case 'T':
  210. Tflag = 1;
  211. /*FALLTHROUGH*/
  212. case 't':
  213. tflag = 1;
  214. break;
  215. }ARGEND
  216. if(argc < 3 || strcmp(argv[0], "query") != 0)
  217. usage();
  218. toc.diskid = strtoul(argv[1], 0, 16);
  219. toc.ntrack = atoi(argv[2]);
  220. if(argc != 3+toc.ntrack+1)
  221. sysfatal("argument count does not match given ntrack");
  222. for(i=0; i<=toc.ntrack; i++)
  223. toc.track[i].n = atoi(argv[3+i]);
  224. if(cddbfilltoc(&toc) < 0)
  225. exits("whoops");
  226. dumpcddb(&toc);
  227. exits(nil);
  228. }