Rivendellaudio/lib/rddownload.cpp
Fred Gleason 929ab22613 2020-09-29 Fred Gleason <fredg@paravelsystems.com>
* Added the name of the Rivendell library component to 'Unknown Error'
	messages.

Signed-off-by: Fred Gleason <fredg@paravelsystems.com>
2020-09-29 20:46:07 -04:00

332 lines
7.4 KiB
C++

// rddownload.cpp
//
// Download a File
//
// (C) Copyright 2010-2020 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 <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fcntl.h>
#include <syslog.h>
#include <curl/curl.h>
#include <qapplication.h>
#include <qfileinfo.h>
#include <rd.h>
#include <rdapplication.h>
#include <rdsystemuser.h>
#include <rddownload.h>
//
// CURL Progress Callback
//
int DownloadProgressCallback(void *clientp,double dltotal,double dlnow,
double ultotal,double ulnow)
{
RDDownload *conv=(RDDownload *)clientp;
conv->UpdateProgress(ulnow);
qApp->processEvents();
if(conv->aborting()) {
return 1;
}
return 0;
}
int DownloadErrorCallback(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;
}
RDDownload::RDDownload(RDConfig *c,QObject *parent)
: RDTransfer(c,parent)
{
conv_aborting=false;
}
QStringList RDDownload::supportedSchemes() const
{
QStringList schemes;
schemes.push_back("file");
schemes.push_back("ftp");
schemes.push_back("ftps");
schemes.push_back("http");
schemes.push_back("https");
schemes.push_back("sftp");
return schemes;
}
void RDDownload::setSourceUrl(const QString &url)
{
conv_src_url=url;
}
void RDDownload::setDestinationFile(const QString &filename)
{
conv_dst_filename=filename;
QFileInfo fi(filename);
conv_dst_size=fi.size();
}
int RDDownload::totalSteps() const
{
return conv_dst_size;
}
RDDownload::ErrorCode RDDownload::runDownload(const QString &username,
const QString &password,
const QString &id_filename,
bool use_id_filename,
bool log_debug)
{
CURL *curl=NULL;
CURLcode curl_err;
FILE *f;
long response_code=0;
RDDownload::ErrorCode ret=RDDownload::ErrorOk;
RDSystemUser *user=NULL;
char userpwd[256];
if(!urlIsSupported(conv_src_url)) {
return RDDownload::ErrorUnsupportedProtocol;
}
//
// Validate User for file: transfers
//
if((getuid()==0)&&(conv_src_url.protocol().lower()=="file")) {
user=new RDSystemUser(username);
if(!user->validatePassword(password)) {
delete user;
return RDDownload::ErrorInvalidUser;
}
}
if((curl=curl_easy_init())==NULL) {
return RDDownload::ErrorInternal;
}
if((f=fopen(conv_dst_filename,"w"))==NULL) {
curl_easy_cleanup(curl);
return RDDownload::ErrorNoDestination;
}
//
// Write out an encoded URL
//
QByteArray url=conv_src_url.toEncoded(QUrl::RemoveUserInfo);
//
// An URL anchor element will never occur here, so treat any '#'
// characters as part of the path.
//
url.replace("#","%23");
//
// Authentication
//
if((conv_src_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,url.constData());
curl_easy_setopt(curl,CURLOPT_WRITEDATA,f);
curl_easy_setopt(curl,CURLOPT_TIMEOUT,RD_CURL_TIMEOUT);
curl_easy_setopt(curl,CURLOPT_FOLLOWLOCATION,1);
curl_easy_setopt(curl,CURLOPT_PROGRESSFUNCTION,DownloadProgressCallback);
curl_easy_setopt(curl,CURLOPT_PROGRESSDATA,this);
curl_easy_setopt(curl,CURLOPT_NOPROGRESS,0);
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,DownloadErrorCallback);
}
if(user!=NULL) {
setegid(user->gid());
seteuid(user->uid());
}
switch((curl_err=curl_easy_perform(curl))) {
case CURLE_OK:
if(conv_src_url.protocol().lower()=="http") {
curl_easy_getinfo(curl,CURLINFO_RESPONSE_CODE,&response_code);
if(response_code!=200) {
ret=RDDownload::ErrorUrlInvalid;
}
}
break;
case CURLE_UNSUPPORTED_PROTOCOL:
ret=RDDownload::ErrorUnsupportedProtocol;
break;
case CURLE_URL_MALFORMAT:
ret=RDDownload::ErrorUrlInvalid;
break;
case CURLE_COULDNT_RESOLVE_HOST:
ret=RDDownload::ErrorInvalidHostname;
break;
case CURLE_LOGIN_DENIED:
ret=RDDownload::ErrorInvalidLogin;
break;
case CURLE_COULDNT_CONNECT:
ret=RDDownload::ErrorRemoteConnection;
break;
case 9: // CURLE_REMOTE_ACCESS_DENIED
ret=RDDownload::ErrorRemoteAccess;
break;
default:
rda->syslog(LOG_ERR,"Unknown CURL Error [%d]: %s",
curl_err,curl_easy_strerror(curl_err));
ret=RDDownload::ErrorUnspecified;
}
if(user!=NULL) {
seteuid(getuid());
setegid(getgid());
delete user;
}
if((curl_err!=CURLE_OK)&&log_debug) {
rda->syslog(LOG_WARNING,"CURL download failed: url: %s username: %s",
(const char *)conv_src_url.toString(),
(const char *)username);
}
curl_easy_cleanup(curl);
fclose(f);
return ret;
}
bool RDDownload::aborting() const
{
return conv_aborting;
}
QString RDDownload::errorText(RDDownload::ErrorCode err)
{
QString ret=QString().sprintf("Unknown RDDownload Error [%u]",err);
switch(err) {
case RDDownload::ErrorOk:
ret=tr("OK");
break;
case RDDownload::ErrorUnsupportedProtocol:
ret=tr("Unsupported protocol");
break;
case RDDownload::ErrorNoSource:
ret=tr("Unable to access source file");
break;
case RDDownload::ErrorNoDestination:
ret=tr("Unable to create destination file");
break;
case RDDownload::ErrorInvalidHostname:
ret=tr("Unable to resolve hostname");
break;
case RDDownload::ErrorRemoteServer:
ret=tr("Remote server error");
break;
case RDDownload::ErrorUrlInvalid:
ret=tr("Invalid URL");
break;
case RDDownload::ErrorUnspecified:
ret=tr("Unspecified error");
break;
case RDDownload::ErrorInvalidUser:
ret=tr("Invalid User");
break;
case RDDownload::ErrorInternal:
ret=tr("Internal Error");
break;
case RDDownload::ErrorAborted:
ret=tr("Download aborted");
break;
case RDDownload::ErrorInvalidLogin:
ret=tr("Invalid username or password");
break;
case RDDownload::ErrorRemoteAccess:
ret=tr("Remote access denied");
break;
case RDDownload::ErrorRemoteConnection:
ret=tr("Couldn't connect to server");
break;
}
return ret;
}
void RDDownload::abort()
{
conv_aborting=true;
}
void RDDownload::UpdateProgress(int step)
{
emit progressChanged(step);
}