From d23a17daa5da8130cd922b91a6f3578bcb471096 Mon Sep 17 00:00:00 2001 From: Dale Glass Date: Sun, 5 Nov 2023 14:02:46 +0100 Subject: [PATCH] Add a test for bug #587 This will fail against the current master, since the current behavior is not correct. --- libraries/script-engine/src/ScriptManager.h | 10 ++ tests/script-engine/CMakeLists.txt | 12 ++ .../src/ScriptEngineNetworkedTests.cpp | 158 ++++++++++++++++++ .../src/ScriptEngineNetworkedTests.h | 37 ++++ tests/script-engine/src/tests/a.js | 3 + tests/script-engine/src/tests/b.js | 6 + tests/script-engine/src/tests/c.js | 6 + 7 files changed, 232 insertions(+) create mode 100644 tests/script-engine/src/ScriptEngineNetworkedTests.cpp create mode 100644 tests/script-engine/src/ScriptEngineNetworkedTests.h create mode 100644 tests/script-engine/src/tests/a.js create mode 100644 tests/script-engine/src/tests/b.js create mode 100644 tests/script-engine/src/tests/c.js diff --git a/libraries/script-engine/src/ScriptManager.h b/libraries/script-engine/src/ScriptManager.h index c4614c401e..01d0a1dbf0 100644 --- a/libraries/script-engine/src/ScriptManager.h +++ b/libraries/script-engine/src/ScriptManager.h @@ -439,6 +439,16 @@ public: QList getListOfEntityScriptIDs(); + /** + * @brief Whether the ScriptManager is stopped and unable to run scripts + * + * This is always false for NETWORKLESS_TEST_SCRIPT scripts. + * + * Otherwise, it checks whether scriptEngines is set and is not stopped. + * + * @return true + * @return false + */ bool isStopped() const; diff --git a/tests/script-engine/CMakeLists.txt b/tests/script-engine/CMakeLists.txt index 1e4428239c..51743129aa 100644 --- a/tests/script-engine/CMakeLists.txt +++ b/tests/script-engine/CMakeLists.txt @@ -8,6 +8,18 @@ macro (setup_testcase_dependencies) link_hifi_libraries(shared test-utils script-engine networking octree avatars entities model-networking material-networking model-serializers graphics gpu ktx shaders hfm image procedural) package_libraries_for_deployment() + + # The test system is a bit unusual in how it works, and generates targets on its own. + # This macro will be called for each of them, so we want to add stuff only to the + # right targets. + if("${TARGET_NAME}" STREQUAL "script-engine-ScriptEngineNetworkedTests") + file(COPY "${CMAKE_CURRENT_SOURCE_DIR}/src/tests" DESTINATION "${CMAKE_CURRENT_BINARY_DIR}/") +# add_custom_command(TARGET "${TARGET_NAME}" POST_BUILD +# COMMAND ${CMAKE_COMMAND} -E copy +# "${CMAKE_CURRENT_SOURCE_DIR}/src/tests" +# "${CMAKE_CURRENT_BINARY_DIR}/tests" +# ) + endif() endmacro () setup_hifi_testcase(Network) diff --git a/tests/script-engine/src/ScriptEngineNetworkedTests.cpp b/tests/script-engine/src/ScriptEngineNetworkedTests.cpp new file mode 100644 index 0000000000..b975563b60 --- /dev/null +++ b/tests/script-engine/src/ScriptEngineNetworkedTests.cpp @@ -0,0 +1,158 @@ +// +// Copyright 2023 Overte e.V. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// SPDX-License-Identifier: Apache-2.0 +// + +#include +#include +#include +#include + + +#include "ScriptEngineNetworkedTests.h" +#include "DependencyManager.h" + +#include "ScriptEngines.h" +#include "ScriptEngine.h" +#include "ScriptCache.h" +#include "ScriptManager.h" +#include "ScriptEngines.h" +#include "AddressManager.h" +#include "AccountManager.h" +#include "DomainAccountManager.h" +#include "ResourceManager.h" +#include "ResourceRequestObserver.h" +#include "StatTracker.h" +#include "MessagesClient.h" +#include "ResourceScriptingInterface.h" +#include "UserActivityLogger.h" +#include "UserActivityLoggerScriptingInterface.h" +#include "EntityScriptingInterface.h" +#include "ResourceManager.h" +#include "NodeList.h" +#include "../../../libraries/entities/src/EntityScriptingInterface.h" +//#include "../../../libraries/entities/src/EntityScriptingInterface.h" + +QTEST_MAIN(ScriptEngineNetworkedTests) + + + +void ScriptEngineNetworkedTests::initTestCase() { + // AudioClient starts networking, but for the purposes of the tests here we don't care, + // so just got to use some port. + int listenPort = 10000; + + DependencyManager::registerInheritance(); + + DependencyManager::set(true); // use the default user agent getter + DependencyManager::set(); + DependencyManager::set(); + DependencyManager::set(NodeType::Agent, listenPort); + + DependencyManager::set(ScriptManager::CLIENT_SCRIPT, QUrl("")); + DependencyManager::set(); + // DependencyManager::set(); + // DependencyManager::set(); + DependencyManager::set(); + DependencyManager::set(); + // DependencyManager::set(true); + + DependencyManager::set(); + DependencyManager::set(); + DependencyManager::set(); + DependencyManager::set(true); + DependencyManager::set(); + DependencyManager::set(); + + + auto nodeList = DependencyManager::get(); + nodeList->startThread(); + nodeList->setFlagTimeForConnectionStep(true); + +} + +ScriptManagerPointer ScriptEngineNetworkedTests::makeManager(const QString &scriptSource, const QString &scriptFilename) { + ScriptManagerPointer sm = scriptManagerFactory(ScriptManager::CLIENT_SCRIPT, scriptSource, scriptFilename); + + + sm->setAbortOnUncaughtException(true); + + connect(sm.get(), &ScriptManager::scriptLoaded, [](const QString& filename){ + qWarning() << "Loaded script" << filename; + }); + + + connect(sm.get(), &ScriptManager::errorLoadingScript, [](const QString& filename){ + qWarning() << "Failed to load script" << filename; + }); + + connect(sm.get(), &ScriptManager::printedMessage, [](const QString& message, const QString& engineName){ + qDebug() << "Printed message from engine" << engineName << ": " << message; + }); + + connect(sm.get(), &ScriptManager::infoMessage, [](const QString& message, const QString& engineName){ + qInfo() << "Info message from engine" << engineName << ": " << message; + }); + + connect(sm.get(), &ScriptManager::warningMessage, [](const QString& message, const QString& engineName){ + qWarning() << "Warning from engine" << engineName << ": " << message; + }); + + connect(sm.get(), &ScriptManager::errorMessage, [](const QString& message, const QString& engineName){ + qCritical() << "Error from engine" << engineName << ": " << message; + }); + + connect(sm.get(), &ScriptManager::finished, [](const QString& fileNameString, ScriptManagerPointer smp){ + qInfo() << "Finished running script" << fileNameString; + }); + + connect(sm.get(), &ScriptManager::runningStateChanged, [sm](){ + qInfo() << "Running state changed. Running = " << sm->isRunning() << "; Stopped = " << sm->isStopped() << "; Finished = " << sm->isFinished(); + }); + + connect(sm.get(), &ScriptManager::unhandledException, [](std::shared_ptr exception){ + qWarning() << "Exception from engine: " << exception; + }); + + + return sm; +} + +void ScriptEngineNetworkedTests::testRequire() { + auto sm = makeManager( + "print(\"Starting\");" + "Script.require('./tests/c.js');" + "print(\"Done\");" + "Script.stop(true);", "testRequire.js"); + QStringList printed; + QStringList expected {"Starting", "Value from A: 6", "Value from B: 6", "Done"}; + + + QVERIFY(!sm->isRunning()); + QVERIFY(!sm->isStopped()); + QVERIFY(!sm->isFinished()); + + connect(sm.get(), &ScriptManager::printedMessage, [&printed](const QString& message, const QString& engineName){ + printed.append(message); + }); + + + qInfo() << "About to run script"; + sm->run(); + + QVERIFY(!sm->isRunning()); + QVERIFY(!sm->isStopped()); + QVERIFY(sm->isFinished()); + + QVERIFY(printed.length() == expected.length()); + for(int i=0;i +#include "ScriptManager.h" +#include "ScriptEngine.h" + +using ScriptManagerPointer = std::shared_ptr; + + + + +class ScriptEngineNetworkedTests : public QObject { + Q_OBJECT +private slots: + void initTestCase(); + void testRequire(); + + +private: + ScriptManagerPointer makeManager(const QString &source, const QString &filename); + +}; + +#endif // overte_ScriptingEngineTests_h diff --git a/tests/script-engine/src/tests/a.js b/tests/script-engine/src/tests/a.js new file mode 100644 index 0000000000..9a9db6d653 --- /dev/null +++ b/tests/script-engine/src/tests/a.js @@ -0,0 +1,3 @@ +// a.js +module.exports = { value: 5 }; + diff --git a/tests/script-engine/src/tests/b.js b/tests/script-engine/src/tests/b.js new file mode 100644 index 0000000000..7606d7dbbb --- /dev/null +++ b/tests/script-engine/src/tests/b.js @@ -0,0 +1,6 @@ + +// b.js +var a = Script.require('./a.js'); +a.value += 1; +console.log('message from b'); +module.exports = a.value; \ No newline at end of file diff --git a/tests/script-engine/src/tests/c.js b/tests/script-engine/src/tests/c.js new file mode 100644 index 0000000000..7e3242337d --- /dev/null +++ b/tests/script-engine/src/tests/c.js @@ -0,0 +1,6 @@ + +// c.js +var a = Script.require('./a.js'); +var b = Script.require('./b.js'); +print("Value from A: " + a.value); +print("Value from B: " + b); \ No newline at end of file