#ifndef SRC_INSPECTOR_IO_H_ #define SRC_INSPECTOR_IO_H_ #include "inspector_socket_server.h" #include "node_debug_options.h" #include "node_mutex.h" #include "uv.h" #include #include #include #if !HAVE_INSPECTOR #error("This header can only be used when inspector is enabled") #endif // Forward declaration to break recursive dependency chain with src/env.h. namespace node { class Environment; } // namespace node namespace v8_inspector { class StringBuffer; class StringView; } // namespace v8_inspector namespace node { namespace inspector { class InspectorIoDelegate; enum class InspectorAction { kStartSession, kEndSession, kSendMessage }; // kKill closes connections and stops the server, kStop only stops the server enum class TransportAction { kKill, kSendMessage, kStop }; class InspectorIo { public: InspectorIo(node::Environment* env, v8::Platform* platform, const std::string& path, const DebugOptions& options, bool wait_for_connect); ~InspectorIo(); // Start the inspector agent thread bool Start(); // Stop the inspector agent void Stop(); bool IsStarted(); bool IsConnected(); void WaitForDisconnect(); void PostIncomingMessage(InspectorAction action, int session_id, const std::string& message); void ResumeStartup() { uv_sem_post(&start_sem_); } void ServerDone() { uv_close(reinterpret_cast(&io_thread_req_), nullptr); } int port() const { return port_; } private: template using MessageQueue = std::vector>>; enum class State { kNew, kAccepting, kConnected, kDone, kError, kShutDown }; static void ThreadCbIO(void* agent); static void MainThreadAsyncCb(uv_async_t* req); template static void WriteCbIO(uv_async_t* async); template void WorkerRunIO(); void SetConnected(bool connected); void DispatchMessages(); void Write(TransportAction action, int session_id, const v8_inspector::StringView& message); template bool AppendMessage(MessageQueue* vector, ActionType action, int session_id, std::unique_ptr buffer); template void SwapBehindLock(MessageQueue* vector1, MessageQueue* vector2); void WaitForFrontendMessage(); void NotifyMessageReceived(); bool StartThread(bool wait); // Message queues ConditionVariable incoming_message_cond_; const DebugOptions options_; uv_sem_t start_sem_; Mutex state_lock_; uv_thread_t thread_; InspectorIoDelegate* delegate_; State state_; node::Environment* parent_env_; uv_async_t io_thread_req_; // Note that this will live while the async is being closed - likely, past // the parent object lifespan std::pair* main_thread_req_; std::unique_ptr session_delegate_; v8::Platform* platform_; MessageQueue incoming_message_queue_; MessageQueue outgoing_message_queue_; bool dispatching_messages_; int session_id_; std::string script_name_; std::string script_path_; const std::string id_; const bool wait_for_connect_; int port_; friend class DispatchMessagesTask; friend class IoSessionDelegate; friend void InterruptCallback(v8::Isolate*, void* agent); }; std::unique_ptr Utf8ToStringView( const std::string& message); } // namespace inspector } // namespace node #endif // SRC_INSPECTOR_IO_H_