// rdaudioinfo.cpp // // Get information about a cut in the audio store. // // (C) Copyright 2011,2016-2018 Fred Gleason // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License version 2 as // published by the Free Software Foundation. // // 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., 675 Mass Ave, Cambridge, MA 02139, USA. // #include #include #include #include #include #include #include #include #include #include "rd.h" #include "rdapplication.h" #include "rdxport_interface.h" #include "rdformpost.h" #include "rdaudioinfo.h" size_t RDAudioInfoCallback(void *ptr,size_t size,size_t nmemb,void *userdata) { QString *xml=(QString *)userdata; for(unsigned i=0;i<(size*nmemb);i++) { *xml+=((const char *)ptr)[i]; } return size*nmemb; } RDAudioInfo::RDAudioInfo(QObject *parent) : QObject(parent) { conv_cart_number=0; conv_cut_number=0; conv_format=RDWaveFile::Pcm16; conv_channels=0; conv_sample_rate=0; conv_bit_rate=0; conv_frames=0; conv_length=0; } RDWaveFile::Format RDAudioInfo::format() const { return conv_format; } unsigned RDAudioInfo::channels() const { return conv_channels; } unsigned RDAudioInfo::sampleRate() const { return conv_sample_rate; } unsigned RDAudioInfo::bitRate() const { return conv_bit_rate; } unsigned RDAudioInfo::frames() const { return conv_frames; } unsigned RDAudioInfo::length() const { return conv_length; } void RDAudioInfo::setCartNumber(unsigned cartnum) { conv_cart_number=cartnum; } void RDAudioInfo::setCutNumber(unsigned cutnum) { conv_cut_number=cutnum; } RDAudioInfo::ErrorCode RDAudioInfo::runInfo(const QString &username, const QString &password) { long response_code; CURL *curl=NULL; char url[1024]; CURLcode curl_err; struct curl_httppost *first=NULL; struct curl_httppost *last=NULL; // // Generate POST Data // curl_formadd(&first,&last,CURLFORM_PTRNAME,"COMMAND", CURLFORM_COPYCONTENTS, (const char *)QString().sprintf("%u",RDXPORT_COMMAND_AUDIOINFO), CURLFORM_END); curl_formadd(&first,&last,CURLFORM_PTRNAME,"LOGIN_NAME", CURLFORM_COPYCONTENTS,(const char *)username.utf8(),CURLFORM_END); curl_formadd(&first,&last,CURLFORM_PTRNAME,"PASSWORD", CURLFORM_COPYCONTENTS,(const char *)password.utf8(),CURLFORM_END); curl_formadd(&first,&last,CURLFORM_PTRNAME,"CART_NUMBER", CURLFORM_COPYCONTENTS, (const char *)QString().sprintf("%u",conv_cart_number), CURLFORM_END); curl_formadd(&first,&last,CURLFORM_PTRNAME,"CUT_NUMBER", CURLFORM_COPYCONTENTS, (const char *)QString().sprintf("%u",conv_cut_number), CURLFORM_END); if((curl=curl_easy_init())==NULL) { curl_formfree(first); return RDAudioInfo::ErrorInternal; } // // Write out URL as a C string before passing to curl_easy_setopt(), // otherwise some versions of LibCurl will throw a 'bad/illegal format' // error. // strncpy(url,rda->station()->webServiceUrl(rda->config()),1024); curl_easy_setopt(curl,CURLOPT_URL,url); curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,RDAudioInfoCallback); curl_easy_setopt(curl,CURLOPT_WRITEDATA,&conv_xml); curl_easy_setopt(curl,CURLOPT_HTTPPOST,first); curl_easy_setopt(curl,CURLOPT_USERAGENT, (const char *)rda->config()->userAgent()); curl_easy_setopt(curl,CURLOPT_TIMEOUT,RD_CURL_TIMEOUT); switch(curl_err=curl_easy_perform(curl)) { case CURLE_OK: break; case CURLE_UNSUPPORTED_PROTOCOL: case CURLE_FAILED_INIT: case CURLE_COULDNT_RESOLVE_PROXY: case CURLE_PARTIAL_FILE: case CURLE_HTTP_RETURNED_ERROR: case CURLE_WRITE_ERROR: case CURLE_OUT_OF_MEMORY: case CURLE_OPERATION_TIMEDOUT: case CURLE_HTTP_POST_ERROR: curl_easy_cleanup(curl); curl_formfree(first); fprintf(stderr,"curl error: %d\n",curl_err); return RDAudioInfo::ErrorInternal; case CURLE_URL_MALFORMAT: case CURLE_COULDNT_RESOLVE_HOST: case CURLE_COULDNT_CONNECT: case 9: // CURLE_REMOTE_ACCESS_DENIED curl_easy_cleanup(curl); curl_formfree(first); return RDAudioInfo::ErrorUrlInvalid; default: curl_easy_cleanup(curl); curl_formfree(first); return RDAudioInfo::ErrorService; } curl_easy_getinfo(curl,CURLINFO_RESPONSE_CODE,&response_code); curl_easy_cleanup(curl); curl_formfree(first); switch(response_code) { case 200: break; case 404: return RDAudioInfo::ErrorNoAudio; default: return RDAudioInfo::ErrorService; } conv_format=(RDWaveFile::Format)ParseInt("format",conv_xml); conv_channels=ParseInt("channels",conv_xml); conv_sample_rate=ParseInt("sampleRate",conv_xml); conv_bit_rate=ParseInt("bitRate",conv_xml); conv_frames=ParseInt("frames",conv_xml); conv_length=ParseInt("length",conv_xml); return RDAudioInfo::ErrorOk; } QString RDAudioInfo::errorText(RDAudioInfo::ErrorCode err) { QString ret=QString().sprintf("Unknown RDAudioInfo Error [%u]",err); switch(err) { case RDAudioInfo::ErrorOk: ret=tr("OK"); break; case RDAudioInfo::ErrorInternal: ret=tr("Internal Error"); break; case RDAudioInfo::ErrorUrlInvalid: ret=tr("Invalid URL"); break; case RDAudioInfo::ErrorService: ret=tr("RDXport service returned an error"); break; case RDAudioInfo::ErrorInvalidUser: ret=tr("Invalid user or password"); break; case RDAudioInfo::ErrorNoAudio: ret=tr("Audio does not exist"); break; } return ret; } int RDAudioInfo::ParseInt(const QString &tag,const QString &xml) { // // FIXME: This is totally ad-hoc, but should work until we settle on // a proper XML parser. // QStringList list=xml.split("\n"); for(int i=0;i=2) { list2=list2[1].split(">"); if(list2.size()>=2) { return list2[1].toInt(); } } } } return -1; }