Added download resume support

This commit is contained in:
Alex Zolotarev 2010-12-08 04:57:20 +00:00 committed by Alex Zolotarev
parent 5e2eeab724
commit 21e37fcd9a
10 changed files with 138 additions and 31 deletions

View file

@ -19,9 +19,11 @@
}
- (void) dealloc;
- (string const &) Url;
- (std::string const &) Url;
- (BOOL) StartDownloadWithUrl: (char const *)originalUrl andFile: (char const *)file
andFinishFunc: (TDownloadFinishedFunction &)finishFunc andProgressFunc: (TDownloadProgressFunction &)progressFunc;
andFinishFunc: (TDownloadFinishedFunction &)finishFunc
andProgressFunc: (TDownloadProgressFunction &)progressFunc
andUseResume: (BOOL)resume;
// Added because release from manager doesn't destroy it immediately...
- (void) Cancel;
@end

View file

@ -21,7 +21,7 @@
- (void) dealloc
{
NSLog(@"~IPhoneDownload() for url: %s", m_url.c_str());
// NSLog(@"~IPhoneDownload() for url: %s", m_url.c_str());
if (m_connection)
{
[m_connection cancel];
@ -39,14 +39,15 @@
- (BOOL) StartDownloadWithUrl: (char const *)originalUrl andFile: (char const *)file
andFinishFunc: (TDownloadFinishedFunction &)finishFunc andProgressFunc: (TDownloadProgressFunction &)progressFunc
andUseResume: (BOOL)resume
{
m_finishObserver = finishFunc;
m_progressObserver = progressFunc;
// try to create file first
string tmpFile = file;
std::string tmpFile = file;
tmpFile += DOWNLOADING_FILE_EXTENSION;
m_file = fopen(tmpFile.c_str(), "wb");
m_file = fopen(tmpFile.c_str(), resume ? "ab" : "wb");
if (m_file == 0)
{
NSLog(@"Error opening %s file for download: %s", tmpFile.c_str(), strerror(errno));
@ -60,15 +61,22 @@
m_url = originalUrl;
// Create the request.
NSURLRequest * request = [NSURLRequest requestWithURL:[NSURL URLWithString: [NSString stringWithUTF8String:m_url.c_str()]]
NSMutableURLRequest * request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString: [NSString stringWithUTF8String:m_url.c_str()]]
cachePolicy:NSURLRequestReloadIgnoringLocalCacheData timeoutInterval:TIMEOUT_IN_SECONDS];
long long fileSize = ftello(m_file);
if (resume && fileSize > 0)
{
NSString * val = [[NSString alloc] initWithFormat: @"bytes=%qi-", fileSize];
[request addValue:val forHTTPHeaderField:@"Range"];
[val release];
}
// create the connection with the request and start loading the data
m_connection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
// [request release];
if (m_connection == 0)
{
NSLog(@"Can't create connection for url %s", originalUrl);
// notify observer about error and exit
NSLog(@"Can't create connection for url %s", originalUrl);
// notify observer about error and exit
if (m_finishObserver)
m_finishObserver(originalUrl, false);
return NO;
@ -82,11 +90,34 @@
// This method is called when the server has determined that it
// has enough information to create the NSURLResponse.
// It can be called multiple times, for example in the case of a
// redirect, so each time we reset the data.
// check if this is OK (not a 404 or the like)
if ([response respondsToSelector:@selector(statusCode)])
{
NSInteger statusCode = [(NSHTTPURLResponse *)response statusCode];
if (statusCode < 200 || statusCode > 299)
{
NSLog(@"Received HTTP error code %d, canceling download", statusCode);
// deleting file
fclose(m_file);
m_file = 0;
remove((m_requestedFileName + DOWNLOADING_FILE_EXTENSION).c_str());
// notify user
if (m_finishObserver)
m_finishObserver(m_url.c_str(), false);
// and selfdestruct...
GetDownloadManager().CancelDownload(m_url.c_str());
return;
}
}
fseeko(m_file, 0, SEEK_SET);
m_projectedFileSize = [response expectedContentLength];
// if server doesn't support resume, make sure we're downloading file from scratch
if (m_projectedFileSize < 0)
{
fclose(m_file);
m_file = fopen((m_requestedFileName + DOWNLOADING_FILE_EXTENSION).c_str(), "wb");
}
NSLog(@"Projected file size: %qi", m_projectedFileSize);
}
- (void) connection: (NSURLConnection *)connection didReceiveData: (NSData *)data

View file

@ -24,7 +24,7 @@ public:
}
virtual void DownloadFile(char const * url, char const * fileName,
TDownloadFinishedFunction finishFunc, TDownloadProgressFunction progressFunc)
TDownloadFinishedFunction finishFunc, TDownloadProgressFunction progressFunc, bool resume)
{
// check if download is already active
for (NSUInteger i = 0; i < [activeDownloads count]; ++i)
@ -38,7 +38,11 @@ public:
}
IPhoneDownload * download = [[IPhoneDownload alloc] init];
if ([download StartDownloadWithUrl:url andFile:fileName andFinishFunc:finishFunc andProgressFunc:progressFunc])
if ([download StartDownloadWithUrl:url
andFile:fileName
andFinishFunc:finishFunc
andProgressFunc:progressFunc
andUseResume:resume])
{
// save download in array to cancel it later if necessary
[activeDownloads addObject:download];

View file

@ -105,7 +105,7 @@ namespace mapinfo
UPDATE_FULL_URL,
(GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE)).c_str(),
boost::bind(&Storage::OnUpdateDownloadFinished, this, _1, _2),
TDownloadProgressFunction());
TDownloadProgressFunction(), false);
return true;
}
@ -268,7 +268,8 @@ namespace mapinfo
it->first.c_str(),
(GetPlatform().WritablePathForFile(mapinfo::FileNameFromUrl(it->first))).c_str(),
boost::bind(&Storage::OnMapDownloadFinished, this, _1, _2),
boost::bind(&Storage::OnMapDownloadProgress, this, _1, _2));
boost::bind(&Storage::OnMapDownloadProgress, this, _1, _2),
true); // enabled resume support by default
// notify GUI - new status for country, "Downloading"
if (m_observerChange)
m_observerChange(index);

