mirror of
https://github.com/ElvishArtisan/rivendell.git
synced 2025-04-07 01:13:50 +02:00
* Added the name of the Rivendell library component to 'Unknown Error' messages. Signed-off-by: Fred Gleason <fredg@paravelsystems.com>
273 lines
6.5 KiB
C++
273 lines
6.5 KiB
C++
// rddelete.cpp
|
|
//
|
|
// Delete a file from the audio store via the Rivendell Web Service
|
|
//
|
|
// (C) Copyright 2010,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.
|
|
//
|
|
|
|
#define _XOPEN_SOURCE
|
|
#include <stdio.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <syslog.h>
|
|
#include <pwd.h>
|
|
|
|
#include <curl/curl.h>
|
|
|
|
#include <qapplication.h>
|
|
#include <qfileinfo.h>
|
|
|
|
#include <rd.h>
|
|
#include <rdapplication.h>
|
|
#include <rdsystemuser.h>
|
|
#include <rddelete.h>
|
|
|
|
//
|
|
// CURL Callbackd
|
|
//
|
|
size_t DeleteWriteCallback(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 DeleteErrorCallback(CURL *curl,curl_infotype type,char *msg,size_t size,
|
|
void *clientp)
|
|
{
|
|
char str[1000];
|
|
|
|
if(type!=CURLINFO_TEXT) {
|
|
return 0;
|
|
}
|
|
if(size>999) {
|
|
size=999;
|
|
}
|
|
memset(&str,0,size+1);
|
|
memcpy(str,msg,size);
|
|
rda->syslog(LOG_DEBUG,"CURL MSG: %s",str);
|
|
return 0;
|
|
}
|
|
|
|
|
|
RDDelete::RDDelete(RDConfig *c,QObject *parent)
|
|
: RDTransfer(c,parent)
|
|
{
|
|
}
|
|
|
|
|
|
QStringList RDDelete::supportedSchemes() const
|
|
{
|
|
QStringList schemes;
|
|
|
|
schemes.push_back("file");
|
|
schemes.push_back("ftp");
|
|
schemes.push_back("sftp");
|
|
schemes.push_back("ftps");
|
|
|
|
return schemes;
|
|
}
|
|
|
|
|
|
void RDDelete::setTargetUrl(const QString &url)
|
|
{
|
|
conv_target_url=url;
|
|
}
|
|
|
|
|
|
RDDelete::ErrorCode RDDelete::runDelete(const QString &username,
|
|
const QString &password,
|
|
const QString &id_filename,
|
|
bool use_id_filename,
|
|
bool log_debug)
|
|
{
|
|
CURL *curl=NULL;
|
|
struct curl_slist *cmds=NULL;
|
|
CURLcode err;
|
|
RDDelete::ErrorCode ret=RDDelete::ErrorOk;
|
|
QString currentdir;
|
|
QString filename;
|
|
QString xml="";
|
|
char userpwd[256];
|
|
|
|
if(!urlIsSupported(conv_target_url)) {
|
|
return RDDelete::ErrorUnsupportedUrlScheme;
|
|
}
|
|
|
|
if(conv_target_url.scheme().toLower()=="file") {
|
|
unlink(conv_target_url.path().toUtf8().constData());
|
|
return RDDelete::ErrorOk;
|
|
}
|
|
|
|
if((curl=curl_easy_init())==NULL) {
|
|
rda->syslog(LOG_ERR,"unable to initialize curl library\n");
|
|
return RDDelete::ErrorInternal;
|
|
}
|
|
|
|
//
|
|
// Authentication
|
|
//
|
|
if((conv_target_url.scheme().toLower()=="sftp")&&
|
|
(!id_filename.isEmpty())&&use_id_filename) {
|
|
curl_easy_setopt(curl,CURLOPT_USERNAME,username.toUtf8().constData());
|
|
curl_easy_setopt(curl,CURLOPT_SSH_PRIVATE_KEYFILE,
|
|
id_filename.toUtf8().constData());
|
|
curl_easy_setopt(curl,CURLOPT_KEYPASSWD,password.toUtf8().constData());
|
|
}
|
|
else {
|
|
strncpy(userpwd,(username+":"+password).utf8(),256);
|
|
curl_easy_setopt(curl,CURLOPT_USERPWD,userpwd);
|
|
}
|
|
|
|
curl_easy_setopt(curl,CURLOPT_URL,conv_target_url.toEncoded().constData());
|
|
curl_easy_setopt(curl,CURLOPT_HTTPAUTH,CURLAUTH_ANY);
|
|
curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,DeleteWriteCallback);
|
|
curl_easy_setopt(curl,CURLOPT_WRITEDATA,&xml);
|
|
curl_easy_setopt(curl,CURLOPT_USERAGENT,
|
|
config()->userAgent().toUtf8().constData());
|
|
if(log_debug) {
|
|
curl_easy_setopt(curl,CURLOPT_VERBOSE,1);
|
|
curl_easy_setopt(curl,CURLOPT_DEBUGFUNCTION,DeleteErrorCallback);
|
|
}
|
|
|
|
if(conv_target_url.scheme().toLower()=="ftp"||conv_target_url.scheme().toLower()=="ftps") {
|
|
QStringList f0=conv_target_url.path().split("/",QString::SkipEmptyParts);
|
|
filename=f0.last();
|
|
f0.removeLast();
|
|
if(f0.size()>0) {
|
|
currentdir=f0.join("/")+"/";
|
|
}
|
|
if(!currentdir.isEmpty()) {
|
|
cmds=curl_slist_append(cmds,(QString("cwd ")+currentdir).toUtf8());
|
|
}
|
|
cmds=curl_slist_append(cmds,(QString("dele ")+filename).toUtf8());
|
|
}
|
|
|
|
if(conv_target_url.scheme().toLower()=="sftp") {
|
|
cmds=
|
|
curl_slist_append(cmds,(QString("rm ")+conv_target_url.path()).toUtf8());
|
|
}
|
|
|
|
curl_easy_setopt(curl,CURLOPT_QUOTE,cmds);
|
|
|
|
switch((err=curl_easy_perform(curl))) {
|
|
case CURLE_OK:
|
|
case CURLE_REMOTE_ACCESS_DENIED: // Sometimes we get this even when
|
|
// successful (?!)
|
|
case CURLE_QUOTE_ERROR: // In case the file is already gone
|
|
case CURLE_REMOTE_FILE_NOT_FOUND:
|
|
ret=RDDelete::ErrorOk;
|
|
break;
|
|
|
|
case CURLE_UNSUPPORTED_PROTOCOL:
|
|
ret=RDDelete::ErrorUnsupportedProtocol;
|
|
break;
|
|
|
|
case CURLE_URL_MALFORMAT:
|
|
ret=RDDelete::ErrorUrlInvalid;
|
|
break;
|
|
|
|
case CURLE_COULDNT_RESOLVE_HOST:
|
|
ret=RDDelete::ErrorInvalidHostname;
|
|
break;
|
|
|
|
case CURLE_LOGIN_DENIED:
|
|
ret=RDDelete::ErrorInvalidLogin;
|
|
break;
|
|
|
|
case CURLE_COULDNT_CONNECT:
|
|
ret=RDDelete::ErrorRemoteConnection;
|
|
break;
|
|
|
|
default:
|
|
ret=RDDelete::ErrorUnknown;
|
|
printf("CURL error: %d\n",err);
|
|
break;
|
|
}
|
|
if(log_debug) {
|
|
rda->syslog(LOG_INFO,curl_easy_strerror(err));
|
|
}
|
|
curl_slist_free_all(cmds);
|
|
curl_easy_cleanup(curl);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
QString RDDelete::errorText(RDDelete::ErrorCode err)
|
|
{
|
|
QString ret=QString().sprintf("Unknown RDDelete Error [%u]",err);
|
|
|
|
switch(err) {
|
|
case RDDelete::ErrorOk:
|
|
ret=tr("OK");
|
|
break;
|
|
|
|
case RDDelete::ErrorUnsupportedProtocol:
|
|
ret=tr("Unsupported protocol");
|
|
break;
|
|
|
|
case RDDelete::ErrorInvalidHostname:
|
|
ret=tr("Unable to resolve hostname");
|
|
break;
|
|
|
|
case RDDelete::ErrorRemoteServer:
|
|
ret=tr("Remote server error");
|
|
break;
|
|
|
|
case RDDelete::ErrorUrlInvalid:
|
|
ret=tr("Invalid URL");
|
|
break;
|
|
|
|
case RDDelete::ErrorUnspecified:
|
|
ret=tr("Unspecified error");
|
|
break;
|
|
|
|
case RDDelete::ErrorInvalidUser:
|
|
ret=tr("Invalid User");
|
|
break;
|
|
|
|
case RDDelete::ErrorInvalidLogin:
|
|
ret=tr("Invalid Login");
|
|
break;
|
|
|
|
case RDDelete::ErrorRemoteAccess:
|
|
ret=tr("Remote Access Denied");
|
|
break;
|
|
|
|
case RDDelete::ErrorRemoteConnection:
|
|
ret=tr("Counldn't Connect");
|
|
break;
|
|
|
|
case RDDelete::ErrorInternal:
|
|
ret=tr("Internal Error");
|
|
break;
|
|
|
|
case RDDelete::ErrorUnknown:
|
|
break;
|
|
|
|
case RDDelete::ErrorUnsupportedUrlScheme:
|
|
ret=tr("Unsupported URL Scheme");
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|