vldpyatkov commented on code in PR #5221:
URL: https://github.com/apache/ignite-3/pull/5221#discussion_r1987284099


##########
modules/raft/src/main/java/org/apache/ignite/raft/jraft/rpc/impl/core/DefaultRaftClientService.java:
##########
@@ -88,19 +93,78 @@ public Future<Message> requestVote(final PeerId peerId, 
final RequestVoteRequest
 
     @Override
     public Future<Message> appendEntries(final PeerId peerId, final 
AppendEntriesRequest request,
-        final int timeoutMs, final RpcResponseClosure<AppendEntriesResponse> 
done) {
+            final int timeoutMs, final 
RpcResponseClosure<AppendEntriesResponse> done) {
 
         // Assign an executor in round-robin fasion.
         final Executor executor = 
this.appendEntriesExecutorMap.computeIfAbsent(peerId,
-            k -> nodeOptions.getStripedExecutor().next());
+                k -> nodeOptions.getStripedExecutor().next());
 
         if (connect(peerId)) { // Replicator should be started asynchronously 
by node joined event.
+            if (isHeartbeatRequest(request)) {
+                return sendHeartbeat(peerId, request, timeoutMs, done, 
executor);
+            }
+
             return invokeWithDone(peerId, request, done, timeoutMs, executor);
         }
 
         return onConnectionFail(executor, request, done, peerId);
     }
 
+    /**
+     * Accumulates heartbeat messages to send them into the batch request.
+     *
+     * @param peerId Remote peer id.
+     * @param request Request.
+     * @param timeoutMs Timeout.
+     * @param done Done callback.
+     * @param executor Executor where the done callback is executed.
+     * @return A future with response.
+     */
+    private Future<Message> sendHeartbeat(
+            PeerId peerId,
+            AppendEntriesRequest request,
+            int timeoutMs,
+            RpcResponseClosure<AppendEntriesResponse> done,
+            Executor executor
+    ) {
+        NodeManager nodeManager = this.nodeOptions.getNodeManager();
+
+        return invokeWithDone(
+                peerId,
+                request,
+                null,
+                done,
+                timeoutMs,
+                executor,
+                (peerId1, request1, ctx, callback, timeoutMs1) ->
+                        nodeManager.enqueue(peerId, (Message) 
request1).whenComplete((res, err) -> {
+                            if (err instanceof ExecutionException) {
+                                err = new RemotingException(err);
+                            } else if (err instanceof TimeoutException) // 
Translate timeout exception.
+                            {
+                                err = new InvokeTimeoutException();
+                            }
+
+                            Throwable finalErr = err;
+
+                            // Avoid deadlocks if a closure has completed in 
the same thread.
+                            Utils.runInThread(callback.executor(), () -> 
callback.complete(res, finalErr));
+                        })
+        );
+    }
+
+    /**
+     * Determines whether it is a heartbeat request.
+     *
+     * @param request Append entries request.
+     * @return True if that request is heartbeat or false otherwise.
+     */
+    private static boolean isHeartbeatRequest(final AppendEntriesRequest 
request) {

Review Comment:
   The same as AppendEntriesRequestProcessor#isHeartbeatRequest



-- 
This is an automated message from the Apache Git Service.
To respond to the message, please log on to GitHub and use the
URL above to go to the specific comment.

To unsubscribe, e-mail: notifications-unsubscr...@ignite.apache.org

For queries about this service, please contact Infrastructure at:
us...@infra.apache.org

Reply via email to