#include //printf #include //memset #include //for exit(0); #include //For errno - the error number #include //hostent #include #include //#include #include #include #include #include #include #include #include #include #include "../lualib.h" #include "utils.h" #define CLIENT_NAME "wurl" #define CLIENT_HOST "192.168.9.249" #define CONN_TIME_OUT_S 3 #define MAX_BUFF 1024 #define REQUEST_BOUNDARY "------wURLFormBoundaryVo4QYVaSVseFNpeK" #define GET 0 #define POST 1 typedef struct{ int type; // POST(1) or GET(0) char* resource; // path char* ctype; // content type, used by POST int clen; // content length, used by POST unsigned char* data ; } wurl_header_t; /*get the ip by hostname*/ int wurl_ip_from_hostname(const char * hostname , char* ip) { struct hostent *he; struct in_addr **addr_list; int i; if ( (he = gethostbyname( hostname ) ) == NULL) { // get the host info herror("gethostbyname"); return -1; } addr_list = (struct in_addr **) he->h_addr_list; for(i = 0; addr_list[i] != NULL; i++) { //Return the first one; strcpy(ip , inet_ntoa(*addr_list[i]) ); return 0; } return -1; } /* send a request */ int wurl_request_socket(const char* ip, int port) { int sockfd, bytes_read; struct sockaddr_in dest; // time out setting struct timeval timeout; timeout.tv_sec = CONN_TIME_OUT_S; timeout.tv_usec = 0; if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0 ) { perror("Socket"); return -1; } if (setsockopt (sockfd, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout,sizeof(timeout)) < 0) perror("setsockopt failed\n"); if (setsockopt (sockfd, SOL_SOCKET, SO_SNDTIMEO, (char *)&timeout,sizeof(timeout)) < 0) perror("setsockopt failed\n"); bzero(&dest, sizeof(dest)); dest.sin_family = AF_INET; dest.sin_port = htons(port); if ( inet_aton(ip, &dest.sin_addr.s_addr) == 0 ) { perror(ip); close(sockfd); return -1; } if ( connect(sockfd, (struct sockaddr*)&dest, sizeof(dest)) != 0 ) { close(sockfd); perror("Connect"); return -1; } return sockfd; } /* POST %s HTTP/1.0\r\n Host: %s\r\n User-Agent: %s\r\n Content-Type: %s\r\n Content-Length: %d\r\n\r\n" maybe cookie support? this can cause security problem GET %s HTTP/1.0\r\n Host: %s\r\n User-Agent: %s\r\n\r\n" multipart POST %s HTTP/1.1 Host: %s User-Agent: %s Content-Type: multipart/form-data; boundary=----------287032381131322 Content-Length: %d ------------287032381131322 Content-Disposition: form-data; name="datafile1"; filename="r.gif" Content-Type: image/gif GIF87a.............,...........D..; ------------287032381131322 Content-Disposition: form-data; name="datafile2"; filename="g.gif" Content-Type: image/gif GIF87a.............,...........D..; ------------287032381131322 Content-Disposition: form-data; name="datafile3"; filename="b.gif" Content-Type: image/gif GIF87a.............,...........D..; ------------287032381131322-- */ int wurl_header(int sockfd, wurl_header_t rq) { char buff[MAX_BUFF]; if(sockfd < 0) return -1; if(rq.type == GET) // GET { send(sockfd,"GET ",4,0); send(sockfd,rq.resource, strlen(rq.resource),0); send(sockfd," HTTP/1.0\r\n",11,0); } else { send(sockfd,"POST ",5, 0); send(sockfd,rq.resource, strlen(rq.resource),0); send(sockfd," HTTP/1.0\r\n",11,0); sprintf(buff,"Content-Type: %s\r\n", rq.ctype); send(sockfd,buff,strlen(buff),0); sprintf(buff,"Content-Length: %d\r\n", rq.clen); send(sockfd,buff,strlen(buff),0); } // host dont need to send the host //sprintf(buff,"Host: %s\r\n",CLIENT_HOST); //send(sockfd,buff,strlen(buff),0); // user agent sprintf(buff,"User-Agent: %s\r\n",CLIENT_NAME); send(sockfd,buff,strlen(buff),0); // terminate request send(sockfd,"\r\n",2,0); // if there is data, send out if(rq.type == POST && rq.data) { send(sockfd,rq.data,rq.clen,0); } return 0; } /* send multiple files to server using multipart/form-data */ void wurl_send_files(int sockfd,char* resource, int n, char* name [], char* files[]) { char buff[MAX_BUFF]; wurl_header_t rq; rq.type = POST; rq.resource = resource; // get the total size of data int totalsize = 0; FILE* fd; struct stat st; for(int i = 0; i < n; ++i) { if(stat(files[i], &st) != 0) continue; totalsize += st.st_size; } rq.clen = totalsize; sprintf(buff,"%s; boundary=%s","multipart/form-data",REQUEST_BOUNDARY); rq.ctype = buff; // now send the header wurl_header(sockfd,rq); // now send the files size_t size; for(int i = 0; i < n; ++i) { fd = fopen(files[i],"rb"); if(!fd) continue; // first send the boundary //printf("sending file %s name:%s\n", files[i],name[i]); sprintf(buff,"%s\r\n",REQUEST_BOUNDARY); send(sockfd, buff, strlen(buff),0); // content disposition sprintf(buff,"Content-Disposition: form-data; name=\"%s\"; filename=\"%s\"\r\n", name[i],basename(files[i])); send(sockfd,buff, strlen(buff),0); // content type sprintf(buff,"Content-Type: %s\r\n\r\n",mime(files[i])); send(sockfd,buff, strlen(buff),0); // now send the file while(!feof(fd)) { size = fread(buff,1,MAX_BUFF,fd); send(sockfd,buff,size,0); //if(!__b(client,buffer,size)) return; } fclose(fd); send(sockfd,"\r\n",2,0); } //end the boudary sprintf(buff,"%s--\r\n",REQUEST_BOUNDARY); send(sockfd,buff,strlen(buff),0); } /** * Read the socket request in to a buffer or size * The data is read until the buffer is full or * there are a carrier return character * @param sock socket * @param buf buffer * @param size size of buffer * @return number of bytes read */ int wurl_read_buf(int sock, char*buf,int size) { int i = 0; char c = '\0'; int n; while ((i < size - 1) && (c != '\n')) { n = recv(sock, &c, 1, 0); if (n > 0) { buf[i] = c; i++; } else c = '\n'; } buf[i] = '\0'; return i; } /* POST example wurl_header_t rq; rq.resource = path; rq.type = POST; rq.data = "s=a&q=b#test";//"{\"name\":\"sang\"}"; rq.clen = strlen(rq.data); rq.ctype = "application/x-www-form-urlencoded";//"application/json"; if(wurl_request(hostname,port,&rq,1) == 0) { printf(rq.data); } else { printf("Cannot connect to host\n"); } DOWNLOAD wurl_header_t rq; rq.resource = path; rq.type = GET; if(wurl_download(hostname,port,&rq,file) == 0) { printf("Download sucess ful\n"); } else { printf("Cannot connect to host\n"); } upload example // send files char * names[2]; names[0] = "zip"; names[1] = "text"; char* files[2]; files[0] = "/Users/mrsang/tmp/Archive.zip"; files[1] = "/Users/mrsang/tmp/test.py"; wurl_send_files(sock,path,2,names, files); printf("RETURN:\n"); size = wurl_read_buf(sock,buff, MAX_BUFF); while(size > 0) { printf("%s", buff); size = wurl_read_buf(sock,buff, MAX_BUFF); } close(sock); */ void wurl_response_header(int sock, wurl_header_t* header) { char buff[MAX_BUFF]; int size = wurl_read_buf(sock,buff,MAX_BUFF); char* token; while (size > 0 && strcmp("\r\n",buff)) { char* line = strdup(buff); //printf("LINE %s\n", line); token = strsep(&line,":"); trim(token,' '); if(token != NULL &&strcasecmp(token,"Content-Type") == 0) { header->ctype = strsep(&line,":"); trim(header->ctype,' '); trim(header->ctype,'\n'); trim(header->ctype,'\r'); } else if(token != NULL &&strcasecmp(token,"Content-Length") == 0) { token = strsep(&line,":"); trim(token,' '); header->clen = atoi(token); } //if(line) free(line); size = wurl_read_buf(sock,buff,MAX_BUFF); } } int wurl_read_data(int sock,char** din) { // read line by line, ignore content length int total_length = 0; char* tmp = NULL; int CHUNK = 512; char buff[MAX_BUFF]; char * data = (unsigned char*) malloc(CHUNK); int cursize = CHUNK; int size = wurl_read_buf(sock,buff,MAX_BUFF); while(size > 0) { if(total_length+size > cursize) { tmp = (unsigned char*) realloc(data,total_length + size+ CHUNK ); if(!tmp) { if(data) free(data); break; } cursize = total_length + size+ CHUNK; data = tmp; } memcpy(data+total_length,buff,size); total_length += size; size = wurl_read_buf(sock,buff,MAX_BUFF); } data[total_length] = '\0'; close(sock); *din = data; return total_length; } /* hostname port header for request and respond lazy : if 1, all data is read to the header if 0, user has the responsibility to handler it */ int wurl_request(const char* hostname, int port, wurl_header_t* header, int lazy) { char ip[100]; char buff[MAX_BUFF]; wurl_ip_from_hostname(hostname ,ip); int sock = wurl_request_socket(ip, port); if(sock <= 0) return -1; // send header wurl_header(sock,*header); // read respond header wurl_response_header(sock,header); if(header->ctype == NULL || header->clen == -1) { LOG("Bad data\n"); return -1; } // read data if lazy if(lazy) { // read line by line, ignore content length char * d; header->clen = wurl_read_data(sock,&header->data); return 0; } return sock; } int wurl_download(const char* hostname, int port, wurl_header_t* h, const char* to) { // we will handler the data reading int sock = wurl_request(hostname, port,h,0); unsigned char buff[MAX_BUFF]; if(sock < 0) return -1; FILE* fp = fopen(to,"wb"); int size; if(fp) { while((size = wurl_read_buf(sock,buff, MAX_BUFF)) > 0) { fwrite(buff, size, 1, fp); } fclose(fp); } else { close(sock); return -1; } close(sock); return 0; } /* int main (int argc, char const *argv[]) { if(argc < 4) { printf("wurl [host] [port] [path]\n"); exit(1); } char *hostname = argv[1]; char* path = argv[3]; int port = atoi(argv[2]); char*file = argv[4]; wurl_header_t rq; rq.resource = path; rq.type = POST; rq.data = "s=a&q=b#test";//"{\"name\":\"sang\"}"; rq.clen = strlen(rq.data); rq.ctype = "application/x-www-form-urlencoded";//"application/json"; if(wurl_download(hostname,port,&rq,file) == 0) { printf("Download sucess ful\n"); } else { printf("Cannot connect to host\n"); } return 0; }*/ static int l_get(lua_State *L) { const char* host = luaL_checkstring(L,1); int port = luaL_checknumber(L,2); const char* resource = luaL_checkstring(L,3); wurl_header_t rq; rq.resource = resource; rq.type = GET; if(wurl_request(host,port,&rq,1) == 0) { //printf("Content type is %s\n", rq.ctype); mime_t m = mime_from_type(rq.ctype); lua_newtable(L); lua_pushstring(L,"contentType"); lua_pushstring(L,rq.ctype); lua_settable(L,-3); lua_pushstring(L,"binary"); lua_pushboolean(L,m.bin); lua_settable(L,-3); lua_pushstring(L,"data"); if(m.bin) { //printf("Data is binary, encode as base64 %s\n", rq.ctype); char* dst = (char*) malloc(3*rq.clen/2); Base64encode(dst, rq.data,rq.clen); lua_pushstring(L,dst); free(dst); } else lua_pushstring(L,rq.data); free(rq.data); // be careful lua_settable(L,-3); return 1; } else { lua_pushnil(L); return 1; } } static int l_post(lua_State *L) { const char* host = luaL_checkstring(L,1); // host int port = luaL_checknumber(L,2); // port const char* res = luaL_checkstring(L,3); // resource const char* ctype = luaL_checkstring(L,4); // content type const char* data = luaL_checkstring(L,5); // post data wurl_header_t rq; rq.resource = res; rq.type = POST; rq.ctype = ctype; rq.clen = strlen(data); rq.data = data; if(wurl_request(host,port,&rq,1) == 0) { //printf("Content type is %s\n", rq.ctype); mime_t m = mime_from_type(rq.ctype); lua_newtable(L); lua_pushstring(L,"contentType"); lua_pushstring(L,rq.ctype); lua_settable(L,-3); lua_pushstring(L,"binary"); lua_pushboolean(L,m.bin); lua_settable(L,-3); lua_pushstring(L,"data"); if(m.bin) { //printf("Data is binary, encode as base64 %s\n", rq.ctype); char* dst = (char*) malloc(3*rq.clen/2); Base64encode(dst, rq.data,rq.clen); lua_pushstring(L,dst); free(dst); } else lua_pushstring(L,rq.data); free(rq.data); // be careful lua_settable(L,-3); return 1; } else { lua_pushnil(L); return 1; } } static int l_download(lua_State* L) { // we user only GET for down load and POST for upload const char* host = luaL_checkstring(L,1); // host int port = luaL_checknumber(L,2); // port const char* res = luaL_checkstring(L,3); // resource const char* file = luaL_checkstring(L,4); // file wurl_header_t rq; rq.resource = res; rq.type = GET; if(wurl_download(host,port,&rq,file) == 0) lua_pushboolean(L, true); else lua_pushboolean(L,false); return 1; } static int l_upload(lua_State* L) { const char* host = luaL_checkstring(L,1); int port = luaL_checknumber(L,2); const char* resource = luaL_checkstring(L,3); const char* name = luaL_checkstring(L,4); const char* file = luaL_checkstring(L,5); if(!_exist(file)) goto fail; // request socket char ip[100]; wurl_ip_from_hostname(host ,ip); int sock = wurl_request_socket(ip, port); if(sock <= 0) goto fail; char * names[2]; names[0] = name; char* files[2]; //files[0] = "/Users/mrsang/tmp/Archive.zip"; files[0] = file; //printf("SENDIND DILE\n"); wurl_send_files(sock, resource,1,names,files); wurl_header_t header; //printf("READ HEADER\n"); wurl_response_header(sock, &header); //printf("read data\n"); wurl_read_data(sock,&header.data); if(header.ctype != NULL && header.data) { lua_pushstring(L,header.data); free(header.data); return 1; } fail: lua_pushnil(L); return 1; } static const struct luaL_Reg _lib [] = { {"_get",l_get}, {"_post",l_post}, {"_download",l_download}, {"_upload",l_upload}, {NULL,NULL} }; int luaopen_wurl(lua_State *L) { luaL_newlib(L, _lib); return 1; }