/* sfdb_freesound_mootcher.cpp ********************************************************************** Adapted for Ardour by Ben Loftis, March 2008 Updated to new Freesound API by Colin Fletcher, November 2011 Mootcher 23-8-2005 Mootcher Online Access to thefreesoundproject website http://freesound.iua.upf.edu/ GPL 2005 Jorn Lemon mail for questions/remarks: mootcher@twistedlemon.nl or go to the freesound website forum ----------------------------------------------------------------- Includes: curl.h (version 7.14.0) Librarys: libcurl.lib ----------------------------------------------------------------- Licence GPL: This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *************************************************************************************/ #include "sfdb_freesound_mootcher.h" #include "pbd/xml++.h" #include "pbd/error.h" #include #include #include #include #include #include "i18n.h" #include "ardour/audio_library.h" using namespace PBD; static const std::string base_url = "http://www.freesound.org/api"; static const std::string api_key = "9d77cb8d841b4bcfa960e1aae62224eb"; // ardour3 //------------------------------------------------------------------------ Mootcher::Mootcher() : curl(curl_easy_init()) { std::string path; path = Glib::get_home_dir() + "/Freesound/"; changeWorkingDir ( path.c_str() ); }; //------------------------------------------------------------------------ Mootcher:: ~Mootcher() { curl_easy_cleanup(curl); } //------------------------------------------------------------------------ void Mootcher::changeWorkingDir(const char *saveLocation) { basePath = saveLocation; #ifdef __WIN32__ std::string replace = "/"; size_t pos = basePath.find("\\"); while( pos != std::string::npos ){ basePath.replace(pos, 1, replace); pos = basePath.find("\\"); } #endif // size_t pos2 = basePath.find_last_of("/"); if(basePath.length() != (pos2+1)) basePath += "/"; } void Mootcher::ensureWorkingDir () { std::string p = Glib::build_filename (basePath, "snd"); if (!Glib::file_test (p, Glib::FILE_TEST_IS_DIR)) { if (g_mkdir_with_parents (p.c_str(), 0775) != 0) { PBD::error << "Unable to create Mootcher working dir" << endmsg; } } } //------------------------------------------------------------------------ size_t Mootcher::WriteMemoryCallback(void *ptr, size_t size, size_t nmemb, void *data) { register int realsize = (int)(size * nmemb); struct MemoryStruct *mem = (struct MemoryStruct *)data; mem->memory = (char *)realloc(mem->memory, mem->size + realsize + 1); if (mem->memory) { memcpy(&(mem->memory[mem->size]), ptr, realsize); mem->size += realsize; mem->memory[mem->size] = 0; } return realsize; } //------------------------------------------------------------------------ std::string Mootcher::sortMethodString(enum sortMethod sort) { // given a sort type, returns the string value to be passed to the API to // sort the results in the requested way. switch (sort) { case sort_duration_desc: return "duration_desc"; case sort_duration_asc: return "duration_asc"; case sort_created_desc: return "created_desc"; case sort_created_asc: return "created_asc"; case sort_downloads_desc: return "downloads_desc"; case sort_downloads_asc: return "downloads_asc"; case sort_rating_desc: return "rating_desc"; case sort_rating_asc: return "rating_asc"; default: return ""; } } //------------------------------------------------------------------------ void Mootcher::setcUrlOptions() { // basic init for curl curl_global_init(CURL_GLOBAL_ALL); // some servers don't like requests that are made without a user-agent field, so we provide one curl_easy_setopt(curl, CURLOPT_USERAGENT, "libcurl-agent/1.0"); // setup curl error buffer curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorBuffer); // Allow redirection curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1); // Allow connections to time out (without using signals) curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1); curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 30); } std::string Mootcher::doRequest(std::string uri, std::string params) { std::string result; struct MemoryStruct xml_page; xml_page.memory = NULL; xml_page.size = 0; setcUrlOptions(); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback); curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *) &xml_page); // curl_easy_setopt(curl, CURLOPT_HTTPGET, 1); // curl_easy_setopt(curl, CURLOPT_POSTFIELDS, postMessage.c_str()); // curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, -1); // the url to get std::string url = base_url + uri + "?"; if (params != "") { url += params + "&api_key=" + api_key + "&format=xml"; } else { url += "api_key=" + api_key + "&format=xml"; } curl_easy_setopt(curl, CURLOPT_URL, url.c_str() ); // perform online request CURLcode res = curl_easy_perform(curl); if( res != 0 ) { error << string_compose (_("curl error %1 (%2)"), res, curl_easy_strerror(res)) << endmsg; return ""; } // free the memory if (xml_page.memory) { result = xml_page.memory; } free (xml_page.memory); xml_page.memory = NULL; xml_page.size = 0; return result; } std::string Mootcher::searchText(std::string query, int page, std::string filter, enum sortMethod sort) { std::string params = ""; char buf[24]; if (page > 1) { snprintf(buf, 23, "p=%d&", page); params += buf; } char *eq = curl_easy_escape(curl, query.c_str(), query.length()); params += "q=\"" + std::string(eq) + "\""; free(eq); if (filter != "") { char *ef = curl_easy_escape(curl, filter.c_str(), filter.length()); params += "&f=" + std::string(ef); free(ef); } if (sort) params += "&s=" + sortMethodString(sort); params += "&fields=id,original_filename,duration,filesize,samplerate,serve"; return doRequest("/sounds/search", params); } //------------------------------------------------------------------------ std::string Mootcher::getSoundResourceFile(std::string ID) { std::string originalSoundURI; std::string audioFileName; std::string xml; // download the xmlfile into xml_page xml = doRequest("/sounds/" + ID, ""); XMLTree doc; doc.read_buffer( xml.c_str() ); XMLNode *freesound = doc.root(); // if the page is not a valid xml document with a 'freesound' root if (freesound == NULL) { error << _("getSoundResourceFile: There is no valid root in the xml file") << endmsg; return ""; } if (strcmp(doc.root()->name().c_str(), "response") != 0) { error << string_compose (_("getSoundResourceFile: root = %1, != response"), doc.root()->name()) << endmsg; return ""; } XMLNode *name = freesound->child("original_filename"); // get the file name and size from xml file if (name) { audioFileName = basePath + "snd/" + ID + "-" + name->child("text")->content(); //store all the tags in the database XMLNode *tags = freesound->child("tags"); if (tags) { XMLNodeList children = tags->children(); XMLNodeConstIterator niter; std::vector strings; for (niter = children.begin(); niter != children.end(); ++niter) { XMLNode *node = *niter; if( strcmp( node->name().c_str(), "resource") == 0 ) { XMLNode *text = node->child("text"); if (text) { // std::cerr << "tag: " << text->content() << std::endl; strings.push_back(text->content()); } } } ARDOUR::Library->set_tags (std::string("//")+audioFileName, strings); ARDOUR::Library->save_changes (); } } return audioFileName; } int audioFileWrite(void *buffer, size_t size, size_t nmemb, void *file) { return (int)fwrite(buffer, size, nmemb, (FILE*) file); }; //------------------------------------------------------------------------ std::string Mootcher::getAudioFile(std::string originalFileName, std::string ID, std::string audioURL, SoundFileBrowser *caller) { ensureWorkingDir(); std::string audioFileName = basePath + "snd/" + ID + "-" + originalFileName; // check to see if audio file already exists FILE *testFile = g_fopen(audioFileName.c_str(), "r"); if (testFile) { fseek (testFile , 0 , SEEK_END); if (ftell (testFile) > 256) { fclose (testFile); return audioFileName; } // else file was small, probably an error, delete it and try again fclose(testFile); remove( audioFileName.c_str() ); } if (!curl) { return ""; } //if already canceling a previous download, bail out here ( this can happen b/c getAudioFile gets called by various UI update funcs ) if ( caller->freesound_download_cancel ) { return ""; } //now download the actual file FILE* theFile; theFile = g_fopen( audioFileName.c_str(), "wb" ); if (!theFile) { return ""; } // create the download url audioURL += "?api_key=" + api_key; setcUrlOptions(); curl_easy_setopt(curl, CURLOPT_URL, audioURL.c_str() ); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, audioFileWrite); curl_easy_setopt(curl, CURLOPT_WRITEDATA, theFile); /* hack to get rid of the barber-pole stripes */ caller->freesound_progress_bar.hide(); caller->freesound_progress_bar.show(); std::string prog; prog = string_compose (_("%1"), originalFileName); caller->freesound_progress_bar.set_text(prog); curl_easy_setopt (curl, CURLOPT_NOPROGRESS, 0); // turn on the progress bar curl_easy_setopt (curl, CURLOPT_PROGRESSFUNCTION, progress_callback); curl_easy_setopt (curl, CURLOPT_PROGRESSDATA, caller); CURLcode res = curl_easy_perform(curl); fclose(theFile); curl_easy_setopt (curl, CURLOPT_NOPROGRESS, 1); // turn off the progress bar caller->freesound_progress_bar.set_fraction(0.0); caller->freesound_progress_bar.set_text(""); if( res != 0 ) { error << string_compose (_("curl error %1 (%2)"), res, curl_easy_strerror(res)) << endmsg; remove( audioFileName.c_str() ); return ""; } else { // now download the tags &c. getSoundResourceFile(ID); } return audioFileName; } //--------- int Mootcher::progress_callback(void *caller, double dltotal, double dlnow, double /*ultotal*/, double /*ulnow*/) { SoundFileBrowser *sfb = (SoundFileBrowser *) caller; //XXX I hope it's OK to do GTK things in this callback. Otherwise // I'll have to do stuff like in interthread_progress_window. if (sfb->freesound_download_cancel) { return -1; } sfb->freesound_progress_bar.set_fraction(dlnow/dltotal); /* Make sure the progress widget gets updated */ while (Glib::MainContext::get_default()->iteration (false)) { /* do nothing */ } return 0; }