All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.apple.foundationdb.directory.DirectoryLayer Maven / Gradle / Ivy

Go to download

Java bindings for the FoundationDB database. These bindings require the FoundationDB client, which is under a different license. The client can be obtained from https://www.foundationdb.org/download/.

There is a newer version: 7.3.47
Show newest version
/*
 * DirectoryLayer.java
 *
 * This source file is part of the FoundationDB open source project
 *
 * Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
 *
 * 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.apple.foundationdb.directory;

import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Random;
import java.util.concurrent.CompletableFuture;
import java.util.function.Function;

import com.apple.foundationdb.KeyValue;
import com.apple.foundationdb.MutationType;
import com.apple.foundationdb.Range;
import com.apple.foundationdb.ReadTransaction;
import com.apple.foundationdb.ReadTransactionContext;
import com.apple.foundationdb.Transaction;
import com.apple.foundationdb.TransactionContext;
import com.apple.foundationdb.async.AsyncIterable;
import com.apple.foundationdb.async.AsyncIterator;
import com.apple.foundationdb.async.AsyncUtil;
import com.apple.foundationdb.subspace.Subspace;
import com.apple.foundationdb.tuple.ByteArrayUtil;
import com.apple.foundationdb.tuple.Tuple;

/**
 * Provides a class for managing directories in FoundationDB.
 *
 * 

* The FoundationDB API provides directories as a tool for managing related * {@link Subspace}s. Directories are a recommended approach for administering * applications. Each application should create or open at least one directory * to manage its subspaces. * * For general guidance on directory usage, see the discussion in the * Developer Guide. *

*

* Directories are identified by hierarchical paths analogous to the paths * in a Unix-like file system. A path is represented as a List of strings. * Each directory has an associated subspace used to store its content. The * layer maps each path to a short prefix used for the corresponding * subspace. In effect, directories provide a level of indirection for * access to subspaces. *

*/ public class DirectoryLayer implements Directory { private static final Charset UTF_8 = Charset.forName("UTF-8"); private static final byte[] LITTLE_ENDIAN_LONG_ONE = { 1, 0, 0, 0, 0, 0, 0, 0 }; private static final byte[] HIGH_CONTENTION_KEY = "hca".getBytes(UTF_8); private static final byte[] LAYER_KEY = "layer".getBytes(UTF_8); private static final byte[] VERSION_KEY = "version".getBytes(UTF_8); private static final long SUB_DIR_KEY = 0; private static final Integer[] VERSION = { 1, 0, 0 }; static final byte[] EMPTY_BYTES = new byte[0]; static final List EMPTY_PATH = Collections.emptyList(); static final byte[] DEFAULT_NODE_SUBSPACE_PREFIX = { (byte)0xFE }; /** * The default node {@link Subspace} used by a {@code DirectoryLayer} when none is specified. */ public static final Subspace DEFAULT_NODE_SUBSPACE = new Subspace(DEFAULT_NODE_SUBSPACE_PREFIX); /** * The default content {@link Subspace} used by a {@code DirectoryLayer} when none is specified. */ public static final Subspace DEFAULT_CONTENT_SUBSPACE = new Subspace(); private final Subspace rootNode; private final Subspace nodeSubspace; private final Subspace contentSubspace; private final HighContentionAllocator allocator; private final boolean allowManualPrefixes; private List path = EMPTY_PATH; /** * The layer string to pass to {@link Directory#createOrOpen(TransactionContext, List, byte[])} or * {@link Directory#create(TransactionContext, List, byte[])} to create a {@code DirectoryPartition}. */ public static final byte[] PARTITION_LAYER = "partition".getBytes(Charset.forName("UTF-8")); private static DirectoryLayer defaultDirectoryLayer = new DirectoryLayer(); /** * Constructor for a {@code DirectoryLayer} formed with default node and * content subspaces. A {@code DirectoryLayer} * defines a new root directory. The node subspace and content subspace * control where the directory metadata and contents, respectively, are * stored. The default root directory has a node subspace with raw prefix * {@code \xFE} and a content subspace with no prefix. * Prefixes can not be specified in calls to {@link Directory#create(TransactionContext, List, byte[], byte[])}. * * @see #getDefault */ public DirectoryLayer() { this(DEFAULT_NODE_SUBSPACE, DEFAULT_CONTENT_SUBSPACE, false); } /** * Constructor for a {@code DirectoryLayer} formed with default node and * content subspaces. A {@code DirectoryLayer} * defines a new root directory. The node subspace and content subspace * control where the directory metadata and contents, respectively, are * stored. The default root directory has a node subspace with raw prefix * {@code \xFE} and a content subspace with no prefix. * * @param allowManualPrefixes whether or not prefixes can be specified in calls to * {@link Directory#create(TransactionContext, List, byte[], byte[])} */ public DirectoryLayer(boolean allowManualPrefixes) { this(DEFAULT_NODE_SUBSPACE, DEFAULT_CONTENT_SUBSPACE, allowManualPrefixes); } /** * Constructor for a {@code DirectoryLayer} formed with a specified node * subspace and specified content subspace. A {@code DirectoryLayer} * defines a new root directory. The node subspace and content subspace * control where the directory metadata and contents, respectively, are * stored. The default root directory has a node subspace with raw prefix * {@code \xFE} and a content subspace with no prefix. Specifying more * restrictive values for the node subspace and content subspace will allow * using the directory layer alongside other content in a database. * Prefixes can not be specified in calls to {@link Directory#create(TransactionContext, List, byte[], byte[])}. * * @param nodeSubspace a {@link Subspace} used to store directory metadata * @param contentSubspace a {@link Subspace} used to store directory content */ public DirectoryLayer(Subspace nodeSubspace, Subspace contentSubspace) { this(nodeSubspace, contentSubspace, false); } /** * Constructor for a {@code DirectoryLayer} formed with a specified node * subspace and specified content subspace. A {@code DirectoryLayer} * defines a new root directory. The node subspace and content subspace * control where the directory metadata and contents, respectively, are * stored. The default root directory has a node subspace with raw prefix * {@code \xFE} and a content subspace with no prefix. Specifying more * restrictive values for the node subspace and content subspace will allow * using the directory layer alongside other content in a database. * * @param nodeSubspace a {@link Subspace} used to store directory metadata * @param contentSubspace a {@link Subspace} used to store directory content * @param allowManualPrefixes whether or not prefixes can be specified in calls to * {@link Directory#create(TransactionContext, List, byte[], byte[])} */ public DirectoryLayer(Subspace nodeSubspace, Subspace contentSubspace, boolean allowManualPrefixes) { this.nodeSubspace = nodeSubspace; this.contentSubspace = contentSubspace; // The root node is the one whose contents are the node subspace this.rootNode = nodeSubspace.get(nodeSubspace.getKey()); this.allocator = new HighContentionAllocator(rootNode.get(HIGH_CONTENTION_KEY)); this.allowManualPrefixes = allowManualPrefixes; } /** * Creates a new {@code DirectoryLayer} formed with a specified node subspace and default content subspace. * Prefixes can not be specified in calls to {@link Directory#create(TransactionContext, List, byte[], byte[])}. * * @param nodeSubspace a {@link Subspace} used to store directory metadata * @return a {@code DirectoryLayer} formed with {@code nodeSubspace} and a default content subspace */ public static Directory createWithNodeSubspace(Subspace nodeSubspace) { return new DirectoryLayer(nodeSubspace, DEFAULT_CONTENT_SUBSPACE); } /** * Creates a new {@code DirectoryLayer} formed with a default node subspace and specified content subspace. * Prefixes can not be specified in calls to {@link Directory#create(TransactionContext, List, byte[], byte[])}. * * @param contentSubspace a {@link Subspace} used to store directory content * @return a {@code DirectoryLayer} formed with a {@code contentSubspace} and a default node subspace */ public static Directory createWithContentSubspace(Subspace contentSubspace) { return new DirectoryLayer(DEFAULT_NODE_SUBSPACE, contentSubspace); } /** * Gets the default instance of the DirectoryLayer. The default instance * is created with the default node and content subspaces. * * Prefixes can not be specified in calls to {@link Directory#create(TransactionContext, List, byte[], byte[])}. * * @return the default {@code DirectoryLayer} */ public static DirectoryLayer getDefault() { return defaultDirectoryLayer; } /** * Tests whether this {@code DirectoryLayer} is equal to {@code rhs}. * Two {@code DirectoryLayer}s are equal if they have the same node subspace, * content subspace, and path. * * @param rhs the object to check for equality * @return {@code true} if this {@code DirectoryLayer} and {@code rhs} are equal */ @Override public boolean equals(Object rhs) { if(this == rhs) { return true; } if(rhs == null || getClass() != rhs.getClass()) { return false; } DirectoryLayer other = (DirectoryLayer)rhs; return (path == other.path || path.equals(other.path)) && nodeSubspace.equals(other.nodeSubspace) && contentSubspace.equals(other.contentSubspace); } /** * Produces a hash of this {@code DirectoryLayer} based on its path and subspaces. * This satisfies the necessary requirements to allow this class to be used as keys * in hash tables or as values in hash-based sets. * * @return a hash based on the path and subspaces of this {@code DirectoryLayer} */ @Override public int hashCode() { return path.hashCode() ^ (nodeSubspace.hashCode() * 179) ^ (contentSubspace.hashCode() * 937); } /** * Sets path of directory to {@code path} * * @param path a {@code List} specifying a path */ void setPath(List path) { this.path = path; } /** * Gets the path for the root node of this {@code DirectoryLayer}. Normally constructed * {@code DirectoryLayer}s have an empty path, but {@code DirectoryLayer}s returned by * {@link Directory#getDirectoryLayer} for {@link Directory}s inside of a partition * could have non-empty paths. * * @return the path for the root node of this {@code DirectoryLayer} */ @Override public List getPath() { return Collections.unmodifiableList(path); } /** * Returns the layer byte string for this {@code DirectoryLayer}, which is always an empty * array. * * @return an empty byte array */ @Override public byte[] getLayer() { return EMPTY_BYTES; } /** * Returns {@code this}. * * @return {@code this} */ @Override public DirectoryLayer getDirectoryLayer() { return this; } /** * Creates or opens the directory located at {@code path}(creating parent directories, if necessary). * If the directory is new, then the {@code layer} byte string will be recorded as its layer. * If the directory already exists, the {@code layer} byte string will be compared against the {@code layer} * set when the directory was created. * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link MismatchedLayerException} - if the directory has already been created with a different {@code layer} byte string
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path * @param layer a {@code byte[]} specifying a layer to set on a new directory or check for on an existing directory * @return a {@link CompletableFuture} which will be set to the created or opened {@link DirectorySubspace} */ @Override public CompletableFuture createOrOpen(TransactionContext tcx, final List path, final byte[] layer) { return tcx.runAsync(tr -> createOrOpenInternal(tr, tr, path, layer, null, true, true)); } /** * Opens the directory located at {@code path}. * The {@code layer} byte string will be compared against the {@code layer} set when * the directory was created. * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link MismatchedLayerException} - if the directory was created with a different {@code layer} byte string
  • *
  • {@link NoSuchDirectoryException} - if the directory does not exist
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path * @param layer a {@code byte[]} specifying the expected layer * @return a {@link CompletableFuture} which will be set to the opened {@link DirectorySubspace} */ @Override public CompletableFuture open(ReadTransactionContext tcx, final List path, final byte[] layer) { return tcx.readAsync(rtr -> createOrOpenInternal(rtr, null, path, layer, null, false, true)); } /** * Creates a directory located at {@code path} (creating parent directories if necessary). * The {@code layer} byte string will be recorded as the new directory's layer and checked by * future calls to {@link #open(ReadTransactionContext, List, byte[])}. The specified {@code prefix} * will be used for this directory's contents instead of allocating a prefix automatically. * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link DirectoryAlreadyExistsException} - if the given directory already exists
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path of this {@code Directory} * @param layer a {@code byte[]} specifying a layer to set for the directory * @param prefix a {@code byte[]} specifying the key prefix to use for the directory's contents * @return a {@link CompletableFuture} which will be set to the created {@link DirectorySubspace} */ @Override public CompletableFuture create(TransactionContext tcx, final List path, final byte[] layer, final byte[] prefix) { return tcx.runAsync(tr -> createOrOpenInternal(tr, tr, path, layer, prefix, true, false)); } /** * This method should not be called on a {@code DirectoryLayer}. Calling this method will result in the returned * {@link CompletableFuture} being set to a {@link DirectoryMoveException}. * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link DirectoryMoveException}
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param newAbsolutePath a {@code List} specifying a path * @return a {@link CompletableFuture} which will be set to a {@link DirectoryMoveException} */ @Override public CompletableFuture moveTo(TransactionContext tcx, List newAbsolutePath) { CompletableFuture future = new CompletableFuture<>(); future.completeExceptionally(new DirectoryMoveException("The root directory cannot be moved.", path, newAbsolutePath)); return future; } /** * Moves the directory located at {@code oldPath} to {@code newPath}. * *

* There is no effect on the physical prefix of the given directory, or on clients that already * have the directory open. *

*

* It is invalid to move a directory to: *

*
    *
  • A location where a directory already exists
  • *
  • A location whose parent does not exist
  • *
  • A subdirectory of itself
  • *
  • A different partition
  • *
* * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link NoSuchDirectoryException} - if no {@code Directory} exists at {@code oldPath}
  • *
  • {@link DirectoryAlreadyExistsException} - if a directory already exists at {@code newPath}
  • *
  • {@link DirectoryMoveException} - if an invalid move location is specified
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param oldPath a {@code List} specifying the path of the directory to move * @param newPath a {@code List} specifying the path to move to * @return a {@link CompletableFuture} which will be set to the {@link DirectorySubspace} for this {@code Directory} * at its new location. */ @Override public CompletableFuture move(final TransactionContext tcx, final List oldPath, final List newPath) { final List oldPathCopy = new ArrayList<>(oldPath); final List newPathCopy = new ArrayList<>(newPath); return tcx.runAsync(tr -> checkOrWriteVersion(tr).thenComposeAsync(ignore -> { if(oldPathCopy.size() <= newPathCopy.size() && oldPathCopy.equals(newPathCopy.subList(0, oldPathCopy.size()))) throw new DirectoryMoveException("The destination directory cannot be a subdirectory of the source directory.", toAbsolutePath(oldPathCopy), toAbsolutePath(newPathCopy)); ArrayList> futures = new ArrayList<>(); futures.add(new NodeFinder(oldPathCopy).find(tr).thenComposeAsync(new NodeMetadataLoader(tr), tr.getExecutor())); futures.add(new NodeFinder(newPathCopy).find(tr).thenComposeAsync(new NodeMetadataLoader(tr), tr.getExecutor())); return AsyncUtil.getAll(futures); }, tr.getExecutor()) .thenComposeAsync(nodes -> { final Node oldNode = nodes.get(0); final Node newNode = nodes.get(1); if(!oldNode.exists()) throw new NoSuchDirectoryException(toAbsolutePath(oldPathCopy)); if(oldNode.isInPartition(false) || newNode.isInPartition(false)) { if(!oldNode.isInPartition(false) || !newNode.isInPartition(false) || !oldNode.path.equals(newNode.path)) throw new DirectoryMoveException("Cannot move between partitions.", toAbsolutePath(oldPathCopy), toAbsolutePath(newPathCopy)); return newNode.getContents().move(tr, oldNode.getPartitionSubpath(), newNode.getPartitionSubpath()); } if(newNode.exists()) throw new DirectoryAlreadyExistsException(toAbsolutePath(newPathCopy)); final List parentPath = PathUtil.popBack(newPathCopy); return new NodeFinder(parentPath).find(tr).thenComposeAsync(parentNode -> { if(!parentNode.exists()) throw new NoSuchDirectoryException(toAbsolutePath(parentPath)); tr.set( parentNode.subspace.get(SUB_DIR_KEY).get(getLast(newPathCopy)).getKey(), contentsOfNode(oldNode.subspace, EMPTY_PATH, EMPTY_BYTES).getKey() ); return removeFromParent(tr, oldPathCopy) .thenApply(ignore -> contentsOfNode(oldNode.subspace, newPathCopy, oldNode.layer)); }, tr.getExecutor()); }, tr.getExecutor())); } /** * Removes the directory located at {@code path} and all of its subdirectories, * as well as all of their contents. * *

* Warning: Clients that have already opened the directory might * still insert data into its contents after it is removed. *

* *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link NoSuchDirectoryException} - if no directory exists at {@code path}
  • *
* * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path * @return a {@link CompletableFuture} which will be set once the {@code Directory} has been removed */ @Override public CompletableFuture remove(TransactionContext tcx, List path) { return AsyncUtil.success(removeInternal(tcx, path, true)); } /** * Removes the directory located at {@code subpath} and all of its subdirectories, * as well as all of their contents. * *

* Warning: Clients that have already opened the directory might * still insert data into its contents after it is removed. *

* * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path * @return a {@link CompletableFuture} which will be set to true once the {@code Directory} has been removed, * or false if it didn't exist. */ @Override public CompletableFuture removeIfExists(TransactionContext tcx, List path) { return removeInternal(tcx, path, false); } /** * List the subdirectories of the directory at a given {@code path}. * *

The returned {@link CompletableFuture} can be set to the following errors:

*
    *
  • {@link NoSuchDirectoryException} - if no directory exists at {@code path}
  • *
* * @param tcx the {@link ReadTransactionContext} to execute this operation in * @param path a {@code List} specifying a path * @return a {@link CompletableFuture} which will be set to a {@code List} of names of the subdirectories * of the directory at {@code path}. Each name is a unicode string representing the last component * of a subdirectory's path. */ @Override public CompletableFuture> list(final ReadTransactionContext tcx, final List path) { final List pathCopy = new ArrayList(path); return tcx.readAsync(tr -> checkVersion(tr) .thenComposeAsync(ignore -> new NodeFinder(pathCopy).find(tr).thenComposeAsync(new NodeMetadataLoader(tr), tr.getExecutor()), tr.getExecutor()) .thenComposeAsync(node -> { if(!node.exists()) throw new NoSuchDirectoryException(toAbsolutePath(pathCopy)); if(node.isInPartition(true)) return node.getContents().list(tr, node.getPartitionSubpath()); final Subspace subdir = node.subspace.get(SUB_DIR_KEY); return AsyncUtil.collect( AsyncUtil.mapIterable(tr.getRange(subdir.range()), kv -> subdir.unpack(kv.getKey()).getString(0) ), tr.getExecutor() ); }, tr.getExecutor()) ); } /** * Returns {@code true}. * * @param tcx the {@link TransactionContext} to execute this operation in * @return {@code true} */ @Override public CompletableFuture exists(ReadTransactionContext tcx) { return AsyncUtil.READY_TRUE; } /** * Checks if the directory located at {@code path} exists. * * @param tcx the {@link TransactionContext} to execute this operation in * @param path a {@code List} specifying a path of this {@code Directory} * @return a {@link CompletableFuture} which will be set to {@code true} if the specified directory exists, or {@code false} if it * doesn't */ @Override public CompletableFuture exists(final ReadTransactionContext tcx, final List path) { final List pathCopy = new ArrayList<>(path); return tcx.readAsync(tr -> checkVersion(tr) .thenComposeAsync(ignore -> new NodeFinder(pathCopy).find(tr).thenComposeAsync(new NodeMetadataLoader(tr), tr.getExecutor()), tr.getExecutor()) .thenComposeAsync(node -> { if(!node.exists()) return AsyncUtil.READY_FALSE; else if(node.isInPartition(false)) return node.getContents().exists(tr, node.getPartitionSubpath()); return AsyncUtil.READY_TRUE; }, tr.getExecutor())); } // // Internal // private Subspace nodeWithPrefix(byte[] prefix) { if(prefix == null) { return null; } return nodeSubspace.get(prefix); } private CompletableFuture nodeContainingKey(final ReadTransaction tr, final byte[] key) { // Right now this is only used for _is_prefix_free(), but if we add // parent pointers to directory nodes, it could also be used to find a // path based on a key. if(ByteArrayUtil.startsWith(key, nodeSubspace.getKey())) { return CompletableFuture.completedFuture(rootNode); } return tr.getRange(nodeSubspace.range().begin, ByteArrayUtil.join(nodeSubspace.pack(key), new byte[]{0x00}), 1, true) .asList() .thenApply(results -> { if(results.size() > 0) { byte[] resultKey = results.get(0).getKey(); byte[] prevPrefix = nodeSubspace.unpack(resultKey).getBytes(0); if(ByteArrayUtil.startsWith(key, prevPrefix)) { return nodeWithPrefix(prevPrefix); } } return null; }); } private List toAbsolutePath(List subPath) { return PathUtil.join(path, subPath); } private DirectorySubspace contentsOfNode(Subspace node, List path, byte[] layer) { byte[] prefix = nodeSubspace.unpack(node.getKey()).getBytes(0); if(Arrays.equals(layer, DirectoryLayer.PARTITION_LAYER)) return new DirectoryPartition(toAbsolutePath(path), prefix, this); else return new DirectorySubspace(toAbsolutePath(path), prefix, this, layer); } private CompletableFuture removeInternal(final TransactionContext tcx, final List path, final boolean mustExist) { final List pathCopy = new ArrayList<>(path); return tcx.runAsync(tr -> checkOrWriteVersion(tr).thenComposeAsync(ignore -> { if(pathCopy.size() == 0) throw new DirectoryException("The root directory cannot be removed.", toAbsolutePath(pathCopy)); return new NodeFinder(pathCopy).find(tr).thenComposeAsync(new NodeMetadataLoader(tr), tr.getExecutor()); }, tr.getExecutor()) .thenComposeAsync(node -> { if(!node.exists()) { if(mustExist) throw new NoSuchDirectoryException(toAbsolutePath(pathCopy)); else return AsyncUtil.READY_FALSE; } if(node.isInPartition(false)) return node.getContents().getDirectoryLayer().removeInternal(tr, node.getPartitionSubpath(), mustExist); else { ArrayList> futures = new ArrayList<>(); futures.add(removeRecursive(tr, node.subspace)); futures.add(removeFromParent(tr, pathCopy)); return AsyncUtil.tag(AsyncUtil.whenAll(futures), true); } }, tr.getExecutor())); } private CompletableFuture removeFromParent(final Transaction tr, final List path) { return new NodeFinder(PathUtil.popBack(path)).find(tr) .thenAccept(parent -> tr.clear(parent.subspace.get(SUB_DIR_KEY).get(getLast(path)).getKey())); } private CompletableFuture removeRecursive(final Transaction tr, final Subspace node) { Subspace subdir = node.get(SUB_DIR_KEY); final AsyncIterator rangeItr = tr.getRange(subdir.range()).iterator(); tr.clear(Range.startsWith(nodeSubspace.unpack(node.getKey()).getBytes(0))); tr.clear(node.range()); return AsyncUtil.whileTrue(() -> { CompletableFuture subdirRemoveFuture; if(rangeItr.onHasNext().isDone() && rangeItr.hasNext()) subdirRemoveFuture = removeRecursive(tr, nodeWithPrefix(rangeItr.next().getValue())); else subdirRemoveFuture = AsyncUtil.DONE; return subdirRemoveFuture.thenCompose(ignore -> rangeItr.onHasNext()); }, tr.getExecutor()); } private CompletableFuture isPrefixFree(final ReadTransaction tr, final byte[] prefix) { // Returns true if the given prefix does not "intersect" any currently // allocated prefix (including the root node). This means that it neither // contains any other prefix nor is contained by any other prefix. if(prefix == null || prefix.length == 0) return AsyncUtil.READY_FALSE; return nodeContainingKey(tr, prefix).thenComposeAsync(node -> { if(node != null) return AsyncUtil.READY_FALSE; final AsyncIterator it = tr.getRange(nodeSubspace.pack(prefix), nodeSubspace.pack(ByteArrayUtil.strinc(prefix)), 1).iterator(); return it.onHasNext().thenApply(hasNext -> !hasNext); }, tr.getExecutor()); } private CompletableFuture getVersionValue(final ReadTransaction tr) { return tr.get(rootNode.pack(VERSION_KEY)); } private CompletableFuture checkOrWriteVersion(final Transaction tr) { return getVersionValue(tr).thenApply(new WritableVersionCheck(tr)); } private CompletableFuture checkVersion(final ReadTransaction tr) { return getVersionValue(tr).thenApply(new VersionCheck()); } private CompletableFuture createOrOpenInternal(final ReadTransaction rtr, final Transaction tr, final List path, final byte[] layer, final byte[] prefix, final boolean allowCreate, final boolean allowOpen) { final List pathCopy = new ArrayList<>(path); if(prefix != null && !allowManualPrefixes) { String errorMessage; if(this.path.size() == 0) errorMessage = "Cannot specify a prefix unless manual prefixes are enabled."; else errorMessage = "Cannot specify a prefix in a partition."; CompletableFuture future = new CompletableFuture<>(); future.completeExceptionally(new IllegalArgumentException(errorMessage)); return future; } return checkVersion(rtr).thenComposeAsync(ignore -> { // Root directory contains node metadata and so may not be opened. if(pathCopy.size() == 0) { throw new IllegalArgumentException("The root directory may not be opened."); } return new NodeFinder(pathCopy).find(rtr).thenComposeAsync(new NodeMetadataLoader(rtr), rtr.getExecutor()); }, rtr.getExecutor()) .thenComposeAsync(existingNode -> { if(existingNode.exists()) { if(existingNode.isInPartition(false)) { List subpath = existingNode.getPartitionSubpath(); DirectoryLayer directoryLayer = existingNode.getContents().getDirectoryLayer(); return directoryLayer.createOrOpenInternal( rtr, tr, subpath, layer, prefix, allowCreate, allowOpen); } DirectorySubspace opened = openInternal(pathCopy, layer, existingNode, allowOpen); return CompletableFuture.completedFuture(opened); } else return createInternal(tr, pathCopy, layer, prefix, allowCreate); }, rtr.getExecutor()); } private DirectorySubspace openInternal(final List path, final byte[] layer, final Node existingNode, final boolean allowOpen) { if(!allowOpen) { throw new DirectoryAlreadyExistsException(toAbsolutePath(path)); } else { if(layer.length > 0 && !Arrays.equals(layer, existingNode.layer)) { throw new MismatchedLayerException(toAbsolutePath(path), existingNode.layer, layer); } return existingNode.getContents(); } } private CompletableFuture createInternal(final Transaction tr, final List path, final byte[] layer, final byte[] prefix, final boolean allowCreate) { if(!allowCreate) { throw new NoSuchDirectoryException(toAbsolutePath(path)); } return checkOrWriteVersion(tr).thenComposeAsync(ignore -> { if(prefix == null) { return allocator.allocate(tr).thenComposeAsync(allocated -> { final byte[] finalPrefix = ByteArrayUtil.join(contentSubspace.getKey(), allocated); return tr.getRange(Range.startsWith(finalPrefix), 1).iterator().onHasNext().thenApply(hasAny -> { if(hasAny) { throw new IllegalStateException("The database has keys stored at the prefix chosen by the automatic " + "prefix allocator: " + ByteArrayUtil.printable(finalPrefix) + "."); } return finalPrefix; }); }, tr.getExecutor()); } else return CompletableFuture.completedFuture(prefix); }, tr.getExecutor()) .thenComposeAsync(actualPrefix -> isPrefixFree(prefix == null ? tr.snapshot() : tr, actualPrefix) .thenComposeAsync(prefixFree -> { if(!prefixFree) { if(prefix == null) { throw new IllegalStateException("The directory layer has manually allocated prefixes that conflict " + "with the automatic prefix allocator."); } else throw new IllegalArgumentException("Prefix already in use: " + ByteArrayUtil.printable(actualPrefix) + "."); } else if(path.size() > 1) { return createOrOpen(tr, PathUtil.popBack(path)).thenApply(dir -> nodeWithPrefix(dir.getKey())); } else return CompletableFuture.completedFuture(rootNode); }, tr.getExecutor()) .thenApplyAsync(parentNode -> { if(parentNode == null) throw new IllegalStateException("The parent directory does not exist."); //Shouldn't happen Subspace node = nodeWithPrefix(actualPrefix); tr.set(parentNode.get(SUB_DIR_KEY).get(getLast(path)).getKey(), actualPrefix); tr.set(node.get(LAYER_KEY).getKey(), layer); return contentsOfNode(node, path, layer); }, tr.getExecutor()), tr.getExecutor()); } // // Helpers // private static long unpackLittleEndian(byte[] bytes) { assert bytes.length == 8; long value = 0; for(int i = 0; i < 8; ++i) { value += (Byte.toUnsignedLong(bytes[i]) << (i * 8)); } return value; } private static String getLast(List list) { assert list.size() > 0; return list.get(list.size() - 1); } private class VersionCheck implements Function { @Override public Void apply(byte[] versionBytes) { if(versionBytes == null) { return null; } ByteBuffer versionBuf = ByteBuffer.wrap(versionBytes); versionBuf.order(ByteOrder.LITTLE_ENDIAN); Integer[] version = new Integer[3]; for(int i = 0; i < version.length; ++i) version[i] = versionBuf.getInt(); String dirVersion = String.format("version %d.%d.%d", (Object[])version); String layerVersion = String.format("directory layer %d.%d.%d", (Object[])VERSION); throwOnError(version, dirVersion, layerVersion); return null; } protected void throwOnError(Integer[] version, String dirVersion, String layerVersion) { if(version[0] > VERSION[0]) throw new DirectoryVersionException("Cannot load directory with " + dirVersion + " using " + layerVersion + "."); } } private class WritableVersionCheck extends VersionCheck { private final Transaction tr; private WritableVersionCheck(Transaction tr) { this.tr = tr; } @Override public Void apply(byte[] versionBytes) { if(versionBytes == null) { // initializeVersion ByteBuffer buf = ByteBuffer.allocate(VERSION.length * 4); buf.order(ByteOrder.LITTLE_ENDIAN); for(int ver : VERSION) buf.putInt(ver); tr.set(rootNode.pack(VERSION_KEY), buf.array()); return null; } return super.apply(versionBytes); } @Override protected void throwOnError(Integer[] version, String dirVersion, String layerVersion) { super.throwOnError(version, dirVersion, layerVersion); if(version[1] > VERSION[1]) throw new DirectoryVersionException("Directory with " + dirVersion + " is read-only when opened with " + layerVersion + "."); } } private class NodeFinder { private List path; private int index; private Node node; private List currentPath; NodeFinder(List path) { this.path = path; } public CompletableFuture find(final ReadTransaction tr) { index = 0; node = new Node(rootNode, currentPath, path); currentPath = new ArrayList<>(); return AsyncUtil.whileTrue(() -> { if(index == path.size()) return AsyncUtil.READY_FALSE; return tr.get(node.subspace.get(SUB_DIR_KEY).get(path.get(index)).getKey()).thenComposeAsync(key -> { currentPath.add(path.get(index)); node = new Node(nodeWithPrefix(key), currentPath, path); if(!node.exists()) return AsyncUtil.READY_FALSE; return node.loadMetadata(tr).thenApply(ignore -> { ++index; return !Arrays.equals(node.layer, DirectoryLayer.PARTITION_LAYER); }); }, tr.getExecutor()); }, tr.getExecutor()) .thenApply(ignore -> node); } } private static class NodeMetadataLoader implements Function> { private final ReadTransaction tr; NodeMetadataLoader(ReadTransaction tr) { this.tr = tr; } @Override public CompletableFuture apply(Node node) { return node.loadMetadata(tr); } } private class Node { public final Subspace subspace; public final List path; public final List targetPath; public byte[] layer; private boolean loadedMetadata; Node(Subspace subspace, List path, List targetPath) { this.subspace = subspace; this.path = path; this.targetPath = targetPath; layer = null; loadedMetadata = false; } public boolean exists() { return subspace != null; } public CompletableFuture loadMetadata(ReadTransaction tr) { if(!exists()) { loadedMetadata = true; return CompletableFuture.completedFuture(this); } return tr.get(subspace.pack(new Tuple().add(LAYER_KEY))) .thenApply(value -> { layer = value; loadedMetadata = true; return Node.this; }); } public void ensureMetadataLoaded() { if(!loadedMetadata) throw new IllegalStateException("Metadata for node has not been loaded"); } public boolean isInPartition(boolean includeEmptySubpath) { ensureMetadataLoaded(); return exists() && Arrays.equals(layer, DirectoryLayer.PARTITION_LAYER) && (includeEmptySubpath || targetPath.size() > path.size()); } public List getPartitionSubpath() { ensureMetadataLoaded(); return targetPath.subList(path.size(), targetPath.size()); } public DirectorySubspace getContents() { ensureMetadataLoaded(); return contentsOfNode(subspace, path, layer); } } private static class PrefixFinder { private final Random random; private long windowStart; private int windowSize; private long candidate; private boolean restart; PrefixFinder() { this.random = new Random(); this.windowStart = 0; } public CompletableFuture find(final Transaction tr, final HighContentionAllocator allocator) { return AsyncUtil.whileTrue(() -> { final AsyncIterator rangeItr = tr.snapshot().getRange(allocator.counters.range(), 1, true).iterator(); return rangeItr.onHasNext().thenApply(hasNext -> { if(hasNext) { KeyValue kv = rangeItr.next(); windowStart = allocator.counters.unpack(kv.getKey()).getLong(0); } return null; }) .thenComposeAsync(ignore -> chooseWindow(tr, allocator), tr.getExecutor()) .thenComposeAsync(ignore -> choosePrefix(tr, allocator), tr.getExecutor()); }, tr.getExecutor()) .thenApply(ignore -> Tuple.from(candidate).pack()); } public CompletableFuture chooseWindow(final Transaction tr, final HighContentionAllocator allocator) { final long initialWindowStart = windowStart; return AsyncUtil.whileTrue(() -> { final byte[] counterKey = allocator.counters.get(windowStart).getKey(); Range oldCounters = new Range(allocator.counters.getKey(), counterKey); Range oldAllocations = new Range(allocator.recent.getKey(), allocator.recent.get(windowStart).getKey()); CompletableFuture newCountRead; // SOMEDAY: this code would work as written if synchronized on something transaction local. // The reason we don't synchronize on the Transaction tr itself is that the user could also be using it // for synchronization. If they did, then synchronizing on tr here could lead to performance issues or // deadlocks. synchronized(HighContentionAllocator.class) { if(windowStart > initialWindowStart) { tr.clear(oldCounters); tr.options().setNextWriteNoWriteConflictRange(); tr.clear(oldAllocations); } tr.mutate(MutationType.ADD, counterKey, LITTLE_ENDIAN_LONG_ONE); newCountRead = tr.snapshot().get(counterKey); } return newCountRead.thenApply(newCountBytes -> { long newCount = newCountBytes == null ? 0 : unpackLittleEndian(newCountBytes); windowSize = getWindowSize(windowStart); if(newCount * 2 >= windowSize) { windowStart += windowSize; return true; } return false; // exit the loop }); }, tr.getExecutor()); } public CompletableFuture choosePrefix(final Transaction tr, final HighContentionAllocator allocator) { restart = false; return AsyncUtil.whileTrue(() -> { // As of the snapshot being read from, the window is less than half // full, so this should be expected to take 2 tries. Under high // contention (and when the window advances), there is an additional // subsequent risk of conflict for this transaction. candidate = windowStart + random.nextInt(windowSize); final byte[] allocationKey = allocator.recent.get(candidate).getKey(); Range countersRange = allocator.counters.range(); AsyncIterable counterRange; CompletableFuture allocationTemp; // SOMEDAY: this code would work as written if synchronized on something transaction local. // The reason we don't synchronize on the Transaction tr itself is that the user could also be using it // for synchronization. If they did, then synchronizing on tr here could lead to performance issues or // deadlocks. synchronized(HighContentionAllocator.class) { counterRange = tr.snapshot().getRange(countersRange, 1, true); allocationTemp = tr.get(allocationKey); tr.options().setNextWriteNoWriteConflictRange(); tr.set(allocationKey, EMPTY_BYTES); } final CompletableFuture> lastCounter = counterRange.asList(); final CompletableFuture allocation = allocationTemp; return lastCounter.thenCombineAsync(allocation, (result, allocationValue) -> { long currentWindowStart = 0; if(!result.isEmpty()) { currentWindowStart = allocator.counters.unpack(result.get(0).getKey()).getLong(0); } if(currentWindowStart > windowStart) { restart = true; return false; // exit the loop and rerun the allocation from the beginning } if(allocationValue == null) { tr.addWriteConflictKey(allocationKey); return false; // exit the loop and return this candidate } return true; }, tr.getExecutor()); }, tr.getExecutor()) .thenApply(ignore -> restart); } private static int getWindowSize(long start) { // Larger window sizes are better for high contention, smaller sizes for // keeping the keys small. But if there are many allocations, the keys // can't be too small. So start small and scale up. We don't want this // to ever get *too* big because we have to store about window_size/2 // recent items. if(start < 255) { return 64; } if(start < 65535) { return 1024; } return 8192; } } private static class HighContentionAllocator { public final Subspace counters; public final Subspace recent; HighContentionAllocator(Subspace subspace) { this.counters = subspace.get(0); this.recent = subspace.get(1); } /** * Returns a byte string that: *
    *
  1. has never and will never be returned by another call to this method on the same subspace
  2. *
  3. is nearly as short as possible given the above
  4. *
*/ public CompletableFuture allocate(final Transaction tr) { return new PrefixFinder().find(tr, this); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy