/* * Copyright (c) 2008-2017, Hazelcast, Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.hazelcast.client.impl; import com.hazelcast.client.ClientEndpoint; import com.hazelcast.client.ClientEngine; import com.hazelcast.core.ClientType; import com.hazelcast.nio.Connection; import com.hazelcast.spi.ExecutionService; import com.hazelcast.spi.properties.GroupProperty; import com.hazelcast.spi.properties.HazelcastProperties; import com.hazelcast.util.Clock; import static com.hazelcast.util.StringUtil.timeToString; import static java.util.concurrent.TimeUnit.SECONDS; /** * Monitors client heartbeats.. As soon as a client has not used its connection for a certain amount of time, * the client is disconnected. */ public class ClientHeartbeatMonitor implements Runnable { private static final int HEART_BEAT_CHECK_INTERVAL_SECONDS = 10; private static final int DEFAULT_CLIENT_HEARTBEAT_TIMEOUT_SECONDS = 60; private final ClientEndpointManagerImpl clientEndpointManager; private final ClientEngine clientEngine; private final long heartbeatTimeoutSeconds; private final ExecutionService executionService; public ClientHeartbeatMonitor(ClientEndpointManagerImpl endpointManager, ClientEngine clientEngine, ExecutionService executionService, HazelcastProperties hazelcastProperties) { this.clientEndpointManager = endpointManager; this.clientEngine = clientEngine; this.executionService = executionService; this.heartbeatTimeoutSeconds = getHeartbeatTimeout(hazelcastProperties); } private long getHeartbeatTimeout(HazelcastProperties hazelcastProperties) { long configuredTimeout = hazelcastProperties.getSeconds(GroupProperty.CLIENT_HEARTBEAT_TIMEOUT_SECONDS); if (configuredTimeout > 0) { return configuredTimeout; } return DEFAULT_CLIENT_HEARTBEAT_TIMEOUT_SECONDS; } public void start() { executionService.scheduleWithRepetition(this, HEART_BEAT_CHECK_INTERVAL_SECONDS, HEART_BEAT_CHECK_INTERVAL_SECONDS, SECONDS); } @Override public void run() { String memberUuid = clientEngine.getThisUuid(); for (ClientEndpoint ce : clientEndpointManager.getEndpoints()) { ClientEndpointImpl clientEndpoint = (ClientEndpointImpl) ce; monitor(memberUuid, clientEndpoint); } } private void monitor(String memberUuid, ClientEndpointImpl clientEndpoint) { if (clientEndpoint.isFirstConnection() && ClientType.CPP.equals(clientEndpoint.getClientType())) { return; } Connection connection = clientEndpoint.getConnection(); long lastTimePacketReceived = connection.lastReadTimeMillis(); long timeoutInMillis = SECONDS.toMillis(heartbeatTimeoutSeconds); long currentTimeMillis = Clock.currentTimeMillis(); if (lastTimePacketReceived + timeoutInMillis < currentTimeMillis) { if (memberUuid.equals(clientEngine.getOwnerUuid(clientEndpoint.getUuid()))) { String message = "Client heartbeat is timed out, closing connection to " + connection + ". Now: " + timeToString(currentTimeMillis) + ". LastTimePacketReceived: " + timeToString(lastTimePacketReceived); connection.close(message, null); } } } }