com.palantir.atlasdb.transaction.impl.ConflictDetectionManagers Maven / Gradle / Ivy
/*
* (c) Copyright 2018 Palantir Technologies 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.palantir.atlasdb.transaction.impl;
import com.google.common.collect.Maps;
import com.palantir.atlasdb.keyvalue.api.KeyValueService;
import com.palantir.atlasdb.logging.LoggingArgs;
import com.palantir.atlasdb.table.description.TableMetadata;
import com.palantir.atlasdb.transaction.api.ConflictHandler;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
public final class ConflictDetectionManagers {
private static final SafeLogger log = SafeLoggerFactory.get(ConflictDetectionManagers.class);
private ConflictDetectionManagers() {}
public static ConflictDetectionManager createWithNoConflictDetection() {
return new ConflictDetectionManager(tableReference -> ConflictHandler.IGNORE_ALL);
}
/**
* @deprecated use {@link #create(KeyValueService)} instead. This constructor will be removed in a future release.
*/
@Deprecated
@SuppressWarnings("InlineMeSuggester")
public static ConflictDetectionManager createDefault(KeyValueService kvs) {
return create(kvs);
}
/**
* Creates a ConflictDetectionManager without warming the cache.
*/
public static ConflictDetectionManager createWithoutWarmingCache(KeyValueService kvs) {
return create(kvs, false);
}
/**
* Creates a ConflictDetectionManager and kicks off an asynchronous thread to warm the cache that is used for
* conflict detection.
*/
public static ConflictDetectionManager create(KeyValueService kvs) {
return create(kvs, true);
}
private static ConflictDetectionManager create(KeyValueService kvs, boolean warmCache) {
ConflictDetectionManager conflictDetectionManager = new ConflictDetectionManager(tableReference -> {
byte[] metadata = kvs.getMetadataForTable(tableReference);
if (metadata.length == 0) {
log.error(
"Tried to make a transaction over a table that has no metadata: {}.",
LoggingArgs.tableRef("tableReference", tableReference));
return null;
} else {
return getConflictHandlerFromMetadata(metadata);
}
});
if (warmCache) {
// kick off an async thread that attempts to fully warm this cache
// if it fails (e.g. probably this user has way too many tables), that's okay,
// we will be falling back on individually loading in tables as needed.
new Thread(
() -> {
try {
conflictDetectionManager.warmCacheWith(
Maps.transformValues(kvs.getMetadataForTables(), metadata -> {
if (metadata == null) {
log.debug("Metadata was null for a table. likely because the table"
+ " is currently being created. Skipping warming"
+ " cache for the table.");
return null;
} else {
return getConflictHandlerFromMetadata(metadata);
}
}));
} catch (Throwable t) {
log.warn(
"There was a problem with pre-warming the conflict detection cache; if you"
+ " have unusually high table scale, this might be expected."
+ " Performance may be degraded until normal usage adequately warms"
+ " the cache.",
t);
}
},
"ConflictDetectionManager Cache Async Pre-Warm")
.start();
}
return conflictDetectionManager;
}
private static ConflictHandler getConflictHandlerFromMetadata(byte[] metadata) {
return TableMetadata.BYTES_HYDRATOR.hydrateFromBytes(metadata).getConflictHandler();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy