#include "http_parser.h" #include "ddv_socket.h" HTTPReader::HTTPReader(){Clean();} void HTTPReader::Clean(){ seenHeaders = false; seenReq = false; method = "GET"; url = "/"; protocol = "HTTP/1.1"; body = ""; length = 0; HTTPbuffer = ""; headers.erase(headers.begin(), headers.end()); vars.erase(vars.begin(), vars.end()); } bool HTTPReader::CleanForNext(){ seenHeaders = false; seenReq = false; method = "GET"; url = "/"; protocol = "HTTP/1.1"; body = ""; length = 0; headers.erase(headers.begin(), headers.end()); vars.erase(vars.begin(), vars.end()); return parse(); } std::string HTTPReader::BuildRequest(){ std::map::iterator it; std::string tmp = method+" "+url+" "+protocol+"\n"; for (it=headers.begin(); it != headers.end(); it++){ tmp += (*it).first + ": " + (*it).second + "\n"; } tmp += "\n"; tmp += HTTPbuffer; return tmp; } std::string HTTPReader::BuildResponse(std::string code, std::string message){ std::map::iterator it; std::string tmp = protocol+" "+code+" "+message+"\n"; for (it=headers.begin(); it != headers.end(); it++){ tmp += (*it).first + ": " + (*it).second + "\n"; } tmp += "\n"; tmp += HTTPbuffer; return tmp; } void HTTPReader::Trim(std::string & s){ size_t startpos = s.find_first_not_of(" \t"); size_t endpos = s.find_last_not_of(" \t"); if ((std::string::npos == startpos) || (std::string::npos == endpos)){s = "";}else{s = s.substr(startpos, endpos-startpos+1);} } void HTTPReader::SetBody(std::string s){ HTTPbuffer = s; SetHeader("Content-Length", s.length()); } void HTTPReader::SetBody(char * buffer, int len){ HTTPbuffer = ""; HTTPbuffer.append(buffer, len); SetHeader("Content-Length", len); } std::string HTTPReader::GetHeader(std::string i){return headers[i];} std::string HTTPReader::GetVar(std::string i){return vars[i];} void HTTPReader::SetHeader(std::string i, std::string v){ Trim(i); Trim(v); headers[i] = v; } void HTTPReader::SetHeader(std::string i, int v){ Trim(i); char val[128]; sprintf(val, "%i", v); headers[i] = val; } void HTTPReader::SetVar(std::string i, std::string v){ Trim(i); Trim(v); vars[i] = v; } bool HTTPReader::Read(DDV::Socket & sock){ //returned true als hele http packet gelezen is int r = 0; int b = 0; char buffer[500]; while (true){ r = sock.ready(); if (r < 1){ if (r == -1){ #if DEBUG >= 1 fprintf(stderr, "User socket is disconnected.\n"); #endif } return parse(); } b = sock.iread(buffer, 500); HTTPbuffer.append(buffer, b); } return false; }//HTTPReader::ReadSocket bool HTTPReader::Read(FILE * F){ //returned true als hele http packet gelezen is int b = 1; char buffer[500]; while (b > 0){ b = fread(buffer, 1, 500, F); HTTPbuffer.append(buffer, b); } return false; }//HTTPReader::ReadSocket bool HTTPReader::parse(){ size_t f; std::string tmpA, tmpB, tmpC; while (HTTPbuffer != ""){ if (!seenHeaders){ f = HTTPbuffer.find('\n'); if (f == std::string::npos) return false; tmpA = HTTPbuffer.substr(0, f); HTTPbuffer.erase(0, f+1); while (tmpA.find('\r') != std::string::npos){tmpA.erase(tmpA.find('\r'));} if (!seenReq){ seenReq = true; f = tmpA.find(' '); if (f != std::string::npos){method = tmpA.substr(0, f); tmpA.erase(0, f+1);} f = tmpA.find(' '); if (f != std::string::npos){url = tmpA.substr(0, f); tmpA.erase(0, f+1);} f = tmpA.find(' '); if (f != std::string::npos){protocol = tmpA.substr(0, f); tmpA.erase(0, f+1);} //TODO: GET variable parsing? }else{ if (tmpA.size() == 0){ seenHeaders = true; if (GetHeader("Content-Length") != ""){length = atoi(GetHeader("Content-Length").c_str());} }else{ f = tmpA.find(':'); if (f == std::string::npos) continue; tmpB = tmpA.substr(0, f); tmpC = tmpA.substr(f+1); SetHeader(tmpB, tmpC); } } } if (seenHeaders){ if (length > 0){ //TODO: POST variable parsing? if (HTTPbuffer.length() >= length){ body = HTTPbuffer.substr(0, length); HTTPbuffer.erase(0, length); return true; }else{ return false; } }else{ return true; } } } return false; //we should never get here... }//HTTPReader::parse void HTTPReader::SendResponse(DDV::Socket & conn, std::string code, std::string message){ std::string tmp = BuildResponse(code, message); conn.write(tmp); } void HTTPReader::SendBodyPart(DDV::Socket & conn, char * buffer, int len){ std::string tmp; tmp.append(buffer, len); SendBodyPart(conn, tmp); } void HTTPReader::SendBodyPart(DDV::Socket & conn, std::string bodypart){ static char len[10]; int sizelen; sizelen = snprintf(len, 10, "%x\r\n", (unsigned int)bodypart.size()); conn.write(len, sizelen); conn.write(bodypart); conn.write(len+sizelen-2, 2); }