/***************************************************************************************** * * * OpenSpace * * * * Copyright (c) 2014-2019 * * * * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace { constexpr const char* KeyAsset = "Asset"; constexpr const char* _loggerCat = "SyncAssetTask"; constexpr std::chrono::milliseconds ProgressPollInterval(200); } // namespace namespace openspace { documentation::Documentation SyncAssetTask::documentation() { using namespace documentation; return { "SyncAssetTask", "sync_asset_task", { { "Type", new StringEqualVerifier("SyncAssetTask"), Optional::No, "The type of this task" }, { KeyAsset, new StringAnnotationVerifier("A file path to an asset"), Optional::No, "The asset file to sync" } } }; } class RequestListener : public AssetListener { public: virtual ~RequestListener() = default; void assetStateChanged(std::shared_ptr asset, Asset::State state) override { if (state == Asset::State::LoadingFailed) { LERROR(fmt::format("Failed to load asset: {}", asset->id())); } if (state == Asset::State::SyncRejected) { LERROR(fmt::format("Failed to sync asset: {}", asset->id())); } } void assetRequested(std::shared_ptr, std::shared_ptr) override {}; void assetUnrequested(std::shared_ptr, std::shared_ptr) override {}; }; SyncAssetTask::SyncAssetTask(const ghoul::Dictionary& dictionary) { documentation::testSpecificationAndThrow( documentation(), dictionary, "SyncAssetTask" ); _asset = dictionary.value(KeyAsset); } std::string SyncAssetTask::description() { return "Synchronize asset " + _asset; } void SyncAssetTask::perform(const Task::ProgressCallback& progressCallback) { SynchronizationWatcher watcher; scripting::ScriptEngine scriptEngine; registerCoreClasses(scriptEngine); for (OpenSpaceModule* m : global::moduleEngine.modules()) { scriptEngine.addLibrary(m->luaLibrary()); for (scripting::LuaLibrary& l : m->luaLibraries()) { scriptEngine.addLibrary(l); } } scriptEngine.initialize(); ghoul::lua::LuaState luaState; scriptEngine.initializeLuaState(luaState); AssetLoader loader(luaState, &watcher, "${ASSETS}"); RequestListener listener; loader.addAssetListener(&listener); loader.add(_asset); loader.rootAsset()->startSynchronizations(); std::vector> allAssets = loader.rootAsset()->subTreeAssets(); while (true) { bool inProgress = false; for (const std::shared_ptr& asset : allAssets) { Asset::State state = asset->state(); if (state == Asset::State::Unloaded || state == Asset::State::Loaded || state == Asset::State::Synchronizing) { inProgress = true; } } progressCallback(loader.rootAsset()->requestedSynchronizationProgress()); std::this_thread::sleep_for(ProgressPollInterval); watcher.notify(); if (!inProgress) { return; } } progressCallback(1.f); } } // namespace openspace