View file

@ -8,7 +8,7 @@
/// Appended to all downloading files and removed after successful download
#define DOWNLOADING_FILE_EXTENSION ".downloading"
typedef std::pair<uint64_t, uint64_t> TDownloadProgress;
typedef std::pair<int64_t, int64_t> TDownloadProgress;
typedef boost::function<void (char const *, TDownloadProgress)> TDownloadProgressFunction;
typedef boost::function<void (char const *, bool)> TDownloadFinishedFunction;
@ -19,7 +19,8 @@ class DownloadManager
public:
virtual ~DownloadManager() {}
virtual void DownloadFile(char const * url, char const * fileName,
TDownloadFinishedFunction finish, TDownloadProgressFunction progress) = 0;
TDownloadFinishedFunction finish, TDownloadProgressFunction progress,
bool useResume = false) = 0;
/// @note Doesn't notifies clients on canceling!
virtual void CancelDownload(char const * url) = 0;
virtual void CancelAllDownloads() = 0;

View file

@ -73,7 +73,10 @@ struct DlObserver
void OnDownloadProgress(char const * url, TDownloadProgress progress)
{
m_progressUrl = url;
TEST_NOT_EQUAL( progress.second, -1, ("Your hosting doesn't support total file size", GetHostFromUrl(url)) );
if (progress.second < 0)
{
cerr << "Your hosting doesn't support total file size or invalid resume range was given for " << url << endl;
}
// for big file - cancel downloading after 40Kb were transferred
if (progress.first > 40 * 1024)
gMgr.CancelDownload(url);
@ -204,3 +207,62 @@ UNIT_TEST(LongDownloadCanceling)
FileWriter::DeleteFile(TEST_FILE_NAME1);
}
*/
UNIT_TEST(DownloadResume)
{
size_t const NUM = 1;
string const fileContent("aLeX");
{
FileWriter f(TEST_FILE_NAME1 DOWNLOADING_FILE_EXTENSION);
f.Write(fileContent.c_str(), fileContent.size());
}
DlObserver<NUM> observer1;
gMgr.DownloadFile(TEST_FILE_URL1, TEST_FILE_NAME1,
boost::bind(&DlObserver<NUM>::OnDownloadFinished, &observer1, _1, _2),
boost::bind(&DlObserver<NUM>::OnDownloadProgress, &observer1, _1, _2),
false);
WAIT_FOR_ASYNC_DOWNLOAD;
TEST_EQUAL( observer1.m_result[0], true, () );
DlObserver<NUM> observer2;
gMgr.DownloadFile(TEST_FILE_URL1, TEST_FILE_NAME2,
boost::bind(&DlObserver<NUM>::OnDownloadFinished, &observer2, _1, _2),
boost::bind(&DlObserver<NUM>::OnDownloadProgress, &observer2, _1, _2),
false);
WAIT_FOR_ASYNC_DOWNLOAD;
TEST_EQUAL( observer2.m_result[0], true, () );
uint64_t size1 = 4, size2 = 5;
TEST( GetPlatform().GetFileSize(TEST_FILE_NAME1, size1), ());
TEST( GetPlatform().GetFileSize(TEST_FILE_NAME2, size2), ());
TEST_EQUAL(size1, size2, ("Sizes should be equal - no resume was enabled"));
// resume should append content to 1st file
{
FileWriter f(TEST_FILE_NAME1 DOWNLOADING_FILE_EXTENSION);
f.Write(fileContent.c_str(), fileContent.size());
}
DlObserver<NUM> observer3;
gMgr.DownloadFile(TEST_FILE_URL1, TEST_FILE_NAME1,
boost::bind(&DlObserver<NUM>::OnDownloadFinished, &observer3, _1, _2),
boost::bind(&DlObserver<NUM>::OnDownloadProgress, &observer3, _1, _2),
true);
WAIT_FOR_ASYNC_DOWNLOAD;
TEST_EQUAL( observer3.m_result[0], true, () );
TEST( GetPlatform().GetFileSize(TEST_FILE_NAME1, size1), ());
TEST_EQUAL( size1, size2, () );
{
string str;
ReadFileToString(TEST_FILE_NAME1, str);
TEST_EQUAL(fileContent, str.substr(0, fileContent.size()), ());
TEST_NOT_EQUAL(fileContent, str, ());
}
FileWriter::DeleteFile(TEST_FILE_NAME1);
FileWriter::DeleteFile(TEST_FILE_NAME2);
}

