/src/http.c

https://github.com/itanko/last.fm-lib · C · 207 lines · 169 code · 37 blank · 1 comment · 29 complexity · 90e344bab7aaa1077844ab6f66d6ef70 MD5 · raw file

  1. #ifdef WIN32
  2. #include <windows.h>
  3. #include <winsock.h>
  4. #endif
  5. #ifdef LINUX
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <stdarg.h>
  9. #include <stdio.h>
  10. #include <sys/socket.h>
  11. #include <netinet/in.h>
  12. #include <netdb.h>
  13. #include <arpa/inet.h>
  14. #endif
  15. #include "u.h"
  16. #include "http.h"
  17. #ifdef LINUX
  18. #define SOCKET int
  19. #define closesocket close
  20. #endif
  21. #ifdef WIN32
  22. #define vsprintf wvsprintf
  23. #endif
  24. SOCKET sock;
  25. char buff[1024];
  26. char zbuf[4096]; /* zeroed buff */
  27. char * host;
  28. uint server_ip;
  29. ushort server_port;
  30. uint parse_url(struct url * s) {
  31. char * p;
  32. char * n;
  33. s->host = strstr(s->srcstr, "://") + 3;
  34. p = s->host;
  35. while(*p != ':') { p++; } *p = 0;
  36. n = ++p;
  37. while(*p != '/') { p++; } *p = 0;
  38. s->port = atoi(n);
  39. s->path = ++p;
  40. return 0;
  41. }
  42. struct url * _chkurl(struct url ** u)
  43. {
  44. if((*u)->host == 0) {
  45. parse_url(*u);
  46. }
  47. return *u;
  48. }
  49. int httpconnect(char * server, int port)
  50. {
  51. uint res;
  52. struct hostent * h;
  53. struct sockaddr_in adr;
  54. sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  55. if((sock == -1) | (sock == 0)) {
  56. printf("socket() error %i\n", errno);
  57. return;
  58. }
  59. h = gethostbyname(server);
  60. server_ip = *(uint*)h->h_addr_list[0];
  61. server_port = port;
  62. adr.sin_family = AF_INET;
  63. adr.sin_port = htons(server_port);
  64. adr.sin_addr.s_addr = server_ip;
  65. for(;;) {
  66. res = connect(sock,
  67. (struct sockaddr *)&adr,
  68. 16);
  69. if(res == -1) {
  70. printf("connect() error %i (IP=%i:%i)\n",
  71. errno,
  72. server_ip,
  73. server_port);
  74. continue;
  75. } else {
  76. break;
  77. }
  78. }
  79. }
  80. void httpclose() {
  81. closesocket(sock);
  82. }
  83. char * httpreadln() {
  84. uint recvd;
  85. char * c;
  86. *buff = 0;
  87. for(;;) {
  88. recvd = recv(sock, buff, 1024, MSG_PEEK);
  89. if((recvd != 0 ) & (recvd != -1)) break;
  90. recvd = errno;
  91. if (recvd == 0) recvd = 1;
  92. #ifdef WIN32
  93. if(recvd != WSAEWOULDBLOCK)
  94. #endif
  95. #ifdef LINUX
  96. if(recvd != EAGAIN && recvd != EINPROGRESS)
  97. #endif
  98. {
  99. printf("recv() error %i\n", recvd);
  100. return buff;
  101. }
  102. }
  103. c = strchr(buff, '\n');
  104. if (c) c++;
  105. if (c)
  106. {
  107. recv(sock, buff, c - buff, 0);
  108. *(c-1) = 0;
  109. }
  110. return buff;
  111. }
  112. uint _http_sendbuff(u, proc, cookie, buff)
  113. struct url * u;
  114. msgproc_proto proc;
  115. uint cookie;
  116. char * buff;
  117. {
  118. uint r;
  119. httpconnect(u->host, u->port);
  120. send(sock, buff, strlen(buff), 0);
  121. #ifdef DEBUG
  122. printf("Sent: %s\n", buff);
  123. #endif
  124. r = proc (cookie);
  125. httpclose();
  126. return r;
  127. }
  128. /* Add */
  129. void httpskiphdr() {
  130. while(*httpreadln() != 13) {}
  131. }
  132. uint __cdecl httpost(struct url * u,
  133. msgproc_proto proc,
  134. uint cookie,
  135. const char * format,
  136. ...)
  137. {
  138. uint result;
  139. va_list argptr;
  140. va_start( argptr, format );
  141. result = vsprintf( buff, format, argptr );
  142. va_end( argptr );
  143. _chkurl(&u);
  144. result = sprintf(zbuf,
  145. "POST /%s HTTP/1.1\nHost: %s\nContent-Length: %i\nContent-Type: application/x-www-form-urlencoded\n\n%s\n\n",
  146. u->path,
  147. u->host,
  148. strlen(buff),
  149. buff);
  150. return _http_sendbuff(u, proc, cookie, zbuf);
  151. }
  152. uint __cdecl httpget(struct url * u,
  153. msgproc_proto proc,
  154. uint cookie,
  155. const char * format,
  156. ...)
  157. {
  158. uint result;
  159. va_list argptr;
  160. va_start( argptr, format );
  161. result = vsprintf( buff, format, argptr );
  162. va_end( argptr );
  163. _chkurl(&u);
  164. result = sprintf(zbuf,
  165. "GET /%s%s HTTP/1.1\nHost: %s\n\n",
  166. u->path,
  167. buff,
  168. u->host);
  169. return _http_sendbuff(u, proc, cookie, zbuf);
  170. }