sidebar = $sidebar; } /** * Handle session lock acquire/refresh (heartbeat). * * @since 0.2.0 * @param WP_REST_Request $request REST request. * @return WP_REST_Response|WP_Error */ public function handle_session_lock( $request ) { $params = $request->get_json_params(); $session_id = sanitize_text_field( $request->get_param( 'session_id' ) ); $tab_id = sanitize_text_field( $params['tab_id'] ?? '' ); $force = ! empty( $params['force'] ); if ( ! $tab_id ) { return new WP_Error( 'missing_tab_id', __( 'tab_id is required.', 'wp-agentic-writer' ), [ 'status' => 400 ], ); } $manager = WP_Agentic_Writer_Conversation_Manager::get_instance(); if ( ! $manager->current_user_can_access( $session_id ) ) { return new WP_Error( 'forbidden', __( 'You do not have permission to access this conversation.', 'wp-agentic-writer', ), [ 'status' => 403 ], ); } // Force mode: use a 0 window so any existing lock is treated as expired. $window = $force ? 0 : 60; $result = $manager->acquire_lock( $session_id, $tab_id, $window ); return new WP_REST_Response( $result, 200 ); } /** * Handle session lock release. * * @since 0.2.0 * @param WP_REST_Request $request REST request. * @return WP_REST_Response|WP_Error */ public function handle_session_unlock( $request ) { $params = $request->get_json_params(); $session_id = sanitize_text_field( $request->get_param( 'session_id' ) ); $tab_id = sanitize_text_field( $params['tab_id'] ?? '' ); if ( ! $tab_id ) { return new WP_Error( 'missing_tab_id', __( 'tab_id is required.', 'wp-agentic-writer' ), [ 'status' => 400 ], ); } $manager = WP_Agentic_Writer_Conversation_Manager::get_instance(); if ( ! $manager->current_user_can_access( $session_id ) ) { return new WP_Error( 'forbidden', __( 'You do not have permission to access this conversation.', 'wp-agentic-writer', ), [ 'status' => 403 ], ); } $released = $manager->release_lock( $session_id, $tab_id ); return new WP_REST_Response( [ 'released' => $released ], 200 ); } }