View file

@ -5,7 +5,8 @@
#include "../base/assert.hpp"
QtDownload::QtDownload(QtDownloadManager & manager, char const * url,
char const * fileName, TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress)
char const * fileName, TDownloadFinishedFunction & finish,
TDownloadProgressFunction & progress, bool useResume)
: QObject(&manager), m_currentUrl(url), m_reply(0), m_file(0),
m_httpRequestAborted(false), m_finish(finish), m_progress(progress)
{
@ -13,11 +14,8 @@ QtDownload::QtDownload(QtDownloadManager & manager, char const * url,
QString tmpFileName(fileName);
tmpFileName += DOWNLOADING_FILE_EXTENSION;
// @TODO implement resume download
QFile::remove(tmpFileName);
m_file = new QFile(tmpFileName);
if (!m_file->open(QIODevice::WriteOnly))
if (!m_file->open(useResume ? QIODevice::Append : QIODevice::WriteOnly))
{
QString const err = m_file->errorString();
LOG(LERROR, ("Can't open file while downloading", qPrintable(tmpFileName), qPrintable(err)));
@ -50,16 +48,21 @@ QtDownload::~QtDownload()
}
void QtDownload::StartDownload(QtDownloadManager & manager, char const * url,
char const * fileName, TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress)
char const * fileName, TDownloadFinishedFunction & finish,
TDownloadProgressFunction & progress, bool useResume)
{
ASSERT(url && fileName, ());
// manager is responsible for auto deleting
new QtDownload(manager, url, fileName, finish, progress);
new QtDownload(manager, url, fileName, finish, progress, useResume);
}
void QtDownload::StartRequest()
{
m_reply = static_cast<QtDownloadManager *>(parent())->NetAccessManager().get(QNetworkRequest(m_currentUrl));
QNetworkRequest httpRequest(m_currentUrl);
qint64 fileSize = m_file->size();
if (fileSize > 0) // need resume
httpRequest.setRawHeader("Range", QString("bytes=%1-").arg(fileSize).toAscii());
m_reply = static_cast<QtDownloadManager *>(parent())->NetAccessManager().get(httpRequest);
connect(m_reply, SIGNAL(finished()), this, SLOT(OnHttpFinished()));
connect(m_reply, SIGNAL(readyRead()), this, SLOT(OnHttpReadyRead()));
connect(m_reply, SIGNAL(downloadProgress(qint64, qint64)),

View file

@ -19,7 +19,8 @@ private:
TDownloadProgressFunction m_progress;
QtDownload(QtDownloadManager & manager, char const * url, char const * fileName,
TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress);
TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress,
bool useResume);
void StartRequest();
private slots:
@ -31,6 +32,6 @@ public:
/// Created instance is automanaged as a manager's child
/// and can be manipulated later by manager->findChild(url);
static void StartDownload(QtDownloadManager & manager, char const * url, char const * fileName,
TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress);
TDownloadFinishedFunction & finish, TDownloadProgressFunction & progress, bool useResume);
virtual ~QtDownload();
};

View file

@ -4,11 +4,12 @@
#include "../base/assert.hpp"
void QtDownloadManager::DownloadFile(char const * url, char const * fileName,
TDownloadFinishedFunction finish, TDownloadProgressFunction progress)
TDownloadFinishedFunction finish, TDownloadProgressFunction progress,
bool useResume)
{
QList<QtDownload *> downloads = findChildren<QtDownload *>(url);
if (downloads.empty())
QtDownload::StartDownload(*this, url, fileName, finish, progress);
QtDownload::StartDownload(*this, url, fileName, finish, progress, useResume);
else
{
ASSERT(false, ("Download with the same url is already in progress!"));

View file

@ -10,7 +10,8 @@ class QtDownloadManager : public QObject, public DownloadManager
public:
virtual void DownloadFile(char const * url, char const * fileName,
TDownloadFinishedFunction finish, TDownloadProgressFunction progress);
TDownloadFinishedFunction finish, TDownloadProgressFunction progress,
bool useResume);
virtual void CancelDownload(char const * url);
virtual void CancelAllDownloads();