mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-14 23:11:06 +00:00
1442 lines
44 KiB
C++
1442 lines
44 KiB
C++
/****************************************************************************
|
||
Copyright (c) 2014 cocos2d-x.org
|
||
Copyright (c) 2015-2016 Chukong Technologies Inc.
|
||
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
||
|
||
http://www.cocos2d-x.org
|
||
|
||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
of this software and associated documentation files (the "Software"), to deal
|
||
in the Software without restriction, including without limitation the rights
|
||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
copies of the Software, and to permit persons to whom the Software is
|
||
furnished to do so, subject to the following conditions:
|
||
|
||
The above copyright notice and this permission notice shall be included in
|
||
all copies or substantial portions of the Software.
|
||
|
||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
THE SOFTWARE.
|
||
****************************************************************************/
|
||
#include "AssetsManagerEx.h"
|
||
#include "base/ccUTF8.h"
|
||
#include "CCAsyncTaskPool.h"
|
||
|
||
#include <stdio.h>
|
||
#include <errno.h>
|
||
|
||
#ifdef MINIZIP_FROM_SYSTEM
|
||
#include <minizip/unzip.h>
|
||
#else // from our embedded sources
|
||
#include "unzip/unzip.h"
|
||
#endif
|
||
|
||
NS_CC_EXT_BEGIN
|
||
|
||
#define VERSION_FILENAME "version.manifest"
|
||
#define TEMP_MANIFEST_FILENAME "project.manifest.temp"
|
||
#define TEMP_PACKAGE_SUFFIX "_temp"
|
||
#define MANIFEST_FILENAME "project.manifest"
|
||
|
||
#define BUFFER_SIZE 8192
|
||
#define MAX_FILENAME 512
|
||
|
||
#define DEFAULT_CONNECTION_TIMEOUT 45
|
||
|
||
#define SAVE_POINT_INTERVAL 0.1
|
||
|
||
const std::string AssetsManagerEx::VERSION_ID = "@version";
|
||
const std::string AssetsManagerEx::MANIFEST_ID = "@manifest";
|
||
|
||
// Implementation of AssetsManagerEx
|
||
|
||
AssetsManagerEx::AssetsManagerEx(const std::string& manifestUrl, const std::string& storagePath)
|
||
: _updateState(State::UNINITED)
|
||
, _assets(nullptr)
|
||
, _storagePath("")
|
||
, _tempVersionPath("")
|
||
, _cacheManifestPath("")
|
||
, _tempManifestPath("")
|
||
, _localManifest(nullptr)
|
||
, _tempManifest(nullptr)
|
||
, _remoteManifest(nullptr)
|
||
, _updateEntry(UpdateEntry::NONE)
|
||
, _percent(0)
|
||
, _percentByFile(0)
|
||
, _totalSize(0)
|
||
, _sizeCollected(0)
|
||
, _totalDownloaded(0)
|
||
, _totalToDownload(0)
|
||
, _totalWaitToDownload(0)
|
||
, _nextSavePoint(0.0)
|
||
, _downloadResumed(false)
|
||
, _maxConcurrentTask(32)
|
||
, _currConcurrentTask(0)
|
||
, _verifyCallback(nullptr)
|
||
, _inited(false)
|
||
, _canceled(false)
|
||
{
|
||
init(manifestUrl, storagePath);
|
||
}
|
||
|
||
AssetsManagerEx::AssetsManagerEx(const std::string& manifestUrl, const std::string& storagePath, const VersionCompareHandle& handle)
|
||
: _updateState(State::UNINITED)
|
||
, _assets(nullptr)
|
||
, _storagePath("")
|
||
, _tempVersionPath("")
|
||
, _cacheManifestPath("")
|
||
, _tempManifestPath("")
|
||
, _localManifest(nullptr)
|
||
, _tempManifest(nullptr)
|
||
, _remoteManifest(nullptr)
|
||
, _updateEntry(UpdateEntry::NONE)
|
||
, _percent(0)
|
||
, _percentByFile(0)
|
||
, _totalSize(0)
|
||
, _sizeCollected(0)
|
||
, _totalDownloaded(0)
|
||
, _totalToDownload(0)
|
||
, _totalWaitToDownload(0)
|
||
, _nextSavePoint(0.0)
|
||
, _downloadResumed(false)
|
||
, _maxConcurrentTask(32)
|
||
, _currConcurrentTask(0)
|
||
, _versionCompareHandle(handle)
|
||
, _verifyCallback(nullptr)
|
||
, _eventCallback(nullptr)
|
||
, _inited(false)
|
||
{
|
||
init(manifestUrl, storagePath);
|
||
}
|
||
|
||
void AssetsManagerEx::init(const std::string& manifestUrl, const std::string& storagePath)
|
||
{
|
||
// Init variables
|
||
std::string pointer = StringUtils::format("%p", this);
|
||
_eventName = "__cc_assets_manager_" + pointer;
|
||
_fileUtils = FileUtils::getInstance();
|
||
|
||
network::DownloaderHints hints =
|
||
{
|
||
static_cast<uint32_t>(_maxConcurrentTask),
|
||
DEFAULT_CONNECTION_TIMEOUT,
|
||
".tmp"
|
||
};
|
||
_downloader = std::shared_ptr<network::Downloader>(new network::Downloader(hints));
|
||
_downloader->onTaskError = std::bind(&AssetsManagerEx::onError, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4);
|
||
_downloader->onTaskProgress = [this](const network::DownloadTask& task,
|
||
int64_t /*bytesReceived*/,
|
||
int64_t totalBytesReceived,
|
||
int64_t totalBytesExpected)
|
||
{
|
||
this->onProgress(totalBytesExpected, totalBytesReceived, task.requestURL, task.identifier);
|
||
};
|
||
_downloader->onFileTaskSuccess = [this](const network::DownloadTask& task)
|
||
{
|
||
this->onSuccess(task.requestURL, task.storagePath, task.identifier);
|
||
};
|
||
setStoragePath(storagePath);
|
||
_tempVersionPath = _tempStoragePath + VERSION_FILENAME;
|
||
_cacheManifestPath = _storagePath + MANIFEST_FILENAME;
|
||
_tempManifestPath = _tempStoragePath + TEMP_MANIFEST_FILENAME;
|
||
|
||
if (manifestUrl.size() > 0)
|
||
{
|
||
loadLocalManifest(manifestUrl);
|
||
}
|
||
}
|
||
|
||
AssetsManagerEx::~AssetsManagerEx()
|
||
{
|
||
_downloader->onTaskError = (nullptr);
|
||
_downloader->onFileTaskSuccess = (nullptr);
|
||
_downloader->onTaskProgress = (nullptr);
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
// _tempManifest could share a ptr with _remoteManifest or _localManifest
|
||
if (_tempManifest != _localManifest && _tempManifest != _remoteManifest)
|
||
CC_SAFE_RELEASE(_tempManifest);
|
||
CC_SAFE_RELEASE(_remoteManifest);
|
||
}
|
||
|
||
AssetsManagerEx* AssetsManagerEx::create(const std::string& manifestUrl, const std::string& storagePath)
|
||
{
|
||
AssetsManagerEx* ret = new (std::nothrow) AssetsManagerEx(manifestUrl, storagePath);
|
||
if (ret)
|
||
{
|
||
ret->autorelease();
|
||
}
|
||
else
|
||
{
|
||
CC_SAFE_DELETE(ret);
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
void AssetsManagerEx::initManifests()
|
||
{
|
||
_inited = true;
|
||
_canceled = false;
|
||
// Init and load temporary manifest
|
||
_tempManifest = new (std::nothrow) Manifest();
|
||
if (_tempManifest)
|
||
{
|
||
_tempManifest->parseFile(_tempManifestPath);
|
||
// Previous update is interrupted
|
||
if (_fileUtils->isFileExist(_tempManifestPath))
|
||
{
|
||
// Manifest parse failed, remove all temp files
|
||
if (!_tempManifest->isLoaded())
|
||
{
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
CC_SAFE_RELEASE(_tempManifest);
|
||
_tempManifest = nullptr;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
_inited = false;
|
||
}
|
||
|
||
// Init remote manifest for future usage
|
||
_remoteManifest = new (std::nothrow) Manifest();
|
||
if (!_remoteManifest)
|
||
{
|
||
_inited = false;
|
||
}
|
||
|
||
if (!_inited)
|
||
{
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
CC_SAFE_RELEASE(_tempManifest);
|
||
CC_SAFE_RELEASE(_remoteManifest);
|
||
_localManifest = nullptr;
|
||
_tempManifest = nullptr;
|
||
_remoteManifest = nullptr;
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::prepareLocalManifest()
|
||
{
|
||
// An alias to assets
|
||
_assets = &(_localManifest->getAssets());
|
||
|
||
// Add search paths
|
||
_localManifest->prependSearchPaths();
|
||
}
|
||
|
||
bool AssetsManagerEx::loadLocalManifest(Manifest* localManifest, const std::string& storagePath)
|
||
{
|
||
if (_updateState > State::UNINITED)
|
||
{
|
||
return false;
|
||
}
|
||
if (!localManifest || !localManifest->isLoaded())
|
||
{
|
||
return false;
|
||
}
|
||
_inited = true;
|
||
_canceled = false;
|
||
// Reset storage path
|
||
if (storagePath.size() > 0)
|
||
{
|
||
setStoragePath(storagePath);
|
||
_tempVersionPath = _tempStoragePath + VERSION_FILENAME;
|
||
_cacheManifestPath = _storagePath + MANIFEST_FILENAME;
|
||
_tempManifestPath = _tempStoragePath + TEMP_MANIFEST_FILENAME;
|
||
}
|
||
// Release existing local manifest
|
||
if (_localManifest)
|
||
{
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
}
|
||
_localManifest = localManifest;
|
||
_localManifest->retain();
|
||
// Find the cached manifest file
|
||
Manifest *cachedManifest = nullptr;
|
||
if (_fileUtils->isFileExist(_cacheManifestPath))
|
||
{
|
||
cachedManifest = new (std::nothrow) Manifest();
|
||
if (cachedManifest)
|
||
{
|
||
cachedManifest->parseFile(_cacheManifestPath);
|
||
if (!cachedManifest->isLoaded())
|
||
{
|
||
_fileUtils->removeFile(_cacheManifestPath);
|
||
CC_SAFE_RELEASE(cachedManifest);
|
||
cachedManifest = nullptr;
|
||
}
|
||
}
|
||
}
|
||
// Compare with cached manifest to determine which one to use
|
||
if (cachedManifest)
|
||
{
|
||
bool localNewer = _localManifest->versionGreater(cachedManifest, _versionCompareHandle);
|
||
if (localNewer)
|
||
{
|
||
// Recreate storage, to empty the content
|
||
_fileUtils->removeDirectory(_storagePath);
|
||
_fileUtils->createDirectory(_storagePath);
|
||
CC_SAFE_RELEASE(cachedManifest);
|
||
}
|
||
else
|
||
{
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
_localManifest = cachedManifest;
|
||
}
|
||
}
|
||
prepareLocalManifest();
|
||
|
||
// Init temp manifest and remote manifest
|
||
initManifests();
|
||
|
||
if (!_inited)
|
||
{
|
||
return false;
|
||
}
|
||
else
|
||
{
|
||
_updateState = State::UNCHECKED;
|
||
return true;
|
||
}
|
||
}
|
||
|
||
bool AssetsManagerEx::loadLocalManifest(const std::string& manifestUrl)
|
||
{
|
||
if (manifestUrl.size() == 0)
|
||
{
|
||
return false;
|
||
}
|
||
if (_updateState > State::UNINITED)
|
||
{
|
||
return false;
|
||
}
|
||
_manifestUrl = manifestUrl;
|
||
// Init and load local manifest
|
||
_localManifest = new (std::nothrow) Manifest();
|
||
if (!_localManifest)
|
||
{
|
||
return false;
|
||
}
|
||
Manifest *cachedManifest = nullptr;
|
||
// Find the cached manifest file
|
||
if (_fileUtils->isFileExist(_cacheManifestPath))
|
||
{
|
||
cachedManifest = new (std::nothrow) Manifest();
|
||
if (cachedManifest)
|
||
{
|
||
cachedManifest->parseFile(_cacheManifestPath);
|
||
if (!cachedManifest->isLoaded())
|
||
{
|
||
_fileUtils->removeFile(_cacheManifestPath);
|
||
CC_SAFE_RELEASE(cachedManifest);
|
||
cachedManifest = nullptr;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Ensure no search path of cached manifest is used to load this manifest
|
||
std::vector<std::string> searchPaths = _fileUtils->getSearchPaths();
|
||
if (cachedManifest)
|
||
{
|
||
std::vector<std::string> cacheSearchPaths = cachedManifest->getSearchPaths();
|
||
std::vector<std::string> trimmedPaths = searchPaths;
|
||
for (auto path : cacheSearchPaths)
|
||
{
|
||
const auto pos = std::find(trimmedPaths.begin(), trimmedPaths.end(), path);
|
||
if (pos != trimmedPaths.end())
|
||
{
|
||
trimmedPaths.erase(pos);
|
||
}
|
||
}
|
||
_fileUtils->setSearchPaths(trimmedPaths);
|
||
}
|
||
// Load local manifest in app package
|
||
_localManifest->parseFile(_manifestUrl);
|
||
if (cachedManifest)
|
||
{
|
||
// Restore search paths
|
||
_fileUtils->setSearchPaths(searchPaths);
|
||
}
|
||
if (_localManifest->isLoaded())
|
||
{
|
||
// Compare with cached manifest to determine which one to use
|
||
if (cachedManifest)
|
||
{
|
||
bool localNewer = _localManifest->versionGreater(cachedManifest, _versionCompareHandle);
|
||
if (localNewer)
|
||
{
|
||
// Recreate storage, to empty the content
|
||
_fileUtils->removeDirectory(_storagePath);
|
||
_fileUtils->createDirectory(_storagePath);
|
||
CC_SAFE_RELEASE(cachedManifest);
|
||
}
|
||
else
|
||
{
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
_localManifest = cachedManifest;
|
||
}
|
||
}
|
||
prepareLocalManifest();
|
||
}
|
||
|
||
// Fail to load local manifest
|
||
if (!_localManifest->isLoaded())
|
||
{
|
||
CCLOG("AssetsManagerEx : No local manifest file found error.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return false;
|
||
}
|
||
initManifests();
|
||
_updateState = State::UNCHECKED;
|
||
return true;
|
||
}
|
||
|
||
bool AssetsManagerEx::loadRemoteManifest(Manifest* remoteManifest)
|
||
{
|
||
if (!_inited || _updateState > State::UNCHECKED)
|
||
{
|
||
return false;
|
||
}
|
||
if (!remoteManifest || !remoteManifest->isLoaded())
|
||
{
|
||
return false;
|
||
}
|
||
// Release existing remote manifest
|
||
if (_remoteManifest)
|
||
{
|
||
CC_SAFE_RELEASE(_remoteManifest);
|
||
}
|
||
_remoteManifest = remoteManifest;
|
||
_remoteManifest->retain();
|
||
// Compare manifest version and set state
|
||
if (_localManifest->versionGreaterOrEquals(_remoteManifest, _versionCompareHandle))
|
||
{
|
||
_updateState = State::UP_TO_DATE;
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ALREADY_UP_TO_DATE);
|
||
}
|
||
else
|
||
{
|
||
_updateState = State::NEED_UPDATE;
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::NEW_VERSION_FOUND);
|
||
}
|
||
return true;
|
||
}
|
||
|
||
std::string AssetsManagerEx::basename(const std::string& path) const
|
||
{
|
||
size_t found = path.find_last_of("/\\");
|
||
|
||
if (std::string::npos != found)
|
||
{
|
||
return path.substr(0, found);
|
||
}
|
||
else
|
||
{
|
||
return path;
|
||
}
|
||
}
|
||
|
||
std::string AssetsManagerEx::get(const std::string& key) const
|
||
{
|
||
auto it = _assets->find(key);
|
||
if (it != _assets->cend()) {
|
||
return _storagePath + it->second.path;
|
||
}
|
||
else return "";
|
||
}
|
||
|
||
const Manifest* AssetsManagerEx::getLocalManifest() const
|
||
{
|
||
return _localManifest;
|
||
}
|
||
|
||
const Manifest* AssetsManagerEx::getRemoteManifest() const
|
||
{
|
||
return _remoteManifest;
|
||
}
|
||
|
||
const std::string& AssetsManagerEx::getStoragePath() const
|
||
{
|
||
return _storagePath;
|
||
}
|
||
|
||
void AssetsManagerEx::setStoragePath(const std::string& storagePath)
|
||
{
|
||
_storagePath = storagePath;
|
||
adjustPath(_storagePath);
|
||
_fileUtils->createDirectory(_storagePath);
|
||
|
||
_tempStoragePath = _storagePath;
|
||
_tempStoragePath.insert(_storagePath.size() - 1, TEMP_PACKAGE_SUFFIX);
|
||
_fileUtils->createDirectory(_tempStoragePath);
|
||
}
|
||
|
||
void AssetsManagerEx::adjustPath(std::string &path)
|
||
{
|
||
if (path.size() > 0 && path[path.size() - 1] != '/')
|
||
{
|
||
path.append("/");
|
||
}
|
||
}
|
||
|
||
bool AssetsManagerEx::decompress(const std::string &zip)
|
||
{
|
||
// Find root path for zip file
|
||
size_t pos = zip.find_last_of("/\\");
|
||
if (pos == std::string::npos)
|
||
{
|
||
CCLOG("AssetsManagerEx : no root path specified for zip file %s\n", zip.c_str());
|
||
return false;
|
||
}
|
||
const std::string rootPath = zip.substr(0, pos+1);
|
||
|
||
// Open the zip file
|
||
unzFile zipfile = unzOpen(FileUtils::getInstance()->getSuitableFOpen(zip).c_str());
|
||
if (! zipfile)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not open downloaded zip file %s\n", zip.c_str());
|
||
return false;
|
||
}
|
||
|
||
// Get info about the zip file
|
||
unz_global_info global_info;
|
||
if (unzGetGlobalInfo(zipfile, &global_info) != UNZ_OK)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not read file global info of %s\n", zip.c_str());
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
|
||
// Buffer to hold data read from the zip file
|
||
char readBuffer[BUFFER_SIZE];
|
||
// Loop to extract all files.
|
||
uLong i;
|
||
for (i = 0; i < global_info.number_entry; ++i)
|
||
{
|
||
// Get info about current file.
|
||
unz_file_info fileInfo;
|
||
char fileName[MAX_FILENAME];
|
||
if (unzGetCurrentFileInfo(zipfile,
|
||
&fileInfo,
|
||
fileName,
|
||
MAX_FILENAME,
|
||
NULL,
|
||
0,
|
||
NULL,
|
||
0) != UNZ_OK)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not read compressed file info\n");
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
const std::string fullPath = rootPath + fileName;
|
||
|
||
// Check if this entry is a directory or a file.
|
||
const size_t filenameLength = strlen(fileName);
|
||
if (fileName[filenameLength-1] == '/')
|
||
{
|
||
//There are not directory entry in some case.
|
||
//So we need to create directory when decompressing file entry
|
||
if ( !_fileUtils->createDirectory(basename(fullPath)) )
|
||
{
|
||
// Failed to create directory
|
||
CCLOG("AssetsManagerEx : can not create directory %s\n", fullPath.c_str());
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// Create all directories in advance to avoid issue
|
||
std::string dir = basename(fullPath);
|
||
if (!_fileUtils->isDirectoryExist(dir)) {
|
||
if (!_fileUtils->createDirectory(dir)) {
|
||
// Failed to create directory
|
||
CCLOG("AssetsManagerEx : can not create directory %s\n", fullPath.c_str());
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
}
|
||
// Entry is a file, so extract it.
|
||
// Open current file.
|
||
if (unzOpenCurrentFile(zipfile) != UNZ_OK)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not extract file %s\n", fileName);
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
|
||
// Create a file to store current file.
|
||
FILE *out = fopen(FileUtils::getInstance()->getSuitableFOpen(fullPath).c_str(), "wb");
|
||
if (!out)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not create decompress destination file %s (errno: %d)\n", fullPath.c_str(), errno);
|
||
unzCloseCurrentFile(zipfile);
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
|
||
// Write current file content to destinate file.
|
||
int error = UNZ_OK;
|
||
do
|
||
{
|
||
error = unzReadCurrentFile(zipfile, readBuffer, BUFFER_SIZE);
|
||
if (error < 0)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not read zip file %s, error code is %d\n", fileName, error);
|
||
fclose(out);
|
||
unzCloseCurrentFile(zipfile);
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
|
||
if (error > 0)
|
||
{
|
||
fwrite(readBuffer, error, 1, out);
|
||
}
|
||
} while(error > 0);
|
||
|
||
fclose(out);
|
||
}
|
||
|
||
unzCloseCurrentFile(zipfile);
|
||
|
||
// Goto next entry listed in the zip file.
|
||
if ((i+1) < global_info.number_entry)
|
||
{
|
||
if (unzGoToNextFile(zipfile) != UNZ_OK)
|
||
{
|
||
CCLOG("AssetsManagerEx : can not read next file for decompressing\n");
|
||
unzClose(zipfile);
|
||
return false;
|
||
}
|
||
}
|
||
}
|
||
|
||
unzClose(zipfile);
|
||
return true;
|
||
}
|
||
|
||
void AssetsManagerEx::decompressDownloadedZip(const std::string &customId, const std::string &storagePath)
|
||
{
|
||
struct AsyncData
|
||
{
|
||
std::string customId;
|
||
std::string zipFile;
|
||
bool succeed;
|
||
};
|
||
|
||
AsyncData* asyncData = new AsyncData;
|
||
asyncData->customId = customId;
|
||
asyncData->zipFile = storagePath;
|
||
asyncData->succeed = false;
|
||
|
||
std::function<void(void*)> decompressFinished = [this](void* param) {
|
||
auto dataInner = reinterpret_cast<AsyncData*>(param);
|
||
if (dataInner->succeed)
|
||
{
|
||
fileSuccess(dataInner->customId, dataInner->zipFile);
|
||
}
|
||
else
|
||
{
|
||
std::string errorMsg = "Unable to decompress file " + dataInner->zipFile;
|
||
// Ensure zip file deletion (if decompress failure cause task thread exit anormally)
|
||
_fileUtils->removeFile(dataInner->zipFile);
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_DECOMPRESS, "", errorMsg);
|
||
fileError(dataInner->customId, errorMsg);
|
||
}
|
||
delete dataInner;
|
||
};
|
||
AsyncTaskPool::getInstance()->enqueue(AsyncTaskPool::TaskType::TASK_OTHER, decompressFinished, (void*)asyncData, [this, asyncData]() {
|
||
// Decompress all compressed files
|
||
if (decompress(asyncData->zipFile))
|
||
{
|
||
asyncData->succeed = true;
|
||
}
|
||
_fileUtils->removeFile(asyncData->zipFile);
|
||
});
|
||
}
|
||
|
||
void AssetsManagerEx::dispatchUpdateEvent(EventAssetsManagerEx::EventCode code, const std::string &assetId/* = ""*/, const std::string &message/* = ""*/, int curle_code/* = CURLE_OK*/, int curlm_code/* = CURLM_OK*/)
|
||
{
|
||
switch (code)
|
||
{
|
||
case EventAssetsManagerEx::EventCode::ERROR_UPDATING:
|
||
case EventAssetsManagerEx::EventCode::ERROR_PARSE_MANIFEST:
|
||
case EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST:
|
||
case EventAssetsManagerEx::EventCode::ERROR_DECOMPRESS:
|
||
case EventAssetsManagerEx::EventCode::ERROR_DOWNLOAD_MANIFEST:
|
||
case EventAssetsManagerEx::EventCode::UPDATE_FAILED:
|
||
case EventAssetsManagerEx::EventCode::UPDATE_FINISHED:
|
||
case EventAssetsManagerEx::EventCode::ALREADY_UP_TO_DATE:
|
||
_updateEntry = UpdateEntry::NONE;
|
||
break;
|
||
case EventAssetsManagerEx::EventCode::UPDATE_PROGRESSION:
|
||
break;
|
||
case EventAssetsManagerEx::EventCode::ASSET_UPDATED:
|
||
break;
|
||
case EventAssetsManagerEx::EventCode::NEW_VERSION_FOUND:
|
||
if (_updateEntry == UpdateEntry::CHECK_UPDATE)
|
||
{
|
||
_updateEntry = UpdateEntry::NONE;
|
||
}
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
if (_eventCallback != nullptr) {
|
||
EventAssetsManagerEx* event = new (std::nothrow) EventAssetsManagerEx(_eventName, this, code, assetId, message, curle_code, curlm_code);
|
||
_eventCallback(event);
|
||
event->release();
|
||
}
|
||
}
|
||
|
||
AssetsManagerEx::State AssetsManagerEx::getState() const
|
||
{
|
||
return _updateState;
|
||
}
|
||
|
||
void AssetsManagerEx::downloadVersion()
|
||
{
|
||
if (_updateState > State::PREDOWNLOAD_VERSION)
|
||
return;
|
||
|
||
std::string versionUrl = _localManifest->getVersionFileUrl();
|
||
|
||
if (versionUrl.size() > 0)
|
||
{
|
||
_updateState = State::DOWNLOADING_VERSION;
|
||
// Download version file asynchronously
|
||
_downloader->createDownloadFileTask(versionUrl, _tempVersionPath, VERSION_ID);
|
||
}
|
||
// No version file found
|
||
else
|
||
{
|
||
CCLOG("AssetsManagerEx : No version file found, step skipped\n");
|
||
_updateState = State::PREDOWNLOAD_MANIFEST;
|
||
downloadManifest();
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::parseVersion()
|
||
{
|
||
if (_updateState != State::VERSION_LOADED)
|
||
return;
|
||
|
||
_remoteManifest->parseVersion(_tempVersionPath);
|
||
|
||
if (!_remoteManifest->isVersionLoaded())
|
||
{
|
||
CCLOG("AssetsManagerEx : Fail to parse version file, step skipped\n");
|
||
_updateState = State::PREDOWNLOAD_MANIFEST;
|
||
downloadManifest();
|
||
}
|
||
else
|
||
{
|
||
if (_localManifest->versionGreaterOrEquals(_remoteManifest, _versionCompareHandle))
|
||
{
|
||
_updateState = State::UP_TO_DATE;
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ALREADY_UP_TO_DATE);
|
||
}
|
||
else
|
||
{
|
||
_updateState = State::PREDOWNLOAD_MANIFEST;
|
||
downloadManifest();
|
||
}
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::downloadManifest()
|
||
{
|
||
if (_updateState != State::PREDOWNLOAD_MANIFEST)
|
||
return;
|
||
|
||
std::string manifestUrl = _localManifest->getManifestFileUrl();
|
||
|
||
if (manifestUrl.size() > 0)
|
||
{
|
||
_updateState = State::DOWNLOADING_MANIFEST;
|
||
// Download version file asynchronously
|
||
_downloader->createDownloadFileTask(manifestUrl, _tempManifestPath, MANIFEST_ID);
|
||
}
|
||
// No manifest file found
|
||
else
|
||
{
|
||
CCLOG("AssetsManagerEx : No manifest file found, check update failed\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_DOWNLOAD_MANIFEST);
|
||
_updateState = State::UNCHECKED;
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::parseManifest()
|
||
{
|
||
if (_updateState != State::MANIFEST_LOADED)
|
||
return;
|
||
|
||
_remoteManifest->parseFile(_tempManifestPath);
|
||
|
||
if (!_remoteManifest->isLoaded())
|
||
{
|
||
CCLOG("AssetsManagerEx : Error parsing manifest file, %s", _tempManifestPath.c_str());
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_PARSE_MANIFEST);
|
||
_updateState = State::UNCHECKED;
|
||
}
|
||
else
|
||
{
|
||
if (_localManifest->versionGreaterOrEquals(_remoteManifest, _versionCompareHandle))
|
||
{
|
||
_updateState = State::UP_TO_DATE;
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ALREADY_UP_TO_DATE);
|
||
}
|
||
else
|
||
{
|
||
_updateState = State::NEED_UPDATE;
|
||
|
||
if (_updateEntry == UpdateEntry::DO_UPDATE)
|
||
{
|
||
startUpdate();
|
||
}
|
||
else if (_updateEntry == UpdateEntry::CHECK_UPDATE)
|
||
{
|
||
prepareUpdate();
|
||
}
|
||
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::NEW_VERSION_FOUND);
|
||
}
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::prepareUpdate()
|
||
{
|
||
if (_updateState != State::NEED_UPDATE)
|
||
return;
|
||
|
||
// Clean up before update
|
||
_failedUnits.clear();
|
||
_downloadUnits.clear();
|
||
_totalWaitToDownload = _totalToDownload = 0;
|
||
_nextSavePoint = 0;
|
||
_percent = _percentByFile = _sizeCollected = _totalDownloaded = _totalSize = 0;
|
||
_downloadResumed = false;
|
||
_downloadedSize.clear();
|
||
_totalEnabled = false;
|
||
|
||
// Temporary manifest exists, previously updating and equals to the remote version, resuming previous download
|
||
if (_tempManifest && _tempManifest->isLoaded() && _tempManifest->isUpdating() && _tempManifest->versionEquals(_remoteManifest))
|
||
{
|
||
_tempManifest->saveToFile(_tempManifestPath);
|
||
_tempManifest->genResumeAssetsList(&_downloadUnits);
|
||
_totalWaitToDownload = _totalToDownload = (int)_downloadUnits.size();
|
||
_downloadResumed = true;
|
||
|
||
// Collect total size
|
||
for(auto iter : _downloadUnits)
|
||
{
|
||
const DownloadUnit& unit = iter.second;
|
||
if (unit.size > 0)
|
||
{
|
||
_totalSize += unit.size;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// Temporary manifest exists, but can't be parsed or version doesn't equals remote manifest (out of date)
|
||
if (_tempManifest)
|
||
{
|
||
// Remove all temp files
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
CC_SAFE_RELEASE(_tempManifest);
|
||
// Recreate temp storage path and save remote manifest
|
||
_fileUtils->createDirectory(_tempStoragePath);
|
||
_remoteManifest->saveToFile(_tempManifestPath);
|
||
}
|
||
|
||
// Temporary manifest will be used to register the download states of each asset,
|
||
// in this case, it equals remote manifest.
|
||
_tempManifest = _remoteManifest;
|
||
|
||
// Check difference between local manifest and remote manifest
|
||
std::unordered_map<std::string, Manifest::AssetDiff> diff_map = _localManifest->genDiff(_remoteManifest);
|
||
if (diff_map.size() == 0)
|
||
{
|
||
updateSucceed();
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
// Generate download units for all assets that need to be updated or added
|
||
std::string packageUrl = _remoteManifest->getPackageUrl();
|
||
// Preprocessing local files in previous version and creating download folders
|
||
for (auto it = diff_map.begin(); it != diff_map.end(); ++it)
|
||
{
|
||
Manifest::AssetDiff diff = it->second;
|
||
if (diff.type != Manifest::DiffType::DELETED)
|
||
{
|
||
std::string path = diff.asset.path;
|
||
DownloadUnit unit;
|
||
unit.customId = it->first;
|
||
unit.srcUrl = packageUrl + path + "?md5=" + diff.asset.md5;
|
||
unit.storagePath = _tempStoragePath + path;
|
||
unit.size = diff.asset.size;
|
||
_downloadUnits.emplace(unit.customId, unit);
|
||
_tempManifest->setAssetDownloadState(it->first, Manifest::DownloadState::UNSTARTED);
|
||
_totalSize += unit.size;
|
||
}
|
||
}
|
||
// Start updating the temp manifest
|
||
_tempManifest->setUpdating(true);
|
||
// Save current download manifest information for resuming
|
||
_tempManifest->saveToFile(_tempManifestPath);
|
||
|
||
_totalWaitToDownload = _totalToDownload = (int)_downloadUnits.size();
|
||
}
|
||
}
|
||
_updateState = State::READY_TO_UPDATE;
|
||
}
|
||
|
||
void AssetsManagerEx::startUpdate()
|
||
{
|
||
if (_updateState == State::NEED_UPDATE)
|
||
{
|
||
prepareUpdate();
|
||
}
|
||
if (_updateState == State::READY_TO_UPDATE)
|
||
{
|
||
_totalSize = 0;
|
||
_updateState = State::UPDATING;
|
||
std::string msg;
|
||
if (_downloadResumed)
|
||
{
|
||
msg = StringUtils::format("Resuming from previous unfinished update, %d files remains to be finished.", _totalToDownload);
|
||
}
|
||
else
|
||
{
|
||
msg = StringUtils::format("Start to update %d files from remote package.", _totalToDownload);
|
||
}
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_PROGRESSION, "", msg);
|
||
batchDownload();
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::updateSucceed()
|
||
{
|
||
// Set temp manifest's updating
|
||
if (_tempManifest != nullptr) {
|
||
_tempManifest->setUpdating(false);
|
||
}
|
||
|
||
// Every thing is correctly downloaded, do the following
|
||
// 1. rename temporary manifest to valid manifest
|
||
if (_fileUtils->isFileExist(_tempManifestPath)) {
|
||
_fileUtils->renameFile(_tempStoragePath, TEMP_MANIFEST_FILENAME, MANIFEST_FILENAME);
|
||
}
|
||
|
||
// 2. Get the delete files
|
||
std::unordered_map<std::string, Manifest::AssetDiff> diff_map = _localManifest->genDiff(_remoteManifest);
|
||
|
||
// 3. merge temporary storage path to storage path so that temporary version turns to cached version
|
||
if (_fileUtils->isDirectoryExist(_tempStoragePath))
|
||
{
|
||
// Merging all files in temp storage path to storage path
|
||
std::vector<std::string> files;
|
||
_fileUtils->listFilesRecursively(_tempStoragePath, &files);
|
||
int baseOffset = (int)_tempStoragePath.length();
|
||
std::string relativePath, dstPath;
|
||
for (std::vector<std::string>::iterator it = files.begin(); it != files.end(); ++it)
|
||
{
|
||
relativePath.assign((*it).substr(baseOffset));
|
||
dstPath.assign(_storagePath + relativePath);
|
||
// Create directory
|
||
if (relativePath.back() == '/')
|
||
{
|
||
_fileUtils->createDirectory(dstPath);
|
||
}
|
||
// Copy file
|
||
else
|
||
{
|
||
if (_fileUtils->isFileExist(dstPath))
|
||
{
|
||
_fileUtils->removeFile(dstPath);
|
||
}
|
||
_fileUtils->renameFile(*it, dstPath);
|
||
}
|
||
|
||
// Remove from delete list for safe, although this is not the case in general.
|
||
auto diff_itr = diff_map.find(relativePath);
|
||
if (diff_itr != diff_map.end()) {
|
||
diff_map.erase(diff_itr);
|
||
}
|
||
}
|
||
|
||
// Preprocessing local files in previous version and creating download folders
|
||
for (auto it = diff_map.begin(); it != diff_map.end(); ++it)
|
||
{
|
||
Manifest::AssetDiff diff = it->second;
|
||
if (diff.type == Manifest::DiffType::DELETED)
|
||
{
|
||
// TODO: Do this when download finish, it don’t matter delete or not.
|
||
std::string exsitedPath = _storagePath + diff.asset.path;
|
||
_fileUtils->removeFile(exsitedPath);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 4. swap the localManifest
|
||
CC_SAFE_RELEASE(_localManifest);
|
||
_localManifest = _remoteManifest;
|
||
_localManifest->setManifestRoot(_storagePath);
|
||
_remoteManifest = nullptr;
|
||
// 5. make local manifest take effect
|
||
prepareLocalManifest();
|
||
// 6. Set update state
|
||
_updateState = State::UP_TO_DATE;
|
||
// 7. Notify finished event
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_FINISHED);
|
||
// 8. Remove temp storage path
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
}
|
||
|
||
void AssetsManagerEx::checkUpdate()
|
||
{
|
||
if (_updateEntry != UpdateEntry::NONE)
|
||
{
|
||
CCLOGERROR("AssetsManagerEx::checkUpdate, updateEntry isn't NONE");
|
||
return;
|
||
}
|
||
|
||
if (!_inited){
|
||
CCLOG("AssetsManagerEx : Manifests uninited.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return;
|
||
}
|
||
if (!_localManifest->isLoaded())
|
||
{
|
||
CCLOG("AssetsManagerEx : No local manifest file found error.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return;
|
||
}
|
||
|
||
_updateEntry = UpdateEntry::CHECK_UPDATE;
|
||
|
||
switch (_updateState) {
|
||
case State::FAIL_TO_UPDATE:
|
||
_updateState = State::UNCHECKED;
|
||
case State::UNCHECKED:
|
||
case State::PREDOWNLOAD_VERSION:
|
||
{
|
||
downloadVersion();
|
||
}
|
||
break;
|
||
case State::UP_TO_DATE:
|
||
{
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ALREADY_UP_TO_DATE);
|
||
}
|
||
break;
|
||
case State::NEED_UPDATE:
|
||
{
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::NEW_VERSION_FOUND);
|
||
}
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::update()
|
||
{
|
||
if (_updateEntry != UpdateEntry::NONE)
|
||
{
|
||
CCLOGERROR("AssetsManagerEx::update, updateEntry isn't NONE");
|
||
return;
|
||
}
|
||
|
||
if (!_inited){
|
||
CCLOG("AssetsManagerEx : Manifests uninited.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return;
|
||
}
|
||
if (!_localManifest->isLoaded())
|
||
{
|
||
CCLOG("AssetsManagerEx : No local manifest file found error.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return;
|
||
}
|
||
|
||
_updateEntry = UpdateEntry::DO_UPDATE;
|
||
|
||
switch (_updateState) {
|
||
case State::UNCHECKED:
|
||
{
|
||
_updateState = State::PREDOWNLOAD_VERSION;
|
||
}
|
||
case State::PREDOWNLOAD_VERSION:
|
||
{
|
||
downloadVersion();
|
||
}
|
||
break;
|
||
case State::VERSION_LOADED:
|
||
{
|
||
parseVersion();
|
||
}
|
||
break;
|
||
case State::PREDOWNLOAD_MANIFEST:
|
||
{
|
||
downloadManifest();
|
||
}
|
||
break;
|
||
case State::MANIFEST_LOADED:
|
||
{
|
||
parseManifest();
|
||
}
|
||
break;
|
||
case State::FAIL_TO_UPDATE:
|
||
case State::READY_TO_UPDATE:
|
||
case State::NEED_UPDATE:
|
||
{
|
||
// Manifest not loaded yet
|
||
if (!_remoteManifest->isLoaded())
|
||
{
|
||
_updateState = State::PREDOWNLOAD_MANIFEST;
|
||
downloadManifest();
|
||
}
|
||
else if (_updateEntry == UpdateEntry::DO_UPDATE)
|
||
{
|
||
startUpdate();
|
||
}
|
||
}
|
||
break;
|
||
case State::UP_TO_DATE:
|
||
case State::UPDATING:
|
||
case State::UNZIPPING:
|
||
_updateEntry = UpdateEntry::NONE;
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::updateAssets(const DownloadUnits& assets)
|
||
{
|
||
if (!_inited){
|
||
CCLOG("AssetsManagerEx : Manifests uninited.\n");
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_NO_LOCAL_MANIFEST);
|
||
return;
|
||
}
|
||
|
||
if (_updateState != State::UPDATING && _localManifest->isLoaded() && _remoteManifest->isLoaded())
|
||
{
|
||
_updateState = State::UPDATING;
|
||
_downloadUnits.clear();
|
||
_downloadedSize.clear();
|
||
_percent = _percentByFile = _sizeCollected = _totalDownloaded = _totalSize = 0;
|
||
_totalWaitToDownload = _totalToDownload = (int)assets.size();
|
||
_nextSavePoint = 0;
|
||
_totalEnabled = false;
|
||
if (_totalToDownload > 0)
|
||
{
|
||
_downloadUnits = assets;
|
||
this->batchDownload();
|
||
}
|
||
else if (_totalToDownload == 0)
|
||
{
|
||
onDownloadUnitsFinished();
|
||
}
|
||
}
|
||
}
|
||
|
||
const DownloadUnits& AssetsManagerEx::getFailedAssets() const
|
||
{
|
||
return _failedUnits;
|
||
}
|
||
|
||
void AssetsManagerEx::downloadFailedAssets()
|
||
{
|
||
CCLOG("AssetsManagerEx : Start update %lu failed assets.\n", static_cast<unsigned long>(_failedUnits.size()));
|
||
updateAssets(_failedUnits);
|
||
}
|
||
|
||
void AssetsManagerEx::fileError(const std::string& identifier, const std::string& errorStr, int errorCode, int errorCodeInternal)
|
||
{
|
||
auto unitIt = _downloadUnits.find(identifier);
|
||
// Found unit and add it to failed units
|
||
if (unitIt != _downloadUnits.end())
|
||
{
|
||
_totalWaitToDownload--;
|
||
|
||
DownloadUnit unit = unitIt->second;
|
||
_failedUnits.emplace(unit.customId, unit);
|
||
}
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_UPDATING, identifier, errorStr, errorCode, errorCodeInternal);
|
||
_tempManifest->setAssetDownloadState(identifier, Manifest::DownloadState::UNSTARTED);
|
||
|
||
_currConcurrentTask = std::max(0, _currConcurrentTask-1);
|
||
queueDowload();
|
||
}
|
||
|
||
void AssetsManagerEx::fileSuccess(const std::string &customId, const std::string &storagePath)
|
||
{
|
||
// Set download state to SUCCESSED
|
||
_tempManifest->setAssetDownloadState(customId, Manifest::DownloadState::SUCCESSED);
|
||
|
||
auto unitIt = _failedUnits.find(customId);
|
||
// Found unit and delete it
|
||
if (unitIt != _failedUnits.end())
|
||
{
|
||
// Remove from failed units list
|
||
_failedUnits.erase(unitIt);
|
||
}
|
||
|
||
unitIt = _downloadUnits.find(customId);
|
||
if (unitIt != _downloadUnits.end())
|
||
{
|
||
// Reduce count only when unit found in _downloadUnits
|
||
_totalWaitToDownload--;
|
||
|
||
_percentByFile = 100 * (float)(_totalToDownload - _totalWaitToDownload) / _totalToDownload;
|
||
// Notify progression event
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_PROGRESSION, "");
|
||
}
|
||
// Notify asset updated event
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ASSET_UPDATED, customId);
|
||
|
||
_currConcurrentTask = std::max(0, _currConcurrentTask-1);
|
||
queueDowload();
|
||
}
|
||
|
||
void AssetsManagerEx::onError(const network::DownloadTask& task,
|
||
int errorCode,
|
||
int errorCodeInternal,
|
||
const std::string& errorStr)
|
||
{
|
||
// Skip version error occurred
|
||
if (task.identifier == VERSION_ID)
|
||
{
|
||
CCLOG("AssetsManagerEx : Fail to download version file, step skipped\n");
|
||
_updateState = State::PREDOWNLOAD_MANIFEST;
|
||
downloadManifest();
|
||
}
|
||
else if (task.identifier == MANIFEST_ID)
|
||
{
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::ERROR_DOWNLOAD_MANIFEST, task.identifier, errorStr, errorCode, errorCodeInternal);
|
||
_updateState = State::FAIL_TO_UPDATE;
|
||
}
|
||
else
|
||
{
|
||
if (_downloadingTask.find(task.identifier) != _downloadingTask.end()) {
|
||
_downloadingTask.erase(task.identifier);
|
||
}
|
||
fileError(task.identifier, errorStr, errorCode, errorCodeInternal);
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::onProgress(double total, double downloaded, const std::string& /*url*/, const std::string &customId)
|
||
{
|
||
if (customId == VERSION_ID || customId == MANIFEST_ID)
|
||
{
|
||
_percent = 100 * downloaded / total;
|
||
// Notify progression event
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_PROGRESSION, customId);
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
// Calcul total downloaded
|
||
bool found = false;
|
||
_totalDownloaded = 0;
|
||
for (auto it = _downloadedSize.begin(); it != _downloadedSize.end(); ++it)
|
||
{
|
||
if (it->first == customId)
|
||
{
|
||
it->second = downloaded;
|
||
found = true;
|
||
}
|
||
_totalDownloaded += it->second;
|
||
}
|
||
// Collect information if not registed
|
||
if (!found)
|
||
{
|
||
// Set download state to DOWNLOADING, this will run only once in the download process
|
||
_tempManifest->setAssetDownloadState(customId, Manifest::DownloadState::DOWNLOADING);
|
||
// Register the download size information
|
||
_downloadedSize.emplace(customId, downloaded);
|
||
// Check download unit size existance, if not exist collect size in total size
|
||
if (_downloadUnits[customId].size == 0)
|
||
{
|
||
_totalSize += total;
|
||
_sizeCollected++;
|
||
// All collected, enable total size
|
||
if (_sizeCollected == _totalToDownload)
|
||
{
|
||
_totalEnabled = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (_totalEnabled && _updateState == State::UPDATING)
|
||
{
|
||
float currentPercent = 100 * _totalDownloaded / _totalSize;
|
||
// Notify at integer level change
|
||
if ((int)currentPercent != (int)_percent) {
|
||
_percent = currentPercent;
|
||
// Notify progression event
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_PROGRESSION, customId);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::onSuccess(const std::string &/*srcUrl*/, const std::string &storagePath, const std::string &customId)
|
||
{
|
||
if (customId == VERSION_ID)
|
||
{
|
||
_updateState = State::VERSION_LOADED;
|
||
parseVersion();
|
||
}
|
||
else if (customId == MANIFEST_ID)
|
||
{
|
||
_updateState = State::MANIFEST_LOADED;
|
||
parseManifest();
|
||
}
|
||
else
|
||
{
|
||
if (_downloadingTask.find(customId) != _downloadingTask.end()) {
|
||
_downloadingTask.erase(customId);
|
||
}
|
||
|
||
bool ok = true;
|
||
auto &assets = _remoteManifest->getAssets();
|
||
auto assetIt = assets.find(customId);
|
||
if (assetIt != assets.end())
|
||
{
|
||
Manifest::Asset asset = assetIt->second;
|
||
if (_verifyCallback != nullptr)
|
||
{
|
||
ok = _verifyCallback(storagePath, asset);
|
||
}
|
||
}
|
||
|
||
if (ok)
|
||
{
|
||
bool compressed = assetIt != assets.end() ? assetIt->second.compressed : false;
|
||
if (compressed)
|
||
{
|
||
decompressDownloadedZip(customId, storagePath);
|
||
}
|
||
else
|
||
{
|
||
fileSuccess(customId, storagePath);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
fileError(customId, "Asset file verification failed after downloaded");
|
||
}
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::destroyDownloadedVersion()
|
||
{
|
||
_fileUtils->removeDirectory(_storagePath);
|
||
_fileUtils->removeDirectory(_tempStoragePath);
|
||
}
|
||
|
||
void AssetsManagerEx::batchDownload()
|
||
{
|
||
_queue.clear();
|
||
for(auto iter : _downloadUnits)
|
||
{
|
||
const DownloadUnit& unit = iter.second;
|
||
if (unit.size > 0)
|
||
{
|
||
_totalSize += unit.size;
|
||
_sizeCollected++;
|
||
}
|
||
|
||
_queue.push_back(iter.first);
|
||
}
|
||
// All collected, enable total size
|
||
if (_sizeCollected == _totalToDownload)
|
||
{
|
||
_totalEnabled = true;
|
||
}
|
||
|
||
queueDowload();
|
||
}
|
||
|
||
void AssetsManagerEx::queueDowload()
|
||
{
|
||
if (_totalWaitToDownload == 0 || (_canceled && _currConcurrentTask == 0))
|
||
{
|
||
this->onDownloadUnitsFinished();
|
||
return;
|
||
}
|
||
|
||
while (_currConcurrentTask < _maxConcurrentTask && _queue.size() > 0 && !_canceled)
|
||
{
|
||
std::string key = _queue.back();
|
||
_queue.pop_back();
|
||
|
||
_currConcurrentTask++;
|
||
DownloadUnit& unit = _downloadUnits[key];
|
||
_fileUtils->createDirectory(basename(unit.storagePath));
|
||
auto downloadTask = _downloader->createDownloadFileTask(unit.srcUrl, unit.storagePath, unit.customId);
|
||
_downloadingTask.emplace(unit.customId, downloadTask);
|
||
_tempManifest->setAssetDownloadState(key, Manifest::DownloadState::DOWNLOADING);
|
||
}
|
||
if (_percentByFile / 100 > _nextSavePoint)
|
||
{
|
||
// Save current download manifest information for resuming
|
||
_tempManifest->saveToFile(_tempManifestPath);
|
||
_nextSavePoint += SAVE_POINT_INTERVAL;
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::onDownloadUnitsFinished()
|
||
{
|
||
// Always save current download manifest information for resuming
|
||
_tempManifest->saveToFile(_tempManifestPath);
|
||
|
||
// Finished with error check
|
||
if (_failedUnits.size() > 0)
|
||
{
|
||
_updateState = State::FAIL_TO_UPDATE;
|
||
dispatchUpdateEvent(EventAssetsManagerEx::EventCode::UPDATE_FAILED);
|
||
}
|
||
else if (_updateState == State::UPDATING)
|
||
{
|
||
updateSucceed();
|
||
}
|
||
}
|
||
|
||
void AssetsManagerEx::cancelUpdate()
|
||
{
|
||
if (_canceled)
|
||
{
|
||
return;
|
||
}
|
||
_canceled = true;
|
||
std::vector<std::shared_ptr<const network::DownloadTask>> tasks;
|
||
for (const auto& it : _downloadingTask)
|
||
{
|
||
tasks.push_back(it.second);
|
||
}
|
||
for (const auto& it : tasks)
|
||
{
|
||
_downloader->abort(*it);
|
||
}
|
||
_downloadingTask.clear();
|
||
}
|
||
|
||
NS_CC_EXT_END
|