#include #include #include #include #include #include #include #include #ifdef WIN32 #include #else #include #include #endif #include "Config.h" #include "Client.h" #include "MD5.h" #include "Graphics.h" #include "Misc.h" #include "interface/Point.h" #include "client/SaveInfo.h" Client::Client(): authUser(0, "") { int i = 0; std::string proxyString(""); for(i = 0; i < THUMB_CACHE_SIZE; i++) { thumbnailCache[i] = NULL; } for(i = 0; i < IMGCONNS; i++) { activeThumbRequests[i] = NULL; activeThumbRequestTimes[i] = 0; activeThumbRequestCompleteTimes[i] = 0; } //Read config std::ifstream configFile; configFile.open("powder.pref", ios::binary); if(configFile) { int fsize = configFile.tellg(); configFile.seekg(0, std::ios::end); fsize = configFile.tellg() - (std::streampos)fsize; configFile.seekg(0, ios::beg); if(fsize) { json::Reader::Read(configDocument, configFile); try { authUser.ID = ((json::Number)(configDocument["User"]["ID"])).Value(); authUser.SessionID = ((json::String)(configDocument["User"]["SessionID"])).Value(); authUser.SessionKey = ((json::String)(configDocument["User"]["SessionKey"])).Value(); authUser.Username = ((json::String)(configDocument["User"]["Username"])).Value(); std::string userElevation = ((json::String)(configDocument["User"]["Elevation"])).Value(); if(userElevation == "Admin") authUser.UserElevation = ElevationAdmin; else if(userElevation == "Mod") authUser.UserElevation = ElevationModerator; else authUser.UserElevation= ElevationNone; } catch (json::Exception &e) { authUser = User(0, ""); std::cerr << "Error: Client [Read User data from pref] " << e.what() << std::endl; } try { proxyString = ((json::String)(configDocument["Proxy"])).Value(); } catch (json::Exception &e) { proxyString = ""; std::cerr << "Error: Client [Read Proxy from pref] " << e.what() << std::endl; } } configFile.close(); } if(proxyString.length()) { http_init((char *)proxyString.c_str()); } else { http_init(NULL); } //Read stamps library std::ifstream stampsLib; stampsLib.open(STAMPS_DIR PATH_SEP "stamps.def", ios::binary); while(true) { char data[11]; memset(data, 0, 11); if(stampsLib.readsome(data, 10)!=10) break; if(!data[0]) break; stampIDs.push_back(data); } stampsLib.close(); } Client::~Client() { ClearThumbnailRequests(); http_done(); //Save config std::ofstream configFile; configFile.open("powder.pref", ios::trunc); if(configFile) { if(authUser.ID) { configDocument["User"]["ID"] = json::Number(authUser.ID); configDocument["User"]["SessionID"] = json::String(authUser.SessionID); configDocument["User"]["SessionKey"] = json::String(authUser.SessionKey); configDocument["User"]["Username"] = json::String(authUser.Username); if(authUser.UserElevation == ElevationAdmin) configDocument["User"]["Elevation"] = json::String("Admin"); else if(authUser.UserElevation == ElevationModerator) configDocument["User"]["Elevation"] = json::String("Mod"); else configDocument["User"]["Elevation"] = json::String("None"); } else { configDocument["User"] = json::Null(); } json::Writer::Write(configDocument, configFile); configFile.close(); } } void Client::SetAuthUser(User user) { authUser = user; } User Client::GetAuthUser() { return authUser; } RequestStatus Client::UploadSave(SaveInfo * save) { lastError = ""; int gameDataLength; char * gameData = NULL; int dataStatus; char * data; int dataLength = 0; std::stringstream userIDStream; userIDStream << authUser.ID; if(authUser.ID) { if(!save->GetGameSave()) { lastError = "Empty game save"; return RequestFailure; } gameData = save->GetGameSave()->Serialise(gameDataLength); if(!gameData) { lastError = "Cannot upload game save"; return RequestFailure; } char * postNames[] = { "Name", "Description", "Data:save.bin", "Publish", NULL }; char * postDatas[] = { (char *)(save->name.c_str()), (char *)(save->Description.c_str()), gameData, (char *)(save->Published?"Public":"Private") }; int postLengths[] = { save->name.length(), save->Description.length(), gameDataLength, save->Published?6:7 }; //std::cout << postNames[0] << " " << postDatas[0] << " " << postLengths[0] << std::endl; data = http_multipart_post("http://" SERVER "/Save.api", postNames, postDatas, postLengths, (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(data && dataStatus == 200) { if(strncmp((const char *)data, "OK", 2)!=0) { if(gameData) free(gameData); free(data); lastError = std::string((const char *)data); return RequestFailure; } else { int tempID; std::stringstream saveIDStream((char *)(data+3)); saveIDStream >> tempID; if(!tempID) { lastError = "Server did not return Save ID"; return RequestFailure; } else { save->id = tempID; } } free(data); if(gameData) free(gameData); return RequestOkay; } else if(data) { free(data); } if(gameData) free(gameData); return RequestFailure; } SaveFile * Client::GetStamp(string stampID) { std::ifstream stampFile; stampFile.open(string(STAMPS_DIR PATH_SEP + stampID + ".stm").c_str(), ios::binary); if(stampFile.is_open()) { stampFile.seekg(0, ios::end); size_t fileSize = stampFile.tellg(); stampFile.seekg(0); unsigned char * tempData = (unsigned char *)malloc(fileSize); stampFile.read((char *)tempData, fileSize); stampFile.close(); SaveFile * file = new SaveFile(string(STAMPS_DIR PATH_SEP + stampID + ".stm").c_str()); GameSave * tempSave = new GameSave((char *)tempData, fileSize); file->SetGameSave(tempSave); return file; } else { return NULL; } } void Client::DeleteStamp(string stampID) { for(int i = 0; i < stampIDs.size(); i++) { if(stampIDs[i] == stampID) { remove(string(STAMPS_DIR PATH_SEP + stampID + ".stm").c_str()); stampIDs.erase(stampIDs.begin()+i); return; } } } string Client::AddStamp(GameSave * saveData) { unsigned t=(unsigned)time(NULL); if (lastStampTime!=t) { lastStampTime=t; lastStampName=0; } else lastStampName++; std::stringstream saveID; //sprintf(saveID, "%08x%02x", lastStampTime, lastStampName); saveID << std::setw(8) << std::setfill('0') << std::hex << lastStampTime << std::setw(2) << std::setfill('0') << std::hex << lastStampName; #ifdef WIN32 _mkdir(STAMPS_DIR); #else mkdir(STAMPS_DIR, 0755); #endif int gameDataLength; char * gameData = saveData->Serialise(gameDataLength); std::ofstream stampStream; stampStream.open(string(STAMPS_DIR PATH_SEP + saveID.str()+".stm").c_str(), ios::binary); stampStream.write((const char *)gameData, gameDataLength); stampStream.close(); stampIDs.push_back(saveID.str()); updateStamps(); return saveID.str(); } void Client::updateStamps() { #ifdef WIN32 _mkdir(STAMPS_DIR); #else mkdir(STAMPS_DIR, 0755); #endif std::ofstream stampsStream; stampsStream.open(string(STAMPS_DIR PATH_SEP "stamps.def").c_str(), ios::binary); for(int i = 0; i < stampIDs.size(); i++) { stampsStream.write(stampIDs[i].c_str(), 10); } stampsStream.write("\0", 1); stampsStream.close(); return; } vector Client::GetStamps() { return stampIDs; } RequestStatus Client::ExecVote(int saveID, int direction) { lastError = ""; int dataStatus; char * data; int dataLength = 0; std::stringstream idStream; idStream << saveID; std::string directionS; if(direction==1) { directionS = "Up"; } else { directionS = "Down"; } std::stringstream userIDStream; userIDStream << authUser.ID; if(authUser.ID) { char * postNames[] = { "ID", "Action", NULL }; char * postDatas[] = { (char*)(idStream.str().c_str()), (char*)(directionS.c_str()) }; int postLengths[] = { idStream.str().length(), directionS.length() }; //std::cout << postNames[0] << " " << postDatas[0] << " " << postLengths[0] << std::endl; data = http_multipart_post("http://" SERVER "/Vote.api", postNames, postDatas, postLengths, (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(data && dataStatus == 200) { if(strncmp((const char *)data, "OK", 2)!=0) { free(data); lastError = std::string((const char *)data); return RequestFailure; } free(data); return RequestOkay; } else if(data) { free(data); } lastError = http_ret_text(dataStatus); return RequestFailure; } unsigned char * Client::GetSaveData(int saveID, int saveDate, int & dataLength) { lastError = ""; int dataStatus; unsigned char * data; dataLength = 0; std::stringstream urlStream; if(saveDate) { urlStream << "http://" << STATICSERVER << "/" << saveID << "_" << saveDate << ".cps"; } else { urlStream << "http://" << STATICSERVER << "/" << saveID << ".cps"; } data = (unsigned char *)http_simple_get((char *)urlStream.str().c_str(), &dataStatus, &dataLength); if(data && dataStatus == 200) { return data; } else if(data) { free(data); } return NULL; } LoginStatus Client::Login(string username, string password, User & user) { lastError = ""; std::stringstream urlStream; std::stringstream hashStream; char passwordHash[33]; char totalHash[33]; user.ID = 0; user.Username = ""; user.SessionID = ""; user.SessionKey = ""; //Doop md5_ascii(passwordHash, (const unsigned char *)password.c_str(), password.length()); passwordHash[32] = 0; hashStream << username << "-" << passwordHash; md5_ascii(totalHash, (const unsigned char *)(hashStream.str().c_str()), hashStream.str().length()); totalHash[32] = 0; char * data; int dataStatus, dataLength; char * postNames[] = { "Username", "Hash", NULL }; char * postDatas[] = { (char*)username.c_str(), totalHash }; int postLengths[] = { username.length(), 32 }; data = http_multipart_post("http://" SERVER "/Login.json", postNames, postDatas, postLengths, NULL, NULL, NULL, &dataStatus, &dataLength); //data = http_auth_get("http://" SERVER "/Login.json", (char*)username.c_str(), (char*)password.c_str(), NULL, &dataStatus, &dataLength); if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); json::Number tempStatus = objDocument["Status"]; free(data); if(tempStatus.Value() == 1) { json::Number userIDTemp = objDocument["UserID"]; json::String sessionIDTemp = objDocument["SessionID"]; json::String sessionKeyTemp = objDocument["SessionKey"]; json::String userElevationTemp = objDocument["Elevation"]; user.Username = username; user.ID = userIDTemp.Value(); user.SessionID = sessionIDTemp.Value(); user.SessionKey = sessionKeyTemp.Value(); std::string userElevation = userElevationTemp.Value(); if(userElevation == "Admin") user.UserElevation = ElevationAdmin; else if(userElevation == "Mod") user.UserElevation = ElevationModerator; else user.UserElevation= ElevationNone; return LoginOkay; } else { json::String tempError = objDocument["Error"]; lastError = tempError.Value(); return LoginError; } } catch (json::Exception &e) { lastError = "Server responded with crap"; return LoginError; } } else { lastError = http_ret_text(dataStatus); } if(data) { free(data); } return LoginError; } RequestStatus Client::DeleteSave(int saveID) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/Delete.json?ID=" << saveID << "&Mode=Delete&Key=" << authUser.SessionKey; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); int status = ((json::Number)objDocument["Status"]).Value(); if(status!=1) goto failure; } catch (json::Exception &e) { lastError = "Could not read response"; goto failure; } } else { lastError = http_ret_text(dataStatus); goto failure; } if(data) free(data); return RequestOkay; failure: if(data) free(data); return RequestFailure; } RequestStatus Client::FavouriteSave(int saveID, bool favourite) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/Favourite.json?ID=" << saveID << "&Key=" << authUser.SessionKey; if(!favourite) urlStream << "&Mode=Remove"; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); int status = ((json::Number)objDocument["Status"]).Value(); if(status!=1) { lastError = ((json::String)objDocument["Error"]).Value(); goto failure; } } catch (json::Exception &e) { lastError = "Could not read response"; goto failure; } } else { lastError = http_ret_text(dataStatus); goto failure; } if(data) free(data); return RequestOkay; failure: if(data) free(data); return RequestFailure; } RequestStatus Client::ReportSave(int saveID, std::string message) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/Report.json?ID=" << saveID << "&Key=" << authUser.SessionKey; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; char * postNames[] = { "Reason", NULL }; char * postDatas[] = { (char*)(message.c_str()) }; int postLengths[] = { message.length() }; data = http_multipart_post((char *)urlStream.str().c_str(), postNames, postDatas, postLengths, (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); int status = ((json::Number)objDocument["Status"]).Value(); if(status!=1) { lastError = ((json::String)objDocument["Error"]).Value(); goto failure; } } catch (json::Exception &e) { lastError = "Could not read response"; goto failure; } } else { lastError = http_ret_text(dataStatus); goto failure; } if(data) free(data); return RequestOkay; failure: if(data) free(data); return RequestFailure; } RequestStatus Client::UnpublishSave(int saveID) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/Delete.json?ID=" << saveID << "&Mode=Unpublish&Key=" << authUser.SessionKey; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return RequestFailure; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); int status = ((json::Number)objDocument["Status"]).Value(); if(status!=1) goto failure; } catch (json::Exception &e) { lastError = "Could not read response"; goto failure; } } else { lastError = http_ret_text(dataStatus); goto failure; } if(data) free(data); return RequestOkay; failure: if(data) free(data); return RequestFailure; } SaveInfo * Client::GetSave(int saveID, int saveDate) { lastError = ""; std::stringstream urlStream; urlStream << "http://" << SERVER << "/Browse/View.json?ID=" << saveID; if(saveDate) { urlStream << "&Date=" << saveDate; } char * data; int dataStatus, dataLength; //Save(int _id, int _votesUp, int _votesDown, string _userName, string _name, string description_, string date_, bool published_): if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { data = http_simple_get((char *)urlStream.str().c_str(), &dataStatus, &dataLength); } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); json::Number tempID = objDocument["ID"]; json::Number tempScoreUp = objDocument["ScoreUp"]; json::Number tempScoreDown = objDocument["ScoreDown"]; json::Number tempMyScore = objDocument["ScoreMine"]; json::String tempUsername = objDocument["Username"]; json::String tempName = objDocument["Name"]; json::String tempDescription = objDocument["Description"]; json::Number tempDate = objDocument["Date"]; json::Boolean tempPublished = objDocument["Published"]; json::Boolean tempFavourite = objDocument["Favourite"]; json::Array tagsArray = objDocument["Tags"]; vector tempTags; for(int j = 0; j < tagsArray.Size(); j++) { json::String tempTag = tagsArray[j]; tempTags.push_back(tempTag.Value()); } SaveInfo * tempSave = new SaveInfo( tempID.Value(), tempDate.Value(), tempScoreUp.Value(), tempScoreDown.Value(), tempMyScore.Value(), tempUsername.Value(), tempName.Value(), tempDescription.Value(), tempPublished.Value(), tempTags ); tempSave->Favourite = tempFavourite.Value(); return tempSave; } catch (json::Exception &e) { lastError = "Could not read response"; return NULL; } } else { lastError = http_ret_text(dataStatus); } return NULL; } Thumbnail * Client::GetPreview(int saveID, int saveDate) { std::stringstream urlStream; urlStream << "http://" << STATICSERVER << "/" << saveID; if(saveDate) { urlStream << "_" << saveDate; } urlStream << "_large.pti"; pixel * thumbData; char * data; int status, data_size, imgw, imgh; data = http_simple_get((char *)urlStream.str().c_str(), &status, &data_size); if (status == 200 && data) { thumbData = Graphics::ptif_unpack(data, data_size, &imgw, &imgh); if(data) { free(data); } if(thumbData) { return new Thumbnail(saveID, saveDate, thumbData, ui::Point(imgw, imgh)); free(thumbData); } else { thumbData = (pixel *)malloc((128*128) * PIXELSIZE); return new Thumbnail(saveID, saveDate, thumbData, ui::Point(128, 128)); free(thumbData); } } else { if(data) { free(data); } } return new Thumbnail(saveID, saveDate, (pixel *)malloc((128*128) * PIXELSIZE), ui::Point(128, 128)); } std::vector * Client::GetComments(int saveID, int start, int count) { lastError = ""; std::vector * commentArray = new std::vector(); std::stringstream urlStream; char * data; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/View.json?ID=" << saveID << "&Mode=Comments&Start=" << start << "&Count=" << count; data = http_simple_get((char *)urlStream.str().c_str(), &dataStatus, &dataLength); if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Array commentsArray; json::Reader::Read(commentsArray, dataStream); for(int j = 0; j < commentsArray.Size(); j++) { json::Number tempUserID = commentsArray[j]["UserID"]; json::String tempUsername = commentsArray[j]["Username"]; json::String tempComment = commentsArray[j]["Text"]; commentArray->push_back( new SaveComment( tempUserID.Value(), tempUsername.Value(), tempComment.Value() ) ); } } catch (json::Exception &e) { lastError = "Could not read response"; } } else { lastError = http_ret_text(dataStatus); } if(data) free(data); return commentArray; } std::vector * Client::SearchSaves(int start, int count, string query, string sort, std::string category, int & resultCount) { lastError = ""; resultCount = 0; std::vector * saveArray = new std::vector(); std::stringstream urlStream; char * data; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse.json?Start=" << start << "&Count=" << count; if(query.length() || sort.length()) { urlStream << "&Search_Query="; if(query.length()) urlStream << URLEscape(query); if(sort == "date") { if(query.length()) urlStream << URLEscape(" "); urlStream << URLEscape("sort:") << URLEscape(sort); } } if(category.length()) { urlStream << "&Category=" << URLEscape(category); } if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { data = http_simple_get((char *)urlStream.str().c_str(), &dataStatus, &dataLength); } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Object objDocument; json::Reader::Read(objDocument, dataStream); json::Number tempCount = objDocument["Count"]; resultCount = tempCount.Value(); json::Array savesArray = objDocument["Saves"]; for(int j = 0; j < savesArray.Size(); j++) { json::Number tempID = savesArray[j]["ID"]; json::Number tempDate = savesArray[j]["Date"]; json::Number tempScoreUp = savesArray[j]["ScoreUp"]; json::Number tempScoreDown = savesArray[j]["ScoreDown"]; json::String tempUsername = savesArray[j]["Username"]; json::String tempName = savesArray[j]["Name"]; saveArray->push_back( new SaveInfo( tempID.Value(), tempDate.Value(), tempScoreUp.Value(), tempScoreDown.Value(), tempUsername.Value(), tempName.Value() ) ); } } catch (json::Exception &e) { lastError = "Could not read response"; } } else { lastError = http_ret_text(dataStatus); } if(data) free(data); return saveArray; } void Client::ClearThumbnailRequests() { for(int i = 0; i < IMGCONNS; i++) { if(activeThumbRequests[i]) { http_async_req_close(activeThumbRequests[i]); activeThumbRequests[i] = NULL; activeThumbRequestTimes[i] = 0; activeThumbRequestCompleteTimes[i] = 0; } } } Thumbnail * Client::GetThumbnail(int saveID, int saveDate) { std::stringstream urlStream; std::stringstream idStream; int i = 0, currentTime = time(NULL); //Check active requests for any "forgotten" requests for(i = 0; i < IMGCONNS; i++) { //If the request is active, and we've recieved a response if(activeThumbRequests[i] && http_async_req_status(activeThumbRequests[i])) { //If we haven't already, mark the request as completed if(!activeThumbRequestCompleteTimes[i]) { activeThumbRequestCompleteTimes[i] = time(NULL); } else if(activeThumbRequestCompleteTimes[i] < (currentTime-2)) //Otherwise, if it completed more than 2 seconds ago, destroy it. { http_async_req_close(activeThumbRequests[i]); activeThumbRequests[i] = NULL; activeThumbRequestTimes[i] = 0; activeThumbRequestCompleteTimes[i] = 0; } } } for(i = 0; i < THUMB_CACHE_SIZE; i++) { if(thumbnailCache[i] && thumbnailCache[i]->ID == saveID && thumbnailCache[i]->Datestamp == saveDate) return thumbnailCache[i]; } urlStream << "http://" << STATICSERVER << "/" << saveID; if(saveDate) { urlStream << "_" << saveDate; } urlStream << "_small.pti"; idStream << saveID << ":" << saveDate; std::string idString = idStream.str(); bool found = false; for(i = 0; i < IMGCONNS; i++) { if(activeThumbRequests[i] && activeThumbRequestIDs[i] == idString) { found = true; if(http_async_req_status(activeThumbRequests[i])) { pixel * thumbData; char * data; int status, data_size, imgw, imgh; data = http_async_req_stop(activeThumbRequests[i], &status, &data_size); free(activeThumbRequests[i]); activeThumbRequests[i] = NULL; if (status == 200 && data) { thumbData = Graphics::ptif_unpack(data, data_size, &imgw, &imgh); if(data) { free(data); } thumbnailCacheNextID %= THUMB_CACHE_SIZE; if(thumbnailCache[thumbnailCacheNextID]) { delete thumbnailCache[thumbnailCacheNextID]; } if(thumbData) { thumbnailCache[thumbnailCacheNextID] = new Thumbnail(saveID, saveDate, thumbData, ui::Point(imgw, imgh)); free(thumbData); } else { thumbData = (pixel *)malloc((128*128) * PIXELSIZE); thumbnailCache[thumbnailCacheNextID] = new Thumbnail(saveID, saveDate, thumbData, ui::Point(128, 128)); free(thumbData); } return thumbnailCache[thumbnailCacheNextID++]; } else { if(data) { free(data); } thumbnailCacheNextID %= THUMB_CACHE_SIZE; if(thumbnailCache[thumbnailCacheNextID]) { delete thumbnailCache[thumbnailCacheNextID]; } thumbData = (pixel *)malloc((128*128) * PIXELSIZE); thumbnailCache[thumbnailCacheNextID] = new Thumbnail(saveID, saveDate, thumbData, ui::Point(128, 128)); free(thumbData); return thumbnailCache[thumbnailCacheNextID++]; } } } } if(!found) { for(i = 0; i < IMGCONNS; i++) { if(!activeThumbRequests[i]) { activeThumbRequests[i] = http_async_req_start(NULL, (char *)urlStream.str().c_str(), NULL, 0, 1); activeThumbRequestTimes[i] = currentTime; activeThumbRequestCompleteTimes[i] = 0; activeThumbRequestIDs[i] = idString; return NULL; } } } //http_async_req_start(http, urlStream.str().c_str(), NULL, 0, 1); return NULL; } std::vector * Client::RemoveTag(int saveID, string tag) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/EditTag.json?Op=delete&ID=" << saveID << "&Tag=" << tag << "&Key=" << authUser.SessionKey;; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return NULL; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Array tagsArray; json::Reader::Read(tagsArray, dataStream); tags = new std::vector(); for(int j = 0; j < tagsArray.Size(); j++) { json::String tempTag = tagsArray[j]; tags->push_back(tempTag.Value()); } } catch (json::Exception &e) { lastError = "Could not read response"; } } else { lastError = http_ret_text(dataStatus); } if(data) free(data); return tags; } std::vector * Client::AddTag(int saveID, string tag) { lastError = ""; std::vector * tags = NULL; std::stringstream urlStream; char * data = NULL; int dataStatus, dataLength; urlStream << "http://" << SERVER << "/Browse/EditTag.json?Op=add&ID=" << saveID << "&Tag=" << tag << "&Key=" << authUser.SessionKey; if(authUser.ID) { std::stringstream userIDStream; userIDStream << authUser.ID; data = http_auth_get((char *)urlStream.str().c_str(), (char *)(userIDStream.str().c_str()), NULL, (char *)(authUser.SessionID.c_str()), &dataStatus, &dataLength); } else { lastError = "Not authenticated"; return NULL; } if(dataStatus == 200 && data) { try { std::istringstream dataStream(data); json::Array tagsArray; json::Reader::Read(tagsArray, dataStream); tags = new std::vector(); for(int j = 0; j < tagsArray.Size(); j++) { json::String tempTag = tagsArray[j]; tags->push_back(tempTag.Value()); } } catch (json::Exception &e) { lastError = "Could not read response"; } } else { lastError = http_ret_text(dataStatus); } if(data) free(data); return tags; }