Rivendellaudio/lib/rdaudioimport.cpp
Fred Gleason ac4d9305bc 2019-06-25 Fred Gleason <fredg@paravelsystems.com>
* Adjusted to priority of 'operations' system events to use
	LOG_INFO priority.
2019-06-25 15:08:12 -04:00

323 lines
8.1 KiB
C++

// rdaudioimport.cpp
//
// Import an Audio File using the RdXport Web Service
//
// (C) Copyright 2010-2014,2016-2017 Fred Gleason <fredg@paravelsystems.com>
//
// 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 <stdlib.h>
#include <stdio.h>
#include <syslog.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <math.h>
#include <curl/curl.h>
#include <qapplication.h>
#include <rd.h>
#include <rdapplication.h>
#include <rdaudioimport.h>
#include <rdformpost.h>
#include <rdwebresult.h>
#include <rdxport_interface.h>
//
// CURL Callbacks
//
size_t ImportReadCallback(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;
}
int ImportProgressCallback(void *clientp,double dltotal,double dlnow,
double ultotal,double ulnow)
{
RDAudioImport *conv=(RDAudioImport *)clientp;
qApp->processEvents();
if(conv->aborting()) {
return 1;
}
return 0;
}
RDAudioImport::RDAudioImport(QObject *parent)
: QObject(parent)
{
conv_cart_number=0;
conv_cut_number=0;
conv_settings=NULL;
conv_use_metadata=false;
conv_aborting=false;
}
void RDAudioImport::setCartNumber(unsigned cartnum)
{
conv_cart_number=cartnum;
}
void RDAudioImport::setCutNumber(unsigned cutnum)
{
conv_cut_number=cutnum;
}
void RDAudioImport::setSourceFile(const QString &filename)
{
conv_src_filename=filename;
}
void RDAudioImport::setUseMetadata(bool state)
{
conv_use_metadata=state;
}
void RDAudioImport::setDestinationSettings(RDSettings *settings)
{
conv_settings=settings;
}
RDAudioImport::ErrorCode RDAudioImport::runImport(const QString &username,
const QString &password,
RDAudioConvert::ErrorCode *conv_err)
{
long response_code;
CURL *curl=NULL;
CURLcode curl_err;
struct curl_httppost *first=NULL;
struct curl_httppost *last=NULL;
char url[1024];
QString xml;
RDWebResult web_result;
//
// Generate POST Data
//
// We have to use multipart here because we have a file to send.
//
curl_formadd(&first,&last,CURLFORM_PTRNAME,"COMMAND",
CURLFORM_COPYCONTENTS,
(const char *)QString().sprintf("%u",RDXPORT_COMMAND_IMPORT),
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);
curl_formadd(&first,&last,CURLFORM_PTRNAME,"CHANNELS",
CURLFORM_COPYCONTENTS,
(const char *)QString().sprintf("%u",conv_settings->channels()),
CURLFORM_END);
curl_formadd(&first,&last,CURLFORM_PTRNAME,"NORMALIZATION_LEVEL",
CURLFORM_COPYCONTENTS,
(const char *)QString().sprintf("%d",conv_settings->
normalizationLevel()),
CURLFORM_END);
curl_formadd(&first,&last,CURLFORM_PTRNAME,"AUTOTRIM_LEVEL",
CURLFORM_COPYCONTENTS,
(const char *)QString().sprintf("%d",conv_settings->
autotrimLevel()),CURLFORM_END);
curl_formadd(&first,&last,CURLFORM_PTRNAME,"USE_METADATA",
CURLFORM_COPYCONTENTS,
(const char *)QString().sprintf("%u",conv_use_metadata),
CURLFORM_END);
curl_formadd(&first,&last,CURLFORM_PTRNAME,"FILENAME",
CURLFORM_FILE,(const char *)(conv_src_filename.utf8()),
CURLFORM_END);
//
// Set up the transfer
//
if((curl=curl_easy_init())==NULL) {
curl_formfree(first);
return RDAudioImport::ErrorInternal;
}
curl_easy_setopt(curl,CURLOPT_WRITEDATA,stdout);
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);
curl_easy_setopt(curl,CURLOPT_PROGRESSFUNCTION,ImportProgressCallback);
curl_easy_setopt(curl,CURLOPT_PROGRESSDATA,this);
curl_easy_setopt(curl,CURLOPT_NOPROGRESS,0);
curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,ImportReadCallback);
curl_easy_setopt(curl,CURLOPT_WRITEDATA,&xml);
//
// 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);
//
// Send it
//
switch(curl_err=curl_easy_perform(curl)) {
case CURLE_OK:
break;
case CURLE_ABORTED_BY_CALLBACK:
curl_easy_cleanup(curl);
curl_formfree(first);
return RDAudioImport::ErrorAborted;
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:
default:
curl_easy_cleanup(curl);
curl_formfree(first);
return RDAudioImport::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 RDAudioImport::ErrorUrlInvalid;
}
/*
rda->syslog(LOG_DEBUG,"CURL code: %d [%s]\n",curl_err,
curl_easy_strerror(curl_err));
*/
//
// Clean up
//
curl_easy_getinfo(curl,CURLINFO_RESPONSE_CODE,&response_code);
curl_easy_cleanup(curl);
curl_formfree(first);
//
// Process the results
//
if(web_result.readXml(xml)) {
*conv_err=web_result.converterErrorCode();
}
else {
*conv_err=RDAudioConvert::ErrorOk;
}
//printf("resp code: %d\n",response_code);
switch(response_code) {
case 200:
break;
case 400:
return RDAudioImport::ErrorService;
case 401:
return RDAudioImport::ErrorInvalidUser;
case 404:
return RDAudioImport::ErrorNoDestination;
default:
return RDAudioImport::ErrorConverter;
}
return RDAudioImport::ErrorOk;
}
bool RDAudioImport::aborting() const
{
return conv_aborting;
}
QString RDAudioImport::errorText(RDAudioImport::ErrorCode err,
RDAudioConvert::ErrorCode conv_err)
{
QString ret=QString().sprintf("Uknown Error [%u]",err);
switch(err) {
case RDAudioImport::ErrorOk:
ret=tr("OK");
break;
case RDAudioImport::ErrorInvalidSettings:
ret=tr("Invalid/unsupported audio parameters");
break;
case RDAudioImport::ErrorNoSource:
ret=tr("No such cart/cut");
break;
case RDAudioImport::ErrorNoDestination:
ret=tr("Unable to create destination file");
break;
case RDAudioImport::ErrorInternal:
ret=tr("Internal Error");
break;
case RDAudioImport::ErrorUrlInvalid:
ret=tr("Invalid URL");
break;
case RDAudioImport::ErrorService:
ret=tr("RDXport service returned an error");
break;
case RDAudioImport::ErrorInvalidUser:
ret=tr("Invalid user or password");
break;
case RDAudioImport::ErrorAborted:
ret=tr("Aborted");
break;
case RDAudioImport::ErrorConverter:
ret=tr("Audio Converter Error: ")+RDAudioConvert::errorText(conv_err);
break;
}
return ret;
}
void RDAudioImport::abort()
{
conv_aborting=true;
}