/***************************************************************************************** * * * OpenSpace * * * * Copyright (c) 2014-2024 * * * * 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. * ****************************************************************************************/ #ifndef __OPENSPACE_CORE___PARALLELPEER___H__ #define __OPENSPACE_CORE___PARALLELPEER___H__ #include #include #include #include #include #include #include #include #include #include #include #include #include namespace openspace { namespace scripting { struct LuaLibrary; } class ParallelPeer : public properties::PropertyOwner { public: ParallelPeer(); ~ParallelPeer() override; void connect(); void setPort(std::string port); void setAddress(std::string address); void setServerName(std::string name); void setName(std::string name); bool isHost(); const std::string& hostName(); void requestHostship(); void resignHostship(); void setPassword(std::string password); void setHostPassword(std::string hostPassword); void disconnect(); void preSynchronization(); void sendScript(std::string script); void resetTimeOffset(); double latencyStandardDeviation() const; /** * Returns the Lua library that contains all Lua functions available to affect the * remote OS parallel connection. */ static scripting::LuaLibrary luaLibrary(); ParallelConnection::Status status(); int nConnections(); ghoul::Event<>& connectionEvent(); private: void queueInMessage(const ParallelConnection::Message& message); void sendAuthentication(); void handleCommunication(); void handleMessage(const ParallelConnection::Message&); void dataMessageReceived(const std::vector& message); void connectionStatusMessageReceived(const std::vector& message); void nConnectionsMessageReceived(const std::vector& message); void sendCameraKeyframe(); void sendTimeTimeline(); void setStatus(ParallelConnection::Status status); void setHostName(const std::string& hostName); void setNConnections(size_t nConnections); double convertTimestamp(double messageTimestamp); void analyzeTimeDifference(double messageTimestamp); properties::StringProperty _password; properties::StringProperty _hostPassword; properties::StringProperty _serverName; // While the port should in theory be an int, // we use a StringProperty to avoid a slider in the GUI. properties::StringProperty _port; properties::StringProperty _address; properties::StringProperty _name; properties::FloatProperty _bufferTime; properties::FloatProperty _timeKeyframeInterval; properties::FloatProperty _cameraKeyframeInterval; double _lastTimeKeyframeTimestamp = 0.0; double _lastCameraKeyframeTimestamp = 0.0; std::atomic_bool _shouldDisconnect = false; std::atomic _nConnections = 0; std::atomic _status = ParallelConnection::Status::Disconnected; std::string _hostName; std::deque _receiveBuffer; std::mutex _receiveBufferMutex; std::atomic _timeJumped; std::atomic _timeTimelineChanged; std::mutex _latencyMutex; std::deque _latencyDiffs; double _initialTimeDiff = 0.0; std::unique_ptr _receiveThread = nullptr; std::shared_ptr> _connectionEvent; ParallelConnection _connection; TimeManager::CallbackHandle _timeJumpCallback = -1; TimeManager::CallbackHandle _timeTimelineChangeCallback = -1; }; } // namespace openspace #endif // __OPENSPACE_CORE___PARALLELPEER___H__