cddb.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  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 210: /* exact matches */
  137. case 211: /* close matches */
  138. if((p = Brdline(&bin, '\n')) == nil)
  139. goto died;
  140. if(p[0] == '.') /* no close matches? */
  141. goto died;
  142. p[Blinelen(&bin)-1] = '\0';
  143. /* accept first match */
  144. nf = tokenize(p, f, nelem(f));
  145. if(nf < 2)
  146. goto died;
  147. categ = f[0];
  148. id = f[1];
  149. /* snarf rest of buffer */
  150. while(p[0] != '.') {
  151. if((p = Brdline(&bin, '\n')) == nil)
  152. goto died;
  153. p[Blinelen(&bin)-1] = '\0';
  154. DPRINT(2, "cddb: %s\n", p);
  155. }
  156. break;
  157. case 202: /* no match */
  158. default:
  159. goto died;
  160. }
  161. t->title = "";
  162. for(i=0; i<t->ntrack; i++)
  163. t->track[i].title = "";
  164. /* fetch results for this cd */
  165. fprint(fd, "cddb read %s %s\r\n", categ, id);
  166. do {
  167. if((p = Brdline(&bin, '\n')) == nil)
  168. goto died;
  169. q = p+Blinelen(&bin)-1;
  170. while(isspace(*q))
  171. *q-- = 0;
  172. DPRINT(2, "cddb %s\n", p);
  173. if(strncmp(p, "DTITLE=", 7) == 0)
  174. t->title = append(t->title, p+7);
  175. else if(strncmp(p, "TTITLE", 6) == 0 && isdigit(p[6])) {
  176. i = atoi(p+6);
  177. if(i < t->ntrack) {
  178. p += 6;
  179. while(isdigit(*p))
  180. p++;
  181. if(*p == '=')
  182. p++;
  183. t->track[i].title = append(t->track[i].title, estrdup(p));
  184. }
  185. }
  186. } while(*p != '.');
  187. fprint(fd, "quit\r\n");
  188. close(fd);
  189. Bterm(&bin);
  190. return 0;
  191. }
  192. void
  193. usage(void)
  194. {
  195. fprint(2, "usage: aux/cddb [-DTt] [-s server] query diskid n ...\n");
  196. exits("usage");
  197. }
  198. void
  199. main(int argc, char **argv)
  200. {
  201. int i;
  202. Toc toc;
  203. ARGBEGIN{
  204. case 'D':
  205. debug = 1;
  206. break;
  207. case 's':
  208. server = EARGF(usage());
  209. break;
  210. case 'T':
  211. Tflag = 1;
  212. /*FALLTHROUGH*/
  213. case 't':
  214. tflag = 1;
  215. break;
  216. }ARGEND
  217. if(argc < 3 || strcmp(argv[0], "query") != 0)
  218. usage();
  219. toc.diskid = strtoul(argv[1], 0, 16);
  220. toc.ntrack = atoi(argv[2]);
  221. if(argc != 3+toc.ntrack+1)
  222. sysfatal("argument count does not match given ntrack");
  223. for(i=0; i<=toc.ntrack; i++)
  224. toc.track[i].n = atoi(argv[3+i]);
  225. if(cddbfilltoc(&toc) < 0)
  226. exits("whoops");
  227. dumpcddb(&toc);
  228. exits(nil);
  229. }