mirror of
https://github.com/JulianGro/overte.git
synced 2025-04-25 20:15:15 +02:00
240 lines
8.6 KiB
C++
240 lines
8.6 KiB
C++
//
|
|
// ScriptCache.cpp
|
|
// libraries/script-engine/src
|
|
//
|
|
// Created by Brad Hefta-Gaub on 2015-03-30
|
|
// Copyright 2015 High Fidelity, Inc.
|
|
//
|
|
// Distributed under the Apache License, Version 2.0.
|
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
|
//
|
|
|
|
#include "ScriptCache.h"
|
|
|
|
#include <QCoreApplication>
|
|
#include <QEventLoop>
|
|
#include <QNetworkAccessManager>
|
|
#include <QNetworkConfiguration>
|
|
#include <QNetworkReply>
|
|
#include <QObject>
|
|
#include <QThread>
|
|
#include <QRegularExpression>
|
|
|
|
#include <assert.h>
|
|
#include <SharedUtil.h>
|
|
|
|
#include "ScriptEngines.h"
|
|
#include "ScriptEngineLogging.h"
|
|
#include <QtCore/QTimer>
|
|
|
|
ScriptCache::ScriptCache(QObject* parent) {
|
|
// nothing to do here...
|
|
}
|
|
|
|
void ScriptCache::clearCache() {
|
|
Lock lock(_containerLock);
|
|
_scriptCache.clear();
|
|
}
|
|
|
|
void ScriptCache::clearATPScriptsFromCache() {
|
|
Lock lock(_containerLock);
|
|
qCDebug(scriptengine) << "Clearing ATP scripts from ScriptCache";
|
|
for (auto it = _scriptCache.begin(); it != _scriptCache.end();) {
|
|
if (it.key().scheme() == "atp") {
|
|
qCDebug(scriptengine) << "Removing: " << it.key();
|
|
it = _scriptCache.erase(it);
|
|
} else {
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
|
|
QString ScriptCache::getScript(const QUrl& unnormalizedURL, ScriptUser* scriptUser, bool& isPending, bool reload) {
|
|
QUrl url = ResourceManager::normalizeURL(unnormalizedURL);
|
|
QString scriptContents;
|
|
|
|
Lock lock(_containerLock);
|
|
if (_scriptCache.contains(url) && !reload) {
|
|
qCDebug(scriptengine) << "Found script in cache:" << url.toString();
|
|
scriptContents = _scriptCache[url];
|
|
lock.unlock();
|
|
scriptUser->scriptContentsAvailable(url, scriptContents);
|
|
isPending = false;
|
|
} else {
|
|
isPending = true;
|
|
bool alreadyWaiting = _scriptUsers.contains(url);
|
|
_scriptUsers.insert(url, scriptUser);
|
|
lock.unlock();
|
|
|
|
if (alreadyWaiting) {
|
|
qCDebug(scriptengine) << "Already downloading script at:" << url.toString();
|
|
} else {
|
|
auto request = ResourceManager::createResourceRequest(nullptr, url);
|
|
request->setCacheEnabled(!reload);
|
|
connect(request, &ResourceRequest::finished, this, &ScriptCache::scriptDownloaded);
|
|
request->send();
|
|
}
|
|
}
|
|
return scriptContents;
|
|
}
|
|
|
|
void ScriptCache::deleteScript(const QUrl& unnormalizedURL) {
|
|
QUrl url = ResourceManager::normalizeURL(unnormalizedURL);
|
|
Lock lock(_containerLock);
|
|
if (_scriptCache.contains(url)) {
|
|
qCDebug(scriptengine) << "Delete script from cache:" << url.toString();
|
|
_scriptCache.remove(url);
|
|
}
|
|
}
|
|
|
|
void ScriptCache::scriptDownloaded() {
|
|
ResourceRequest* req = qobject_cast<ResourceRequest*>(sender());
|
|
QUrl url = req->getUrl();
|
|
|
|
Lock lock(_containerLock);
|
|
QList<ScriptUser*> scriptUsers = _scriptUsers.values(url);
|
|
_scriptUsers.remove(url);
|
|
|
|
if (!DependencyManager::get<ScriptEngines>()->isStopped()) {
|
|
if (req->getResult() == ResourceRequest::Success) {
|
|
auto scriptContents = req->getData();
|
|
_scriptCache[url] = scriptContents;
|
|
lock.unlock();
|
|
qCDebug(scriptengine) << "Done downloading script at:" << url.toString();
|
|
|
|
foreach(ScriptUser* user, scriptUsers) {
|
|
user->scriptContentsAvailable(url, scriptContents);
|
|
}
|
|
} else {
|
|
lock.unlock();
|
|
qCWarning(scriptengine) << "Error loading script from URL " << url;
|
|
foreach(ScriptUser* user, scriptUsers) {
|
|
user->errorInLoadingScript(url);
|
|
}
|
|
}
|
|
}
|
|
|
|
req->deleteLater();
|
|
}
|
|
|
|
void ScriptCache::getScriptContents(const QString& scriptOrURL, contentAvailableCallback contentAvailable, bool forceDownload) {
|
|
#ifdef THREAD_DEBUGGING
|
|
qCDebug(scriptengine) << "ScriptCache::getScriptContents() on thread [" << QThread::currentThread() << "] expected thread [" << thread() << "]";
|
|
#endif
|
|
QUrl unnormalizedURL(scriptOrURL);
|
|
QUrl url = ResourceManager::normalizeURL(unnormalizedURL);
|
|
|
|
// attempt to determine if this is a URL to a script, or if this is actually a script itself (which is valid in the
|
|
// entityScript use case)
|
|
if (unnormalizedURL.scheme().isEmpty() &&
|
|
scriptOrURL.simplified().replace(" ", "").contains(QRegularExpression(R"(\(function\([a-z]?[\w,]*\){)"))) {
|
|
contentAvailable(scriptOrURL, scriptOrURL, false, true);
|
|
return;
|
|
}
|
|
|
|
// give a similar treatment to javacript: urls
|
|
if (unnormalizedURL.scheme() == "javascript") {
|
|
QString contents { scriptOrURL };
|
|
contents.replace(QRegularExpression("^javascript:"), "");
|
|
contentAvailable(scriptOrURL, contents, false, true);
|
|
return;
|
|
}
|
|
|
|
Lock lock(_containerLock);
|
|
if (_scriptCache.contains(url) && !forceDownload) {
|
|
auto scriptContent = _scriptCache[url];
|
|
lock.unlock();
|
|
qCDebug(scriptengine) << "Found script in cache:" << url.toString();
|
|
contentAvailable(url.toString(), scriptContent, true, true);
|
|
} else {
|
|
auto& scriptRequest = _activeScriptRequests[url];
|
|
|
|
bool alreadyWaiting = scriptRequest.scriptUsers.size() > 0;
|
|
scriptRequest.scriptUsers.push_back(contentAvailable);
|
|
|
|
lock.unlock();
|
|
|
|
if (alreadyWaiting) {
|
|
qCDebug(scriptengine) << "Already downloading script at:" << url.toString();
|
|
} else {
|
|
#ifdef THREAD_DEBUGGING
|
|
qCDebug(scriptengine) << "about to call: ResourceManager::createResourceRequest(this, url); on thread [" << QThread::currentThread() << "] expected thread [" << thread() << "]";
|
|
#endif
|
|
auto request = ResourceManager::createResourceRequest(nullptr, url);
|
|
Q_ASSERT(request);
|
|
request->setCacheEnabled(!forceDownload);
|
|
connect(request, &ResourceRequest::finished, this, &ScriptCache::scriptContentAvailable);
|
|
request->send();
|
|
}
|
|
}
|
|
}
|
|
|
|
static const int MAX_RETRIES = 5;
|
|
static int START_DELAY_BETWEEN_RETRIES = 200;
|
|
|
|
void ScriptCache::scriptContentAvailable() {
|
|
#ifdef THREAD_DEBUGGING
|
|
qCDebug(scriptengine) << "ScriptCache::scriptContentAvailable() on thread [" << QThread::currentThread() << "] expected thread [" << thread() << "]";
|
|
#endif
|
|
ResourceRequest* req = qobject_cast<ResourceRequest*>(sender());
|
|
QUrl url = req->getUrl();
|
|
|
|
QString scriptContent;
|
|
std::vector<contentAvailableCallback> allCallbacks;
|
|
bool finished { false };
|
|
bool success { false };
|
|
|
|
{
|
|
Q_ASSERT(req->getState() == ResourceRequest::Finished);
|
|
success = req->getResult() == ResourceRequest::Success;
|
|
|
|
Lock lock(_containerLock);
|
|
|
|
if (_activeScriptRequests.contains(url)) {
|
|
auto& scriptRequest = _activeScriptRequests[url];
|
|
|
|
if (success) {
|
|
allCallbacks = scriptRequest.scriptUsers;
|
|
|
|
_activeScriptRequests.remove(url);
|
|
|
|
_scriptCache[url] = scriptContent = req->getData();
|
|
finished = true;
|
|
qCDebug(scriptengine) << "Done downloading script at:" << url.toString();
|
|
} else {
|
|
if (scriptRequest.numRetries < MAX_RETRIES) {
|
|
++scriptRequest.numRetries;
|
|
|
|
qDebug() << "Script request failed: " << url;
|
|
|
|
int timeout = exp(scriptRequest.numRetries) * START_DELAY_BETWEEN_RETRIES;
|
|
QTimer::singleShot(timeout, this, [this, url]() {
|
|
qDebug() << "Retrying script request: " << url;
|
|
|
|
auto request = ResourceManager::createResourceRequest(nullptr, url);
|
|
Q_ASSERT(request);
|
|
|
|
// We've already made a request, so the cache must be disabled or it wasn't there, so enabling
|
|
// it will do nothing.
|
|
request->setCacheEnabled(false);
|
|
connect(request, &ResourceRequest::finished, this, &ScriptCache::scriptContentAvailable);
|
|
request->send();
|
|
});
|
|
} else {
|
|
// Dubious, but retained here because it matches the behavior before fixing the threading
|
|
scriptContent = _scriptCache[url];
|
|
finished = true;
|
|
qCWarning(scriptengine) << "Error loading script from URL " << url;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
req->deleteLater();
|
|
|
|
if (finished && !DependencyManager::get<ScriptEngines>()->isStopped()) {
|
|
foreach(contentAvailableCallback thisCallback, allCallbacks) {
|
|
thisCallback(url.toString(), scriptContent, true, success);
|
|
}
|
|
}
|
|
}
|