2022-05-10 10:06:48 +00:00
|
|
|
// Copyright (C) 2017 The Qt Company Ltd.
|
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
#include <QtCore>
|
|
|
|
#include <QtNetwork>
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
#include <cstdio>
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
class QSslError;
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
using namespace std;
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
class DownloadManager: public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
QNetworkAccessManager manager;
|
2020-06-22 08:12:38 +00:00
|
|
|
QList<QNetworkReply *> currentDownloads;
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
DownloadManager();
|
|
|
|
void doDownload(const QUrl &url);
|
2017-09-27 12:11:58 +00:00
|
|
|
static QString saveFileName(const QUrl &url);
|
2011-04-27 10:05:43 +00:00
|
|
|
bool saveToDisk(const QString &filename, QIODevice *data);
|
2017-09-27 12:11:58 +00:00
|
|
|
static bool isHttpRedirect(QNetworkReply *reply);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
public slots:
|
|
|
|
void execute();
|
|
|
|
void downloadFinished(QNetworkReply *reply);
|
|
|
|
void sslErrors(const QList<QSslError> &errors);
|
|
|
|
};
|
|
|
|
|
|
|
|
DownloadManager::DownloadManager()
|
|
|
|
{
|
2019-11-01 12:21:32 +00:00
|
|
|
connect(&manager, &QNetworkAccessManager::finished,
|
|
|
|
this, &DownloadManager::downloadFinished);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadManager::doDownload(const QUrl &url)
|
|
|
|
{
|
|
|
|
QNetworkRequest request(url);
|
|
|
|
QNetworkReply *reply = manager.get(request);
|
2012-05-18 13:42:00 +00:00
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
#if QT_CONFIG(ssl)
|
2019-11-01 12:21:32 +00:00
|
|
|
connect(reply, &QNetworkReply::sslErrors,
|
|
|
|
this, &DownloadManager::sslErrors);
|
2012-05-18 13:42:00 +00:00
|
|
|
#endif
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
currentDownloads.append(reply);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString DownloadManager::saveFileName(const QUrl &url)
|
|
|
|
{
|
|
|
|
QString path = url.path();
|
|
|
|
QString basename = QFileInfo(path).fileName();
|
|
|
|
|
|
|
|
if (basename.isEmpty())
|
|
|
|
basename = "download";
|
|
|
|
|
|
|
|
if (QFile::exists(basename)) {
|
|
|
|
// already exists, don't overwrite
|
|
|
|
int i = 0;
|
|
|
|
basename += '.';
|
|
|
|
while (QFile::exists(basename + QString::number(i)))
|
|
|
|
++i;
|
|
|
|
|
|
|
|
basename += QString::number(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
return basename;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DownloadManager::saveToDisk(const QString &filename, QIODevice *data)
|
|
|
|
{
|
|
|
|
QFile file(filename);
|
|
|
|
if (!file.open(QIODevice::WriteOnly)) {
|
|
|
|
fprintf(stderr, "Could not open %s for writing: %s\n",
|
|
|
|
qPrintable(filename),
|
|
|
|
qPrintable(file.errorString()));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
file.write(data->readAll());
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
bool DownloadManager::isHttpRedirect(QNetworkReply *reply)
|
|
|
|
{
|
|
|
|
int statusCode = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
return statusCode == 301 || statusCode == 302 || statusCode == 303
|
|
|
|
|| statusCode == 305 || statusCode == 307 || statusCode == 308;
|
|
|
|
}
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
void DownloadManager::execute()
|
|
|
|
{
|
|
|
|
QStringList args = QCoreApplication::instance()->arguments();
|
|
|
|
args.takeFirst(); // skip the first argument, which is the program's name
|
|
|
|
if (args.isEmpty()) {
|
|
|
|
printf("Qt Download example - downloads all URLs in parallel\n"
|
|
|
|
"Usage: download url1 [url2... urlN]\n"
|
|
|
|
"\n"
|
|
|
|
"Downloads the URLs passed in the command-line to the local directory\n"
|
|
|
|
"If the target file already exists, a .0, .1, .2, etc. is appended to\n"
|
|
|
|
"differentiate.\n");
|
|
|
|
QCoreApplication::instance()->quit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
for (const QString &arg : qAsConst(args)) {
|
2011-04-27 10:05:43 +00:00
|
|
|
QUrl url = QUrl::fromEncoded(arg.toLocal8Bit());
|
|
|
|
doDownload(url);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadManager::sslErrors(const QList<QSslError> &sslErrors)
|
|
|
|
{
|
2017-09-27 12:11:58 +00:00
|
|
|
#if QT_CONFIG(ssl)
|
|
|
|
for (const QSslError &error : sslErrors)
|
2011-04-27 10:05:43 +00:00
|
|
|
fprintf(stderr, "SSL error: %s\n", qPrintable(error.errorString()));
|
2012-05-18 13:08:56 +00:00
|
|
|
#else
|
|
|
|
Q_UNUSED(sslErrors);
|
2011-04-27 10:05:43 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadManager::downloadFinished(QNetworkReply *reply)
|
|
|
|
{
|
|
|
|
QUrl url = reply->url();
|
2020-02-10 09:29:57 +00:00
|
|
|
if (reply->error()) {
|
2011-04-27 10:05:43 +00:00
|
|
|
fprintf(stderr, "Download of %s failed: %s\n",
|
|
|
|
url.toEncoded().constData(),
|
|
|
|
qPrintable(reply->errorString()));
|
|
|
|
} else {
|
2017-09-27 12:11:58 +00:00
|
|
|
if (isHttpRedirect(reply)) {
|
|
|
|
fputs("Request was redirected.\n", stderr);
|
|
|
|
} else {
|
|
|
|
QString filename = saveFileName(url);
|
|
|
|
if (saveToDisk(filename, reply)) {
|
|
|
|
printf("Download of %s succeeded (saved to %s)\n",
|
|
|
|
url.toEncoded().constData(), qPrintable(filename));
|
|
|
|
}
|
|
|
|
}
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
currentDownloads.removeAll(reply);
|
|
|
|
reply->deleteLater();
|
|
|
|
|
2017-09-27 12:11:58 +00:00
|
|
|
if (currentDownloads.isEmpty()) {
|
2011-04-27 10:05:43 +00:00
|
|
|
// all downloads finished
|
|
|
|
QCoreApplication::instance()->quit();
|
2017-09-27 12:11:58 +00:00
|
|
|
}
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
QCoreApplication app(argc, argv);
|
|
|
|
|
|
|
|
DownloadManager manager;
|
|
|
|
QTimer::singleShot(0, &manager, SLOT(execute()));
|
|
|
|
|
|
|
|
app.exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "main.moc"
|