/***************************************************************************** * Copyright (c) 2014-2018 OpenRCT2 developers * * For a complete list of all authors, please refer to contributors.md * Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2 * * OpenRCT2 is licensed under the GNU General Public License version 3. *****************************************************************************/ #pragma once #ifdef __ENABLE_SCRIPTING__ # include "../common.h" # include "../core/FileWatcher.h" # include "HookEngine.h" # include "Plugin.h" # include # include # include # include # include # include # include struct duk_hthread; typedef struct duk_hthread duk_context; class FileWatcher; class InteractiveConsole; namespace OpenRCT2 { interface IPlatformEnvironment; } namespace OpenRCT2::Scripting { class ScriptExecutionInfo { private: std::shared_ptr _plugin; bool _isGameStateMutable{}; public: class PluginScope { private: ScriptExecutionInfo& _execInfo; std::shared_ptr _plugin; public: PluginScope(ScriptExecutionInfo& execInfo, std::shared_ptr plugin, bool isGameStateMutable) : _execInfo(execInfo) , _plugin(plugin) { _execInfo._plugin = plugin; _execInfo._isGameStateMutable = isGameStateMutable; } PluginScope(const PluginScope&) = delete; ~PluginScope() { _execInfo._plugin = nullptr; _execInfo._isGameStateMutable = false; } }; std::shared_ptr GetCurrentPlugin() { return _plugin; } bool IsGameStateMutable() { return _isGameStateMutable; } }; class DukContext { private: duk_context* _context{}; public: DukContext(); DukContext(DukContext&) = delete; DukContext(DukContext&& src) noexcept : _context(std::move(src._context)) { } ~DukContext(); operator duk_context*() { return _context; } }; class ScriptEngine { private: InteractiveConsole& _console; IPlatformEnvironment& _env; DukContext _context; bool _initialised{}; bool _pluginsLoaded{}; bool _pluginsStarted{}; std::queue, std::string>> _evalQueue; std::vector> _plugins; uint32_t _lastHotReloadCheckTick{}; HookEngine _hookEngine; ScriptExecutionInfo _execInfo; std::unique_ptr _pluginFileWatcher; std::unordered_set _changedPluginFiles; std::mutex _changedPluginFilesMutex; std::vector)>> _pluginStoppedSubscriptions; public: ScriptEngine(InteractiveConsole& console, IPlatformEnvironment& env); ScriptEngine(ScriptEngine&) = delete; duk_context* GetContext() { return _context; } HookEngine& GetHookEngine() { return _hookEngine; } ScriptExecutionInfo& GetExecInfo() { return _execInfo; } std::vector>& GetPlugins() { return _plugins; } void LoadPlugins(); void UnloadPlugins(); void Update(); std::future Eval(const std::string& s); bool ExecutePluginCall( const std::shared_ptr& plugin, const DukValue& func, const std::vector& args, bool isGameStateMutable); void LogPluginInfo(const std::shared_ptr& plugin, const std::string_view& message); void SubscribeToPluginStoppedEvent(std::function)> callback) { _pluginStoppedSubscriptions.push_back(callback); } void AddNetworkPlugin(const std::string_view& code); private: void Initialise(); void StartPlugins(); void StopPlugins(); void LoadPlugin(const std::string& path); void LoadPlugin(std::shared_ptr& plugin); void StopPlugin(std::shared_ptr plugin); bool ShouldLoadScript(const std::string& path); bool ShouldStartPlugin(const std::shared_ptr& plugin); void SetupHotReloading(); void AutoReloadPlugins(); void ProcessREPL(); }; bool IsGameStateMutable(); void ThrowIfGameStateNotMutable(); std::string Stringify(const DukValue& value); } // namespace OpenRCT2::Scripting #endif