
com.hazelcast.replicatedmap.impl.ReplicatedMapProxy Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 2008-2024, 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.replicatedmap.impl;
import com.hazelcast.config.ReplicatedMapConfig;
import com.hazelcast.core.EntryListener;
import com.hazelcast.internal.monitor.impl.EmptyLocalReplicatedMapStats;
import com.hazelcast.internal.partition.impl.InternalPartitionServiceImpl;
import com.hazelcast.internal.serialization.SerializationService;
import com.hazelcast.internal.util.IterationType;
import com.hazelcast.internal.util.ResultSet;
import com.hazelcast.map.impl.MapEntries;
import com.hazelcast.internal.serialization.Data;
import com.hazelcast.query.Predicate;
import com.hazelcast.replicatedmap.LocalReplicatedMapStats;
import com.hazelcast.replicatedmap.ReplicatedMap;
import com.hazelcast.replicatedmap.impl.operation.ClearOperationFactory;
import com.hazelcast.replicatedmap.impl.operation.PutAllOperation;
import com.hazelcast.replicatedmap.impl.operation.PutOperation;
import com.hazelcast.replicatedmap.impl.operation.RemoveOperation;
import com.hazelcast.replicatedmap.impl.operation.RequestMapDataOperation;
import com.hazelcast.replicatedmap.impl.operation.VersionResponsePair;
import com.hazelcast.replicatedmap.impl.record.ReplicatedEntryEventFilter;
import com.hazelcast.replicatedmap.impl.record.ReplicatedQueryEventFilter;
import com.hazelcast.replicatedmap.impl.record.ReplicatedRecordStore;
import com.hazelcast.spi.impl.AbstractDistributedObject;
import com.hazelcast.spi.impl.InitializingObject;
import com.hazelcast.spi.impl.InternalCompletableFuture;
import com.hazelcast.spi.impl.NodeEngine;
import com.hazelcast.spi.impl.eventservice.EventFilter;
import com.hazelcast.spi.impl.eventservice.impl.TrueEventFilter;
import com.hazelcast.spi.impl.operationservice.Operation;
import com.hazelcast.spi.impl.operationservice.OperationService;
import com.hazelcast.splitbrainprotection.SplitBrainProtectionOn;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.BitSet;
import java.util.Collection;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import static com.hazelcast.internal.util.ExceptionUtil.rethrow;
import static com.hazelcast.internal.util.Preconditions.checkNotNull;
import static com.hazelcast.internal.util.SetUtil.createHashSet;
import static com.hazelcast.replicatedmap.impl.ReplicatedMapService.SERVICE_NAME;
import static com.hazelcast.splitbrainprotection.SplitBrainProtectionOn.READ;
import static java.lang.Math.ceil;
import static java.lang.Math.log10;
import static java.lang.Thread.currentThread;
/**
* Proxy implementation of {@link ReplicatedMap} interface.
*
* @param key type
* @param value type
*/
@SuppressWarnings({"checkstyle:methodcount", "checkstyle:classfanoutcomplexity"})
public class ReplicatedMapProxy extends AbstractDistributedObject
implements ReplicatedMap, InitializingObject {
private static final String NULL_KEY_IS_NOT_ALLOWED = "Null key is not allowed!";
private static final String NULL_VALUE_IS_NOT_ALLOWED = "Null value is not allowed!";
private static final String NULL_TIMEUNIT_IS_NOT_ALLOWED = "Null time unit is not allowed!";
private static final String NULL_LISTENER_IS_NOT_ALLOWED = "Null listener is not allowed!";
private static final String NULL_PREDICATE_IS_NOT_ALLOWED = "Null predicate is not allowed!";
private static final int WAIT_INTERVAL_MILLIS = 1000;
private static final int RETRY_INTERVAL_COUNT = 3;
private static final int KEY_SET_MIN_SIZE = 16;
private static final int KEY_SET_STORE_MULTIPLE = 4;
private static final int PARALLEL_INIT_REQUESTS_LIMIT = 100;
private static final LocalReplicatedMapStats EMPTY_LOCAL_MAP_STATS = new EmptyLocalReplicatedMapStats();
private final String name;
private final NodeEngine nodeEngine;
private final ReplicatedMapService service;
private final ReplicatedMapEventPublishingService eventPublishingService;
private final SerializationService serializationService;
private final InternalPartitionServiceImpl partitionService;
private final ReplicatedMapConfig config;
ReplicatedMapProxy(NodeEngine nodeEngine, String name, ReplicatedMapService service, ReplicatedMapConfig config) {
super(nodeEngine, service);
this.name = name;
this.nodeEngine = nodeEngine;
this.service = service;
this.eventPublishingService = service.getEventPublishingService();
this.serializationService = nodeEngine.getSerializationService();
this.partitionService = (InternalPartitionServiceImpl) nodeEngine.getPartitionService();
this.config = config;
}
@Override
public void initialize() {
service.initializeListeners(name);
if (nodeEngine.getClusterService().getSize() == 1) {
return;
}
fireMapDataLoadingTasks();
if (!config.isAsyncFillup()) {
syncFill();
}
}
private void syncFill() {
int partitionCount = nodeEngine.getPartitionService().getPartitionCount();
BitSet nonLoadedStores = new BitSet(partitionCount);
int[] retryCount = new int[partitionCount];
for (int i = 0; i < partitionCount; i++) {
nonLoadedStores.set(i);
}
while (true) {
int remainingParallelRequests = PARALLEL_INIT_REQUESTS_LIMIT;
for (int nonLoadedPartition = nonLoadedStores.nextSetBit(0);
nonLoadedPartition >= 0 && remainingParallelRequests > 0;
nonLoadedPartition = nonLoadedStores.nextSetBit(nonLoadedPartition + 1)) {
ReplicatedRecordStore store = service.getReplicatedRecordStore(name, false, nonLoadedPartition);
if (store == null || !store.isLoaded()) {
if ((retryCount[nonLoadedPartition]++) % RETRY_INTERVAL_COUNT == 0) {
requestDataForPartition(nonLoadedPartition);
remainingParallelRequests--;
}
} else {
nonLoadedStores.clear(nonLoadedPartition);
}
}
if (nonLoadedStores.isEmpty()) {
break;
}
sleep();
}
}
private void sleep() {
try {
TimeUnit.MILLISECONDS.sleep(WAIT_INTERVAL_MILLIS);
} catch (InterruptedException e) {
currentThread().interrupt();
throw rethrow(e);
}
}
private void fireMapDataLoadingTasks() {
for (int i = 0; i < nodeEngine.getPartitionService().getPartitionCount(); i++) {
requestDataForPartition(i);
}
}
private void requestDataForPartition(int partitionId) {
RequestMapDataOperation requestMapDataOperation = new RequestMapDataOperation(name);
OperationService operationService = nodeEngine.getOperationService();
operationService
.createInvocationBuilder(SERVICE_NAME, requestMapDataOperation, partitionId)
.setTryCount(ReplicatedMapService.INVOCATION_TRY_COUNT)
.invoke();
}
@Override
protected boolean preDestroy() {
if (super.preDestroy()) {
eventPublishingService.fireMapClearedEvent(size(), name);
return true;
}
return false;
}
@Override
public String getName() {
return name;
}
@Override
public String getPartitionKey() {
return getName();
}
@Override
public String getServiceName() {
return SERVICE_NAME;
}
@Override
public int size() {
ensureNoSplitBrain(READ);
Collection stores = service.getAllReplicatedRecordStores(getName());
int size = 0;
for (ReplicatedRecordStore store : stores) {
size += store.size();
}
return size;
}
@Override
public boolean isEmpty() {
ensureNoSplitBrain(READ);
Collection stores = service.getAllReplicatedRecordStores(getName());
for (ReplicatedRecordStore store : stores) {
if (!store.isEmpty()) {
return false;
}
}
return true;
}
@Override
public boolean containsKey(@Nonnull Object key) {
ensureNoSplitBrain(READ);
checkNotNull(key, NULL_KEY_IS_NOT_ALLOWED);
int partitionId = partitionService.getPartitionId(key);
ReplicatedRecordStore store = service.getReplicatedRecordStore(name, false, partitionId);
return store != null && store.containsKey(key);
}
@Override
public boolean containsValue(@Nonnull Object value) {
ensureNoSplitBrain(READ);
checkNotNull(value, NULL_VALUE_IS_NOT_ALLOWED);
Collection stores = service.getAllReplicatedRecordStores(getName());
for (ReplicatedRecordStore store : stores) {
if (store.containsValue(value)) {
return true;
}
}
return false;
}
@Override
public V get(@Nonnull Object key) {
ensureNoSplitBrain(READ);
checkNotNull(key, NULL_KEY_IS_NOT_ALLOWED);
int partitionId = partitionService.getPartitionId(key);
ReplicatedRecordStore store = service.getReplicatedRecordStore(getName(), false, partitionId);
if (store == null) {
return null;
}
return (V) store.get(key);
}
@Override
public V put(@Nonnull K key, @Nonnull V value) {
checkNotNull(key, NULL_KEY_IS_NOT_ALLOWED);
checkNotNull(value, NULL_VALUE_IS_NOT_ALLOWED);
Data dataKey = nodeEngine.toData(key);
Data dataValue = nodeEngine.toData(value);
int partitionId = nodeEngine.getPartitionService().getPartitionId(dataKey);
PutOperation putOperation = new PutOperation(getName(), dataKey, dataValue);
InternalCompletableFuture
© 2015 - 2025 Weber Informatics LLC | Privacy Policy