HDDS-357. Use DBStore and TableStore for OzoneManager non-background service.

Contributed by Nandakumar.
This commit is contained in:
Anu Engineer 2018-09-02 11:47:32 -07:00
parent eed8415dc1
commit ff036e49ff
26 changed files with 975 additions and 870 deletions

View File

@ -92,7 +92,6 @@ public final class OzoneConsts {
public static final String CONTAINER_DB_SUFFIX = "container.db"; public static final String CONTAINER_DB_SUFFIX = "container.db";
public static final String SCM_CONTAINER_DB = "scm-" + CONTAINER_DB_SUFFIX; public static final String SCM_CONTAINER_DB = "scm-" + CONTAINER_DB_SUFFIX;
public static final String DN_CONTAINER_DB = "-dn-"+ CONTAINER_DB_SUFFIX; public static final String DN_CONTAINER_DB = "-dn-"+ CONTAINER_DB_SUFFIX;
public static final String BLOCK_DB = "block.db";
public static final String OPEN_CONTAINERS_DB = "openContainers.db"; public static final String OPEN_CONTAINERS_DB = "openContainers.db";
public static final String DELETED_BLOCK_DB = "deletedBlock.db"; public static final String DELETED_BLOCK_DB = "deletedBlock.db";
public static final String OM_DB_NAME = "om.db"; public static final String OM_DB_NAME = "om.db";
@ -113,8 +112,6 @@ public static Versioning getVersioning(boolean versioning) {
public static final String DELETING_KEY_PREFIX = "#deleting#"; public static final String DELETING_KEY_PREFIX = "#deleting#";
public static final String DELETED_KEY_PREFIX = "#deleted#"; public static final String DELETED_KEY_PREFIX = "#deleted#";
public static final String DELETE_TRANSACTION_KEY_PREFIX = "#delTX#"; public static final String DELETE_TRANSACTION_KEY_PREFIX = "#delTX#";
public static final String OPEN_KEY_PREFIX = "#open#";
public static final String OPEN_KEY_ID_DELIMINATOR = "#";
/** /**
* OM LevelDB prefixes. * OM LevelDB prefixes.
@ -138,8 +135,7 @@ public static Versioning getVersioning(boolean versioning) {
* | #deleting#/volumeName/bucketName/keyName | KeyInfo | * | #deleting#/volumeName/bucketName/keyName | KeyInfo |
* ---------------------------------------------------------- * ----------------------------------------------------------
*/ */
public static final String OM_VOLUME_PREFIX = "/#";
public static final String OM_BUCKET_PREFIX = "/#";
public static final String OM_KEY_PREFIX = "/"; public static final String OM_KEY_PREFIX = "/";
public static final String OM_USER_PREFIX = "$"; public static final String OM_USER_PREFIX = "$";

View File

@ -94,7 +94,7 @@ public RocksDBStore(File dbFile, Options options)
} }
} }
private IOException toIOException(String msg, RocksDBException e) { public static IOException toIOException(String msg, RocksDBException e) {
String statusCode = e.getStatus() == null ? "N/A" : String statusCode = e.getStatus() == null ? "N/A" :
e.getStatus().getCodeString(); e.getStatus().getCodeString();
String errMessage = e.getMessage() == null ? "Unknown error" : String errMessage = e.getMessage() == null ? "Unknown error" :

View File

@ -20,6 +20,7 @@
package org.apache.hadoop.utils.db; package org.apache.hadoop.utils.db;
import org.apache.hadoop.classification.InterfaceStability; import org.apache.hadoop.classification.InterfaceStability;
import org.rocksdb.WriteBatch;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
@ -82,6 +83,22 @@ public interface DBStore extends AutoCloseable {
void move(byte[] key, byte[] value, Table source, Table dest) void move(byte[] key, byte[] value, Table source, Table dest)
throws IOException; throws IOException;
/**
* Moves a key from the Source Table to the destination Table and updates the
* destination with the new key name and value.
* This is similar to deleting an entry in one table and adding an entry in
* another table, here it is done atomically.
*
* @param sourceKey - Key to move.
* @param destKey - Destination key name.
* @param value - new value to write to the destination table.
* @param source - Source Table.
* @param dest - Destination Table.
* @throws IOException on Failure
*/
void move(byte[] sourceKey, byte[] destKey, byte[] value,
Table source, Table dest) throws IOException;
/** /**
* Returns an estimated count of keys in this DB. * Returns an estimated count of keys in this DB.
* *
@ -89,5 +106,10 @@ void move(byte[] key, byte[] value, Table source, Table dest)
*/ */
long getEstimatedKeyCount() throws IOException; long getEstimatedKeyCount() throws IOException;
/**
* Writes a transaction into the DB using the default write Options.
* @param batch - Batch to write.
*/
void write(WriteBatch batch) throws IOException;
} }

View File

@ -189,9 +189,16 @@ public void move(byte[] key, Table source, Table dest) throws IOException {
} }
} }
@Override @Override
public void move(byte[] key, byte[] value, Table source, public void move(byte[] key, byte[] value, Table source,
Table dest) throws IOException { Table dest) throws IOException {
move(key, key, value, source, dest);
}
@Override
public void move(byte[] sourceKey, byte[] destKey, byte[] value, Table source,
Table dest) throws IOException {
RDBTable sourceTable; RDBTable sourceTable;
RDBTable destTable; RDBTable destTable;
if (source instanceof RDBTable) { if (source instanceof RDBTable) {
@ -210,13 +217,13 @@ public void move(byte[] key, byte[] value, Table source,
+ "RocksDBTable."); + "RocksDBTable.");
} }
try (WriteBatch batch = new WriteBatch()) { try (WriteBatch batch = new WriteBatch()) {
batch.put(destTable.getHandle(), key, value); batch.put(destTable.getHandle(), destKey, value);
batch.delete(sourceTable.getHandle(), key); batch.delete(sourceTable.getHandle(), sourceKey);
db.write(writeOptions, batch); db.write(writeOptions, batch);
} catch (RocksDBException rockdbException) { } catch (RocksDBException rockdbException) {
LOG.error("Move of key failed. Key:{}", DFSUtil.bytes2String(key)); LOG.error("Move of key failed. Key:{}", DFSUtil.bytes2String(sourceKey));
throw toIOException("Unable to move key: " + DFSUtil.bytes2String(key), throw toIOException("Unable to move key: " +
rockdbException); DFSUtil.bytes2String(sourceKey), rockdbException);
} }
} }
@ -229,6 +236,15 @@ public long getEstimatedKeyCount() throws IOException {
} }
} }
@Override
public void write(WriteBatch batch) throws IOException {
try {
db.write(writeOptions, batch);
} catch (RocksDBException e) {
throw toIOException("Unable to write the batch.", e);
}
}
@VisibleForTesting @VisibleForTesting
protected ObjectName getStatMBeanName() { protected ObjectName getStatMBeanName() {
return statMBeanName; return statMBeanName;

View File

@ -1101,7 +1101,7 @@
<property> <property>
<name>hdds.db.profile</name> <name>hdds.db.profile</name>
<value>DBProfile.SSD</value> <value>SSD</value>
<tag>OZONE, OM, PERFORMANCE, REQUIRED</tag> <tag>OZONE, OM, PERFORMANCE, REQUIRED</tag>
<description>This property allows user to pick a configuration <description>This property allows user to pick a configuration
that tunes the RocksDB settings for the hardware it is running that tunes the RocksDB settings for the hardware it is running

View File

@ -136,4 +136,9 @@ public static File getOzoneMetaDirPath(Configuration conf) {
return dirPath; return dirPath;
} }
public static void setOzoneMetaDirPath(OzoneConfiguration conf,
String path) {
conf.set(OzoneConfigKeys.OZONE_METADATA_DIRS, path);
}
} }

View File

@ -74,7 +74,7 @@ public class ChunkGroupOutputStream extends OutputStream {
private final private final
StorageContainerLocationProtocolClientSideTranslatorPB scmClient; StorageContainerLocationProtocolClientSideTranslatorPB scmClient;
private final OmKeyArgs keyArgs; private final OmKeyArgs keyArgs;
private final int openID; private final long openID;
private final XceiverClientManager xceiverClientManager; private final XceiverClientManager xceiverClientManager;
private final int chunkSize; private final int chunkSize;
private final String requestID; private final String requestID;
@ -115,7 +115,7 @@ public List<ChunkOutputStreamEntry> getStreamEntries() {
} }
@VisibleForTesting @VisibleForTesting
public int getOpenID() { public long getOpenID() {
return openID; return openID;
} }

View File

@ -23,14 +23,14 @@
* that servers can recognize this client, and thus know how to close the key. * that servers can recognize this client, and thus know how to close the key.
*/ */
public class OpenKeySession { public class OpenKeySession {
private final int id; private final long id;
private final OmKeyInfo keyInfo; private final OmKeyInfo keyInfo;
// the version of the key when it is being opened in this session. // the version of the key when it is being opened in this session.
// a block that has a create version equals to open version means it will // a block that has a create version equals to open version means it will
// be committed only when this open session is closed. // be committed only when this open session is closed.
private long openVersion; private long openVersion;
public OpenKeySession(int id, OmKeyInfo info, long version) { public OpenKeySession(long id, OmKeyInfo info, long version) {
this.id = id; this.id = id;
this.keyInfo = info; this.keyInfo = info;
this.openVersion = version; this.openVersion = version;
@ -44,7 +44,7 @@ public OmKeyInfo getKeyInfo() {
return keyInfo; return keyInfo;
} }
public int getId() { public long getId() {
return id; return id;
} }
} }

View File

@ -148,7 +148,7 @@ OmBucketInfo getBucketInfo(String volumeName, String bucketName)
* @param clientID the client identification * @param clientID the client identification
* @throws IOException * @throws IOException
*/ */
void commitKey(OmKeyArgs args, int clientID) throws IOException; void commitKey(OmKeyArgs args, long clientID) throws IOException;
/** /**
* Allocate a new block, it is assumed that the client is having an open key * Allocate a new block, it is assumed that the client is having an open key
@ -159,7 +159,7 @@ OmBucketInfo getBucketInfo(String volumeName, String bucketName)
* @return an allocated block * @return an allocated block
* @throws IOException * @throws IOException
*/ */
OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID) OmKeyLocationInfo allocateBlock(OmKeyArgs args, long clientID)
throws IOException; throws IOException;
/** /**
@ -172,9 +172,10 @@ OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID)
OmKeyInfo lookupKey(OmKeyArgs args) throws IOException; OmKeyInfo lookupKey(OmKeyArgs args) throws IOException;
/** /**
* Rename an existing key within a bucket * Rename an existing key within a bucket.
* @param args the args of the key. * @param args the args of the key.
* @param toKeyName New name to be used for the Key * @param toKeyName New name to be used for the Key
* @throws IOException
*/ */
void renameKey(OmKeyArgs args, String toKeyName) throws IOException; void renameKey(OmKeyArgs args, String toKeyName) throws IOException;

View File

@ -554,7 +554,7 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
} }
@Override @Override
public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID) public OmKeyLocationInfo allocateBlock(OmKeyArgs args, long clientID)
throws IOException { throws IOException {
AllocateBlockRequest.Builder req = AllocateBlockRequest.newBuilder(); AllocateBlockRequest.Builder req = AllocateBlockRequest.newBuilder();
KeyArgs keyArgs = KeyArgs.newBuilder() KeyArgs keyArgs = KeyArgs.newBuilder()
@ -579,7 +579,7 @@ public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID)
} }
@Override @Override
public void commitKey(OmKeyArgs args, int clientID) public void commitKey(OmKeyArgs args, long clientID)
throws IOException { throws IOException {
CommitKeyRequest.Builder req = CommitKeyRequest.newBuilder(); CommitKeyRequest.Builder req = CommitKeyRequest.newBuilder();
List<OmKeyLocationInfo> locationInfoList = args.getLocationInfoList(); List<OmKeyLocationInfo> locationInfoList = args.getLocationInfoList();

View File

@ -273,7 +273,7 @@ message LocateKeyResponse {
optional KeyInfo keyInfo = 2; optional KeyInfo keyInfo = 2;
// clients' followup request may carry this ID for stateful operations (similar // clients' followup request may carry this ID for stateful operations (similar
// to a cookie). // to a cookie).
optional uint32 ID = 3; optional uint64 ID = 3;
// TODO : allow specifiying a particular version to read. // TODO : allow specifiying a particular version to read.
optional uint64 openVersion = 4; optional uint64 openVersion = 4;
} }
@ -319,7 +319,7 @@ message ListKeysResponse {
message AllocateBlockRequest { message AllocateBlockRequest {
required KeyArgs keyArgs = 1; required KeyArgs keyArgs = 1;
required uint32 clientID = 2; required uint64 clientID = 2;
} }
message AllocateBlockResponse { message AllocateBlockResponse {
@ -329,7 +329,7 @@ message AllocateBlockResponse {
message CommitKeyRequest { message CommitKeyRequest {
required KeyArgs keyArgs = 1; required KeyArgs keyArgs = 1;
required uint32 clientID = 2; required uint64 clientID = 2;
} }
message CommitKeyResponse { message CommitKeyResponse {

View File

@ -1,19 +1,18 @@
/** /**
* Licensed to the Apache Software Foundation (ASF) under one * Licensed to the Apache Software Foundation (ASF) under one or more
* or more contributor license agreements. See the NOTICE file * contributor license agreements. See the NOTICE file distributed with this
* distributed with this work for additional information * work for additional information regarding copyright ownership. The ASF
* regarding copyright ownership. The ASF licenses this file * licenses this file to you under the Apache License, Version 2.0 (the
* to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the License.
* "License"); you may not use this file except in compliance * You may obtain a copy of the License at
* with the License. You may obtain a copy of the License at
* <p> * <p>
* http://www.apache.org/licenses/LICENSE-2.0 * http://www.apache.org/licenses/LICENSE-2.0
* <p> * <p>
* Unless required by applicable law or agreed to in writing, software * Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* See the License for the specific language governing permissions and * License for the specific language governing permissions and limitations under
* limitations under the License. * the License.
*/ */
package org.apache.hadoop.ozone.client.rpc; package org.apache.hadoop.ozone.client.rpc;
@ -69,7 +68,6 @@ public class TestCloseContainerHandlingByClient {
private static String bucketName; private static String bucketName;
private static String keyString; private static String keyString;
/** /**
* Create a MiniDFSCluster for testing. * Create a MiniDFSCluster for testing.
* <p> * <p>
@ -288,13 +286,13 @@ private void waitForContainerClose(String keyName,
ChunkGroupOutputStream groupOutputStream = ChunkGroupOutputStream groupOutputStream =
(ChunkGroupOutputStream) outputStream.getOutputStream(); (ChunkGroupOutputStream) outputStream.getOutputStream();
int clientId = groupOutputStream.getOpenID(); long clientId = groupOutputStream.getOpenID();
OMMetadataManager metadataManager = OMMetadataManager metadataManager =
cluster.getOzoneManager().getMetadataManager(); cluster.getOzoneManager().getMetadataManager();
String objectKey = byte[] openKey =
metadataManager.getKeyWithDBPrefix(volumeName, bucketName, keyName); metadataManager.getOpenKeyBytes(
byte[] openKey = metadataManager.getOpenKeyNameBytes(objectKey, clientId); volumeName, bucketName, keyName, clientId);
byte[] openKeyData = metadataManager.get(openKey); byte[] openKeyData = metadataManager.getOpenKeyTable().get(openKey);
OmKeyInfo keyInfo = OmKeyInfo.getFromProtobuf( OmKeyInfo keyInfo = OmKeyInfo.getFromProtobuf(
OzoneManagerProtocolProtos.KeyInfo.parseFrom(openKeyData)); OzoneManagerProtocolProtos.KeyInfo.parseFrom(openKeyData));
List<OmKeyLocationInfo> locationInfoList = List<OmKeyLocationInfo> locationInfoList =
@ -361,7 +359,6 @@ private void validateData(String keyName, byte[] data) throws Exception {
is.close(); is.close();
} }
@Test @Test
public void testBlockWriteViaRatis() throws Exception { public void testBlockWriteViaRatis() throws Exception {
String keyName = "ratis"; String keyName = "ratis";

View File

@ -51,6 +51,7 @@
import org.junit.AfterClass; import org.junit.AfterClass;
import org.junit.Assert; import org.junit.Assert;
import org.junit.BeforeClass; import org.junit.BeforeClass;
import org.junit.Ignore;
import org.junit.Rule; import org.junit.Rule;
import org.junit.Test; import org.junit.Test;
import org.junit.rules.ExpectedException; import org.junit.rules.ExpectedException;
@ -601,6 +602,9 @@ public void testRenameKey()
Assert.assertEquals(toKeyName, key.getName()); Assert.assertEquals(toKeyName, key.getName());
} }
// Listing all volumes in the cluster feature has to be fixed after HDDS-357.
// TODO: fix this
@Ignore
@Test @Test
public void testListVolume() throws IOException, OzoneException { public void testListVolume() throws IOException, OzoneException {
String volBase = "vol-" + RandomStringUtils.randomNumeric(3); String volBase = "vol-" + RandomStringUtils.randomNumeric(3);

View File

@ -30,6 +30,7 @@
import org.apache.hadoop.test.GenericTestUtils; import org.apache.hadoop.test.GenericTestUtils;
import org.junit.After; import org.junit.After;
import org.junit.Before; import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import org.junit.runners.Parameterized; import org.junit.runners.Parameterized;
@ -82,7 +83,8 @@ public class TestOmSQLCli {
@Parameterized.Parameters @Parameterized.Parameters
public static Collection<Object[]> data() { public static Collection<Object[]> data() {
return Arrays.asList(new Object[][] { return Arrays.asList(new Object[][] {
{OzoneConfigKeys.OZONE_METADATA_STORE_IMPL_LEVELDB}, // Uncomment the below line if we support leveldb in future.
//{OzoneConfigKeys.OZONE_METADATA_STORE_IMPL_LEVELDB},
{OzoneConfigKeys.OZONE_METADATA_STORE_IMPL_ROCKSDB} {OzoneConfigKeys.OZONE_METADATA_STORE_IMPL_ROCKSDB}
}); });
} }
@ -161,6 +163,9 @@ public void shutdown() {
} }
} }
// After HDDS-357, we have to fix SQLCli.
// TODO: fix SQLCli
@Ignore
@Test @Test
public void testOmDB() throws Exception { public void testOmDB() throws Exception {
String dbOutPath = GenericTestUtils.getTempPath( String dbOutPath = GenericTestUtils.getTempPath(

View File

@ -56,8 +56,8 @@
import org.apache.hadoop.ozone.web.response.ListKeys; import org.apache.hadoop.ozone.web.response.ListKeys;
import org.apache.hadoop.ozone.web.response.ListVolumes; import org.apache.hadoop.ozone.web.response.ListVolumes;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
import org.apache.hadoop.utils.MetadataKeyFilters; import org.apache.hadoop.utils.db.Table;
import org.apache.hadoop.utils.MetadataStore; import org.apache.hadoop.utils.db.TableIterator;
import org.junit.AfterClass; import org.junit.AfterClass;
import org.junit.Assert; import org.junit.Assert;
import org.junit.BeforeClass; import org.junit.BeforeClass;
@ -75,7 +75,6 @@
import java.net.InetSocketAddress; import java.net.InetSocketAddress;
import java.text.ParseException; import java.text.ParseException;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.Map;
import java.util.Random; import java.util.Random;
import java.util.Set; import java.util.Set;
import java.util.List; import java.util.List;
@ -83,8 +82,8 @@
import java.util.stream.Collectors; import java.util.stream.Collectors;
import java.util.stream.Stream; import java.util.stream.Stream;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS; import static org.apache.hadoop.ozone.OzoneConfigKeys
import static org.apache.hadoop.ozone.OzoneConsts.DELETING_KEY_PREFIX; .OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS;
import static org.apache.hadoop.ozone.om.OMConfigKeys.OZONE_OM_ADDRESS_KEY; import static org.apache.hadoop.ozone.om.OMConfigKeys.OZONE_OM_ADDRESS_KEY;
import static org.apache.hadoop.hdds.scm.ScmConfigKeys import static org.apache.hadoop.hdds.scm.ScmConfigKeys
.OZONE_SCM_CLIENT_ADDRESS_KEY; .OZONE_SCM_CLIENT_ADDRESS_KEY;
@ -631,13 +630,16 @@ public void testDeleteKey() throws IOException, OzoneException {
storageHandler.deleteKey(keyArgs); storageHandler.deleteKey(keyArgs);
Assert.assertEquals(1 + numKeyDeletes, omMetrics.getNumKeyDeletes()); Assert.assertEquals(1 + numKeyDeletes, omMetrics.getNumKeyDeletes());
// Make sure the deleted key has been renamed. // Make sure the deleted key has been moved to the deleted table.
MetadataStore store = cluster.getOzoneManager(). OMMetadataManager manager = cluster.getOzoneManager().
getMetadataManager().getStore(); getMetadataManager();
List<Map.Entry<byte[], byte[]>> list = store.getRangeKVs(null, 10,
new MetadataKeyFilters.KeyPrefixFilter() try(TableIterator<Table.KeyValue> iter =
.addFilter(DELETING_KEY_PREFIX)); manager.getDeletedTable().iterator()) {
Assert.assertEquals(1, list.size()); iter.seekToFirst();
Table.KeyValue kv = iter.next();
Assert.assertNotNull(kv);
}
// Delete the key again to test deleting non-existing key. // Delete the key again to test deleting non-existing key.
try { try {
@ -1016,13 +1018,14 @@ public void testListVolumes() throws IOException, OzoneException {
storageHandler.createVolume(createVolumeArgs); storageHandler.createVolume(createVolumeArgs);
} }
// Test list all volumes // Test list all volumes - Removed Support for this operation for time
// being. TODO: we will need to bring this back if needed.
UserArgs userArgs0 = new UserArgs(user0, OzoneUtils.getRequestID(), UserArgs userArgs0 = new UserArgs(user0, OzoneUtils.getRequestID(),
null, null, null, null); null, null, null, null);
listVolumeArgs = new ListArgs(userArgs0, "Vol-testListVolumes", 100, null); //listVolumeArgs = new ListArgs(userArgs0,"Vol-testListVolumes", 100, null);
listVolumeArgs.setRootScan(true); // listVolumeArgs.setRootScan(true);
volumes = storageHandler.listVolumes(listVolumeArgs); // volumes = storageHandler.listVolumes(listVolumeArgs);
Assert.assertEquals(20, volumes.getVolumes().size()); // Assert.assertEquals(20, volumes.getVolumes().size());
// Test list all volumes belongs to an user // Test list all volumes belongs to an user
listVolumeArgs = new ListArgs(userArgs0, null, 100, null); listVolumeArgs = new ListArgs(userArgs0, null, 100, null);

View File

@ -221,6 +221,9 @@ static void runTestChangeQuotaOnVolume(ClientProtocol client)
assertTrue(newVol.getCreationTime() > 0); assertTrue(newVol.getCreationTime() > 0);
} }
// Listing all volumes in the cluster feature has to be fixed after HDDS-357.
// TODO: fix this
@Ignore
@Test @Test
public void testListVolume() throws OzoneException, IOException { public void testListVolume() throws OzoneException, IOException {
runTestListVolume(client); runTestListVolume(client);
@ -305,6 +308,9 @@ static void runTestListAllVolumes(ClientProtocol client)
assertEquals(volCount / step, pagecount); assertEquals(volCount / step, pagecount);
} }
// Listing all volumes in the cluster feature has to be fixed after HDDS-357.
// TODO: fix this
@Ignore
@Test @Test
public void testListVolumes() throws Exception { public void testListVolumes() throws Exception {
runTestListVolumes(client); runTestListVolumes(client);

View File

@ -18,12 +18,11 @@
import com.google.common.base.Preconditions; import com.google.common.base.Preconditions;
import org.apache.hadoop.fs.StorageType; import org.apache.hadoop.fs.StorageType;
import org.apache.hadoop.ozone.OzoneAcl;
import org.apache.hadoop.ozone.om.exceptions.OMException;
import org.apache.hadoop.ozone.om.helpers.OmBucketArgs; import org.apache.hadoop.ozone.om.helpers.OmBucketArgs;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo; import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.om.exceptions.OMException; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.BucketInfo;
import org.apache.hadoop.ozone.protocol.proto
.OzoneManagerProtocolProtos.BucketInfo;
import org.apache.hadoop.ozone.OzoneAcl;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
import org.iq80.leveldb.DBException; import org.iq80.leveldb.DBException;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -46,6 +45,7 @@ public class BucketManagerImpl implements BucketManager {
/** /**
* Constructs BucketManager. * Constructs BucketManager.
*
* @param metadataManager * @param metadataManager
*/ */
public BucketManagerImpl(OMMetadataManager metadataManager) { public BucketManagerImpl(OMMetadataManager metadataManager) {
@ -73,6 +73,7 @@ public BucketManagerImpl(OMMetadataManager metadataManager){
/** /**
* Creates a bucket. * Creates a bucket.
*
* @param bucketInfo - OmBucketInfo. * @param bucketInfo - OmBucketInfo.
*/ */
@Override @Override
@ -86,13 +87,13 @@ public void createBucket(OmBucketInfo bucketInfo) throws IOException {
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName); byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
//Check if the volume exists //Check if the volume exists
if (metadataManager.get(volumeKey) == null) { if (metadataManager.getVolumeTable().get(volumeKey) == null) {
LOG.debug("volume: {} not found ", volumeName); LOG.debug("volume: {} not found ", volumeName);
throw new OMException("Volume doesn't exist", throw new OMException("Volume doesn't exist",
OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND); OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND);
} }
//Check if bucket already exists //Check if bucket already exists
if (metadataManager.get(bucketKey) != null) { if (metadataManager.getBucketTable().get(bucketKey) != null) {
LOG.debug("bucket: {} already exists ", bucketName); LOG.debug("bucket: {} already exists ", bucketName);
throw new OMException("Bucket already exist", throw new OMException("Bucket already exist",
OMException.ResultCodes.FAILED_BUCKET_ALREADY_EXISTS); OMException.ResultCodes.FAILED_BUCKET_ALREADY_EXISTS);
@ -106,7 +107,8 @@ public void createBucket(OmBucketInfo bucketInfo) throws IOException {
.setIsVersionEnabled(bucketInfo.getIsVersionEnabled()) .setIsVersionEnabled(bucketInfo.getIsVersionEnabled())
.setCreationTime(Time.now()) .setCreationTime(Time.now())
.build(); .build();
metadataManager.put(bucketKey, omBucketInfo.getProtobuf().toByteArray()); metadataManager.getBucketTable().put(bucketKey,
omBucketInfo.getProtobuf().toByteArray());
LOG.debug("created bucket: {} in volume: {}", bucketName, volumeName); LOG.debug("created bucket: {} in volume: {}", bucketName, volumeName);
} catch (IOException | DBException ex) { } catch (IOException | DBException ex) {
@ -134,7 +136,7 @@ public OmBucketInfo getBucketInfo(String volumeName, String bucketName)
metadataManager.readLock().lock(); metadataManager.readLock().lock();
try { try {
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName); byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
byte[] value = metadataManager.get(bucketKey); byte[] value = metadataManager.getBucketTable().get(bucketKey);
if (value == null) { if (value == null) {
LOG.debug("bucket: {} not found in volume: {}.", bucketName, LOG.debug("bucket: {} not found in volume: {}.", bucketName,
volumeName); volumeName);
@ -155,8 +157,9 @@ public OmBucketInfo getBucketInfo(String volumeName, String bucketName)
/** /**
* Sets bucket property from args. * Sets bucket property from args.
*
* @param args - BucketArgs. * @param args - BucketArgs.
* @throws IOException * @throws IOException - On Failure.
*/ */
@Override @Override
public void setBucketProperty(OmBucketArgs args) throws IOException { public void setBucketProperty(OmBucketArgs args) throws IOException {
@ -167,13 +170,13 @@ public void setBucketProperty(OmBucketArgs args) throws IOException {
try { try {
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName); byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
//Check if volume exists //Check if volume exists
if(metadataManager.get(metadataManager.getVolumeKey(volumeName)) == if (metadataManager.getVolumeTable()
null) { .get(metadataManager.getVolumeKey(volumeName)) == null) {
LOG.debug("volume: {} not found ", volumeName); LOG.debug("volume: {} not found ", volumeName);
throw new OMException("Volume doesn't exist", throw new OMException("Volume doesn't exist",
OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND); OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND);
} }
byte[] value = metadataManager.get(bucketKey); byte[] value = metadataManager.getBucketTable().get(bucketKey);
//Check if bucket exist //Check if bucket exist
if (value == null) { if (value == null) {
LOG.debug("bucket: {} not found ", bucketName); LOG.debug("bucket: {} not found ", bucketName);
@ -218,7 +221,7 @@ public void setBucketProperty(OmBucketArgs args) throws IOException {
} }
bucketInfoBuilder.setCreationTime(oldBucketInfo.getCreationTime()); bucketInfoBuilder.setCreationTime(oldBucketInfo.getCreationTime());
metadataManager.put(bucketKey, metadataManager.getBucketTable().put(bucketKey,
bucketInfoBuilder.build().getProtobuf().toByteArray()); bucketInfoBuilder.build().getProtobuf().toByteArray());
} catch (IOException | DBException ex) { } catch (IOException | DBException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
@ -254,9 +257,10 @@ private List<OzoneAcl> getUpdatedAclList(List<OzoneAcl> existingAcls,
/** /**
* Deletes an existing empty bucket from volume. * Deletes an existing empty bucket from volume.
*
* @param volumeName - Name of the volume. * @param volumeName - Name of the volume.
* @param bucketName - Name of the bucket. * @param bucketName - Name of the bucket.
* @throws IOException * @throws IOException - on Failure.
*/ */
public void deleteBucket(String volumeName, String bucketName) public void deleteBucket(String volumeName, String bucketName)
throws IOException { throws IOException {
@ -264,16 +268,17 @@ public void deleteBucket(String volumeName, String bucketName)
Preconditions.checkNotNull(bucketName); Preconditions.checkNotNull(bucketName);
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
//Check if volume exists //Check if volume exists
if (metadataManager.get(metadataManager.getVolumeKey(volumeName)) if (metadataManager.getVolumeTable()
== null) { .get(metadataManager.getVolumeKey(volumeName)) == null) {
LOG.debug("volume: {} not found ", volumeName); LOG.debug("volume: {} not found ", volumeName);
throw new OMException("Volume doesn't exist", throw new OMException("Volume doesn't exist",
OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND); OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND);
} }
//Check if bucket exist
if (metadataManager.get(bucketKey) == null) { //Check if bucket exists
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
if (metadataManager.getBucketTable().get(bucketKey) == null) {
LOG.debug("bucket: {} not found ", bucketName); LOG.debug("bucket: {} not found ", bucketName);
throw new OMException("Bucket doesn't exist", throw new OMException("Bucket doesn't exist",
OMException.ResultCodes.FAILED_BUCKET_NOT_FOUND); OMException.ResultCodes.FAILED_BUCKET_NOT_FOUND);
@ -284,7 +289,7 @@ public void deleteBucket(String volumeName, String bucketName)
throw new OMException("Bucket is not empty", throw new OMException("Bucket is not empty",
OMException.ResultCodes.FAILED_BUCKET_NOT_EMPTY); OMException.ResultCodes.FAILED_BUCKET_NOT_EMPTY);
} }
metadataManager.delete(bucketKey); metadataManager.getBucketTable().delete(bucketKey);
} catch (IOException ex) { } catch (IOException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
LOG.error("Delete bucket failed for bucket:{} in volume:{}", bucketName, LOG.error("Delete bucket failed for bucket:{} in volume:{}", bucketName,

View File

@ -49,7 +49,7 @@ public interface KeyManager {
* @param clientID the client that is committing. * @param clientID the client that is committing.
* @throws IOException * @throws IOException
*/ */
void commitKey(OmKeyArgs args, int clientID) throws IOException; void commitKey(OmKeyArgs args, long clientID) throws IOException;
/** /**
* A client calls this on an open key, to request to allocate a new block, * A client calls this on an open key, to request to allocate a new block,
@ -60,7 +60,7 @@ public interface KeyManager {
* @return the reference to the new block. * @return the reference to the new block.
* @throws IOException * @throws IOException
*/ */
OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID) OmKeyLocationInfo allocateBlock(OmKeyArgs args, long clientID)
throws IOException; throws IOException;
/** /**
* Given the args of a key to put, write an open key entry to meta data. * Given the args of a key to put, write an open key entry to meta data.

View File

@ -17,24 +17,25 @@
package org.apache.hadoop.ozone.om; package org.apache.hadoop.ozone.om;
import com.google.common.base.Preconditions; import com.google.common.base.Preconditions;
import org.apache.hadoop.ozone.OzoneConsts;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.ozone.om.helpers.OmKeyArgs;
import org.apache.hadoop.ozone.common.BlockGroup;
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfo;
import org.apache.hadoop.hdds.conf.OzoneConfiguration; import org.apache.hadoop.hdds.conf.OzoneConfiguration;
import org.apache.hadoop.ozone.om.exceptions.OMException; import org.apache.hadoop.hdds.protocol.proto.HddsProtos.ReplicationFactor;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes; import org.apache.hadoop.hdds.protocol.proto.HddsProtos.ReplicationType;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
import org.apache.hadoop.ozone.om.helpers.OpenKeySession;
import org.apache.hadoop.ozone.protocol.proto
.OzoneManagerProtocolProtos.KeyInfo;
import org.apache.hadoop.hdds.scm.container.common.helpers.AllocatedBlock; import org.apache.hadoop.hdds.scm.container.common.helpers.AllocatedBlock;
import org.apache.hadoop.hdds.scm.protocol.ScmBlockLocationProtocol; import org.apache.hadoop.hdds.scm.protocol.ScmBlockLocationProtocol;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.ozone.OzoneConsts;
import org.apache.hadoop.ozone.common.BlockGroup;
import org.apache.hadoop.ozone.om.exceptions.OMException;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes;
import org.apache.hadoop.ozone.om.helpers.OmKeyArgs;
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
import org.apache.hadoop.ozone.om.helpers.OpenKeySession;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.KeyInfo;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
import org.apache.hadoop.utils.BatchOperation; import org.rocksdb.RocksDBException;
import org.iq80.leveldb.DBException; import org.rocksdb.WriteBatch;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -42,25 +43,13 @@
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Random;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.DFS_CONTAINER_RATIS_ENABLED_DEFAULT;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.DFS_CONTAINER_RATIS_ENABLED_KEY;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.OZONE_KEY_PREALLOCATION_MAXSIZE;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.OZONE_KEY_PREALLOCATION_MAXSIZE_DEFAULT;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.OZONE_SCM_BLOCK_SIZE_DEFAULT;
import static org.apache.hadoop.ozone
.OzoneConfigKeys.OZONE_SCM_BLOCK_SIZE_IN_MB;
import org.apache.hadoop.hdds.protocol
.proto.HddsProtos.ReplicationType;
import org.apache.hadoop.hdds.protocol
.proto.HddsProtos.ReplicationFactor;
import static org.apache.hadoop.ozone.OzoneConfigKeys.DFS_CONTAINER_RATIS_ENABLED_DEFAULT;
import static org.apache.hadoop.ozone.OzoneConfigKeys.DFS_CONTAINER_RATIS_ENABLED_KEY;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_KEY_PREALLOCATION_MAXSIZE;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_KEY_PREALLOCATION_MAXSIZE_DEFAULT;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_SCM_BLOCK_SIZE_DEFAULT;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_SCM_BLOCK_SIZE_IN_MB;
/** /**
* Implementation of keyManager. * Implementation of keyManager.
@ -78,7 +67,6 @@ public class KeyManagerImpl implements KeyManager {
private final boolean useRatis; private final boolean useRatis;
private final long preallocateMax; private final long preallocateMax;
private final Random random;
private final String omId; private final String omId;
public KeyManagerImpl(ScmBlockLocationProtocol scmBlockClient, public KeyManagerImpl(ScmBlockLocationProtocol scmBlockClient,
@ -94,11 +82,9 @@ public KeyManagerImpl(ScmBlockLocationProtocol scmBlockClient,
this.preallocateMax = conf.getLong( this.preallocateMax = conf.getLong(
OZONE_KEY_PREALLOCATION_MAXSIZE, OZONE_KEY_PREALLOCATION_MAXSIZE,
OZONE_KEY_PREALLOCATION_MAXSIZE_DEFAULT); OZONE_KEY_PREALLOCATION_MAXSIZE_DEFAULT);
random = new Random();
this.omId = omId; this.omId = omId;
} }
@Override @Override
public void start() { public void start() {
} }
@ -113,13 +99,13 @@ private void validateBucket(String volumeName, String bucketName)
byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName); byte[] bucketKey = metadataManager.getBucketKey(volumeName, bucketName);
//Check if the volume exists //Check if the volume exists
if(metadataManager.get(volumeKey) == null) { if (metadataManager.getVolumeTable().get(volumeKey) == null) {
LOG.error("volume not found: {}", volumeName); LOG.error("volume not found: {}", volumeName);
throw new OMException("Volume not found", throw new OMException("Volume not found",
OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND); OMException.ResultCodes.FAILED_VOLUME_NOT_FOUND);
} }
//Check if bucket already exists //Check if bucket already exists
if(metadataManager.get(bucketKey) == null) { if (metadataManager.getBucketTable().get(bucketKey) == null) {
LOG.error("bucket not found: {}/{} ", volumeName, bucketName); LOG.error("bucket not found: {}/{} ", volumeName, bucketName);
throw new OMException("Bucket not found", throw new OMException("Bucket not found",
OMException.ResultCodes.FAILED_BUCKET_NOT_FOUND); OMException.ResultCodes.FAILED_BUCKET_NOT_FOUND);
@ -127,7 +113,7 @@ private void validateBucket(String volumeName, String bucketName)
} }
@Override @Override
public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID) public OmKeyLocationInfo allocateBlock(OmKeyArgs args, long clientID)
throws IOException { throws IOException {
Preconditions.checkNotNull(args); Preconditions.checkNotNull(args);
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
@ -137,13 +123,13 @@ public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID)
try { try {
validateBucket(volumeName, bucketName); validateBucket(volumeName, bucketName);
String objectKey = metadataManager.getKeyWithDBPrefix( byte[] openKey = metadataManager.getOpenKeyBytes(
volumeName, bucketName, keyName); volumeName, bucketName, keyName, clientID);
byte[] openKey = metadataManager.getOpenKeyNameBytes(objectKey, clientID);
byte[] keyData = metadataManager.get(openKey); byte[] keyData = metadataManager.getOpenKeyTable().get(openKey);
if (keyData == null) { if (keyData == null) {
LOG.error("Allocate block for a key not in open status in meta store" + LOG.error("Allocate block for a key not in open status in meta store" +
objectKey + " with ID " + clientID); " /{}/{}/{} with ID {}", volumeName, bucketName, keyName, clientID);
throw new OMException("Open Key not found", throw new OMException("Open Key not found",
OMException.ResultCodes.FAILED_KEY_NOT_FOUND); OMException.ResultCodes.FAILED_KEY_NOT_FOUND);
} }
@ -162,7 +148,8 @@ public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID)
// the same version // the same version
keyInfo.appendNewBlocks(Collections.singletonList(info)); keyInfo.appendNewBlocks(Collections.singletonList(info));
keyInfo.updateModifcationTime(); keyInfo.updateModifcationTime();
metadataManager.put(openKey, keyInfo.getProtobuf().toByteArray()); metadataManager.getOpenKeyTable().put(openKey,
keyInfo.getProtobuf().toByteArray());
return info; return info;
} finally { } finally {
metadataManager.writeLock().unlock(); metadataManager.writeLock().unlock();
@ -172,12 +159,15 @@ public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID)
@Override @Override
public OpenKeySession openKey(OmKeyArgs args) throws IOException { public OpenKeySession openKey(OmKeyArgs args) throws IOException {
Preconditions.checkNotNull(args); Preconditions.checkNotNull(args);
metadataManager.writeLock().lock();
String volumeName = args.getVolumeName(); String volumeName = args.getVolumeName();
String bucketName = args.getBucketName(); String bucketName = args.getBucketName();
validateBucket(volumeName, bucketName);
metadataManager.writeLock().lock();
String keyName = args.getKeyName(); String keyName = args.getKeyName();
ReplicationFactor factor = args.getFactor(); ReplicationFactor factor = args.getFactor();
ReplicationType type = args.getType(); ReplicationType type = args.getType();
long currentTime = Time.monotonicNowNanos();
// If user does not specify a replication strategy or // If user does not specify a replication strategy or
// replication factor, OM will use defaults. // replication factor, OM will use defaults.
@ -190,10 +180,9 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
} }
try { try {
validateBucket(volumeName, bucketName);
long requestedSize = Math.min(preallocateMax, args.getDataSize()); long requestedSize = Math.min(preallocateMax, args.getDataSize());
List<OmKeyLocationInfo> locations = new ArrayList<>(); List<OmKeyLocationInfo> locations = new ArrayList<>();
String objectKey = metadataManager.getKeyWithDBPrefix( byte[] objectKey = metadataManager.getOzoneKeyBytes(
volumeName, bucketName, keyName); volumeName, bucketName, keyName);
// requested size is not required but more like a optimization: // requested size is not required but more like a optimization:
// SCM looks at the requested, if it 0, no block will be allocated at // SCM looks at the requested, if it 0, no block will be allocated at
@ -218,9 +207,7 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
// value, then this value is used, otherwise, we allocate a single block // value, then this value is used, otherwise, we allocate a single block
// which is the current size, if read by the client. // which is the current size, if read by the client.
long size = args.getDataSize() >= 0 ? args.getDataSize() : scmBlockSize; long size = args.getDataSize() >= 0 ? args.getDataSize() : scmBlockSize;
byte[] keyKey = metadataManager.getDBKeyBytes( byte[] value = metadataManager.getKeyTable().get(objectKey);
volumeName, bucketName, keyName);
byte[] value = metadataManager.get(keyKey);
OmKeyInfo keyInfo; OmKeyInfo keyInfo;
long openVersion; long openVersion;
if (value != null) { if (value != null) {
@ -233,7 +220,7 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
} else { } else {
// the key does not exist, create a new object, the new blocks are the // the key does not exist, create a new object, the new blocks are the
// version 0 // version 0
long currentTime = Time.now();
keyInfo = new OmKeyInfo.Builder() keyInfo = new OmKeyInfo.Builder()
.setVolumeName(args.getVolumeName()) .setVolumeName(args.getVolumeName())
.setBucketName(args.getBucketName()) .setBucketName(args.getBucketName())
@ -248,31 +235,31 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
.build(); .build();
openVersion = 0; openVersion = 0;
} }
// Generate a random ID which is not already in meta db. byte[] openKey = metadataManager.getOpenKeyBytes(
int id = -1; volumeName, bucketName, keyName, currentTime);
// in general this should finish in a couple times at most. putting some if (metadataManager.getOpenKeyTable().get(openKey) != null) {
// arbitrary large number here to avoid dead loop. // This should not happen. If this condition is satisfied, it means
for (int j = 0; j < 10000; j++) { // that we have generated a same openKeyId (i.e. currentTime) for two
id = random.nextInt(); // different client who are trying to write the same key at the same
byte[] openKey = metadataManager.getOpenKeyNameBytes(objectKey, id); // time. The chance of this happening is very, very minimal.
if (metadataManager.get(openKey) == null) {
metadataManager.put(openKey, keyInfo.getProtobuf().toByteArray()); // Do we really need this check? Can we avoid this to gain some
break; // minor performance improvement?
} LOG.warn("Cannot allocate key. The generated open key id is already" +
} "used for the same key which is currently being written.");
if (id == -1) { throw new OMException("Cannot allocate key. Not able to get a valid" +
throw new IOException("Failed to find a usable id for " + objectKey); "open key id.", OMException.ResultCodes.FAILED_KEY_ALLOCATION);
} }
metadataManager.getOpenKeyTable().put(openKey,
keyInfo.getProtobuf().toByteArray());
LOG.debug("Key {} allocated in volume {} bucket {}", LOG.debug("Key {} allocated in volume {} bucket {}",
keyName, volumeName, bucketName); keyName, volumeName, bucketName);
return new OpenKeySession(id, keyInfo, openVersion); return new OpenKeySession(currentTime, keyInfo, openVersion);
} catch (OMException e) { } catch (OMException e) {
throw e; throw e;
} catch (IOException ex) { } catch (IOException ex) {
if (!(ex instanceof OMException)) {
LOG.error("Key open failed for volume:{} bucket:{} key:{}", LOG.error("Key open failed for volume:{} bucket:{} key:{}",
volumeName, bucketName, keyName, ex); volumeName, bucketName, keyName, ex);
}
throw new OMException(ex.getMessage(), throw new OMException(ex.getMessage(),
OMException.ResultCodes.FAILED_KEY_ALLOCATION); OMException.ResultCodes.FAILED_KEY_ALLOCATION);
} finally { } finally {
@ -281,7 +268,7 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
} }
@Override @Override
public void commitKey(OmKeyArgs args, int clientID) throws IOException { public void commitKey(OmKeyArgs args, long clientID) throws IOException {
Preconditions.checkNotNull(args); Preconditions.checkNotNull(args);
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
String volumeName = args.getVolumeName(); String volumeName = args.getVolumeName();
@ -289,15 +276,14 @@ public void commitKey(OmKeyArgs args, int clientID) throws IOException {
String keyName = args.getKeyName(); String keyName = args.getKeyName();
try { try {
validateBucket(volumeName, bucketName); validateBucket(volumeName, bucketName);
String objectKey = metadataManager.getKeyWithDBPrefix( byte[] openKey = metadataManager.getOpenKeyBytes(volumeName, bucketName,
keyName, clientID);
byte[] objectKey = metadataManager.getOzoneKeyBytes(
volumeName, bucketName, keyName); volumeName, bucketName, keyName);
byte[] objectKeyBytes = metadataManager.getDBKeyBytes(volumeName, byte[] openKeyData = metadataManager.getOpenKeyTable().get(openKey);
bucketName, keyName);
byte[] openKey = metadataManager.getOpenKeyNameBytes(objectKey, clientID);
byte[] openKeyData = metadataManager.get(openKey);
if (openKeyData == null) { if (openKeyData == null) {
throw new OMException("Commit a key without corresponding entry " + throw new OMException("Commit a key without corresponding entry " +
DFSUtil.bytes2String(openKey), ResultCodes.FAILED_KEY_NOT_FOUND); DFSUtil.bytes2String(objectKey), ResultCodes.FAILED_KEY_NOT_FOUND);
} }
OmKeyInfo keyInfo = OmKeyInfo keyInfo =
OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(openKeyData)); OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(openKeyData));
@ -305,12 +291,13 @@ public void commitKey(OmKeyArgs args, int clientID) throws IOException {
keyInfo.setModificationTime(Time.now()); keyInfo.setModificationTime(Time.now());
List<OmKeyLocationInfo> locationInfoList = args.getLocationInfoList(); List<OmKeyLocationInfo> locationInfoList = args.getLocationInfoList();
Preconditions.checkNotNull(locationInfoList); Preconditions.checkNotNull(locationInfoList);
//update the block length for each block //update the block length for each block
keyInfo.updateLocationInfoList(locationInfoList); keyInfo.updateLocationInfoList(locationInfoList);
BatchOperation batch = new BatchOperation(); metadataManager.getStore().move(openKey, objectKey,
batch.delete(openKey); keyInfo.getProtobuf().toByteArray(),
batch.put(objectKeyBytes, keyInfo.getProtobuf().toByteArray()); metadataManager.getOpenKeyTable(),
metadataManager.writeBatch(batch); metadataManager.getKeyTable());
} catch (OMException e) { } catch (OMException e) {
throw e; throw e;
} catch (IOException ex) { } catch (IOException ex) {
@ -331,9 +318,9 @@ public OmKeyInfo lookupKey(OmKeyArgs args) throws IOException {
String bucketName = args.getBucketName(); String bucketName = args.getBucketName();
String keyName = args.getKeyName(); String keyName = args.getKeyName();
try { try {
byte[] keyKey = metadataManager.getDBKeyBytes( byte[] keyBytes = metadataManager.getOzoneKeyBytes(
volumeName, bucketName, keyName); volumeName, bucketName, keyName);
byte[] value = metadataManager.get(keyKey); byte[] value = metadataManager.getKeyTable().get(keyBytes);
if (value == null) { if (value == null) {
LOG.debug("volume:{} bucket:{} Key:{} not found", LOG.debug("volume:{} bucket:{} Key:{} not found",
volumeName, bucketName, keyName); volumeName, bucketName, keyName);
@ -341,7 +328,7 @@ public OmKeyInfo lookupKey(OmKeyArgs args) throws IOException {
OMException.ResultCodes.FAILED_KEY_NOT_FOUND); OMException.ResultCodes.FAILED_KEY_NOT_FOUND);
} }
return OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(value)); return OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(value));
} catch (DBException ex) { } catch (IOException ex) {
LOG.error("Get key failed for volume:{} bucket:{} key:{}", LOG.error("Get key failed for volume:{} bucket:{} key:{}",
volumeName, bucketName, keyName, ex); volumeName, bucketName, keyName, ex);
throw new OMException(ex.getMessage(), throw new OMException(ex.getMessage(),
@ -368,9 +355,9 @@ public void renameKey(OmKeyArgs args, String toKeyName) throws IOException {
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
// fromKeyName should exist // fromKeyName should exist
byte[] fromKey = metadataManager.getDBKeyBytes( byte[] fromKey = metadataManager.getOzoneKeyBytes(
volumeName, bucketName, fromKeyName); volumeName, bucketName, fromKeyName);
byte[] fromKeyValue = metadataManager.get(fromKey); byte[] fromKeyValue = metadataManager.getKeyTable().get(fromKey);
if (fromKeyValue == null) { if (fromKeyValue == null) {
// TODO: Add support for renaming open key // TODO: Add support for renaming open key
LOG.error( LOG.error(
@ -381,10 +368,20 @@ public void renameKey(OmKeyArgs args, String toKeyName) throws IOException {
OMException.ResultCodes.FAILED_KEY_NOT_FOUND); OMException.ResultCodes.FAILED_KEY_NOT_FOUND);
} }
// A rename is a no-op if the target and source name is same.
// TODO: Discuss if we need to throw?.
// TODO: Define the semantics of rename more clearly. Today this code
// will allow rename of a Key across volumes. This should *not* be
// allowed. The documentation of Ozone says that rename is permitted only
// within a volume.
if (fromKeyName.equals(toKeyName)) {
return;
}
// toKeyName should not exist // toKeyName should not exist
byte[] toKey = byte[] toKey =
metadataManager.getDBKeyBytes(volumeName, bucketName, toKeyName); metadataManager.getOzoneKeyBytes(volumeName, bucketName, toKeyName);
byte[] toKeyValue = metadataManager.get(toKey); byte[] toKeyValue = metadataManager.getKeyTable().get(toKey);
if (toKeyValue != null) { if (toKeyValue != null) {
LOG.error( LOG.error(
"Rename key failed for volume:{} bucket:{} fromKey:{} toKey:{}. " "Rename key failed for volume:{} bucket:{} fromKey:{} toKey:{}. "
@ -394,19 +391,18 @@ public void renameKey(OmKeyArgs args, String toKeyName) throws IOException {
OMException.ResultCodes.FAILED_KEY_ALREADY_EXISTS); OMException.ResultCodes.FAILED_KEY_ALREADY_EXISTS);
} }
if (fromKeyName.equals(toKeyName)) {
return;
}
OmKeyInfo newKeyInfo = OmKeyInfo newKeyInfo =
OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(fromKeyValue)); OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(fromKeyValue));
newKeyInfo.setKeyName(toKeyName); newKeyInfo.setKeyName(toKeyName);
newKeyInfo.updateModifcationTime(); newKeyInfo.updateModifcationTime();
BatchOperation batch = new BatchOperation(); try (WriteBatch batch = new WriteBatch()) {
batch.delete(fromKey); batch.delete(metadataManager.getKeyTable().getHandle(), fromKey);
batch.put(toKey, newKeyInfo.getProtobuf().toByteArray()); batch.put(metadataManager.getKeyTable().getHandle(), toKey,
metadataManager.writeBatch(batch); newKeyInfo.getProtobuf().toByteArray());
} catch (DBException ex) { metadataManager.getStore().write(batch);
}
} catch (RocksDBException | IOException ex) {
LOG.error("Rename key failed for volume:{} bucket:{} fromKey:{} toKey:{}", LOG.error("Rename key failed for volume:{} bucket:{} fromKey:{} toKey:{}",
volumeName, bucketName, fromKeyName, toKeyName, ex); volumeName, bucketName, fromKeyName, toKeyName, ex);
throw new OMException(ex.getMessage(), throw new OMException(ex.getMessage(),
@ -424,19 +420,19 @@ public void deleteKey(OmKeyArgs args) throws IOException {
String bucketName = args.getBucketName(); String bucketName = args.getBucketName();
String keyName = args.getKeyName(); String keyName = args.getKeyName();
try { try {
byte[] objectKey = metadataManager.getDBKeyBytes( byte[] objectKey = metadataManager.getOzoneKeyBytes(
volumeName, bucketName, keyName); volumeName, bucketName, keyName);
byte[] objectValue = metadataManager.get(objectKey); byte[] objectValue = metadataManager.getKeyTable().get(objectKey);
if (objectValue == null) { if (objectValue == null) {
throw new OMException("Key not found", throw new OMException("Key not found",
OMException.ResultCodes.FAILED_KEY_NOT_FOUND); OMException.ResultCodes.FAILED_KEY_NOT_FOUND);
} }
byte[] deletingKey = metadataManager.getDeletedKeyName(objectKey); metadataManager.getStore().move(objectKey,
BatchOperation batch = new BatchOperation(); metadataManager.getKeyTable(),
batch.put(deletingKey, objectValue); metadataManager.getDeletedTable());
batch.delete(objectKey); } catch (OMException ex) {
metadataManager.writeBatch(batch); throw ex;
} catch (DBException ex) { } catch (IOException ex) {
LOG.error(String.format("Delete key failed for volume:%s " LOG.error(String.format("Delete key failed for volume:%s "
+ "bucket:%s key:%s", volumeName, bucketName, keyName), ex); + "bucket:%s key:%s", volumeName, bucketName, keyName), ex);
throw new OMException(ex.getMessage(), ex, throw new OMException(ex.getMessage(), ex,
@ -453,48 +449,25 @@ public List<OmKeyInfo> listKeys(String volumeName, String bucketName,
Preconditions.checkNotNull(volumeName); Preconditions.checkNotNull(volumeName);
Preconditions.checkNotNull(bucketName); Preconditions.checkNotNull(bucketName);
metadataManager.readLock().lock(); // We don't take a lock in this path, since we walk the
try { // underlying table using an iterator. That automatically creates a
// snapshot of the data, so we don't need these locks at a higher level
// when we iterate.
return metadataManager.listKeys(volumeName, bucketName, return metadataManager.listKeys(volumeName, bucketName,
startKey, keyPrefix, maxKeys); startKey, keyPrefix, maxKeys);
} finally {
metadataManager.readLock().unlock();
}
} }
@Override @Override
public List<BlockGroup> getPendingDeletionKeys(final int count) public List<BlockGroup> getPendingDeletionKeys(final int count)
throws IOException { throws IOException {
metadataManager.readLock().lock(); //TODO: Fix this in later patches.
try { return null;
return metadataManager.getPendingDeletionKeys(count);
} finally {
metadataManager.readLock().unlock();
}
} }
@Override @Override
public void deletePendingDeletionKey(String objectKeyName) public void deletePendingDeletionKey(String objectKeyName)
throws IOException { throws IOException {
Preconditions.checkNotNull(objectKeyName); // TODO : Fix in later patches.
if (!objectKeyName.startsWith(OzoneConsts.DELETING_KEY_PREFIX)) {
throw new IllegalArgumentException("Invalid key name,"
+ " the name should be the key name with deleting prefix");
}
// Simply removes the entry from OM DB.
metadataManager.writeLock().lock();
try {
byte[] pendingDelKey = DFSUtil.string2Bytes(objectKeyName);
byte[] delKeyValue = metadataManager.get(pendingDelKey);
if (delKeyValue == null) {
throw new IOException("Failed to delete key " + objectKeyName
+ " because it is not found in DB");
}
metadataManager.delete(pendingDelKey);
} finally {
metadataManager.writeLock().unlock();
}
} }
@Override @Override
@ -510,23 +483,6 @@ public List<BlockGroup> getExpiredOpenKeys() throws IOException {
@Override @Override
public void deleteExpiredOpenKey(String objectKeyName) throws IOException { public void deleteExpiredOpenKey(String objectKeyName) throws IOException {
Preconditions.checkNotNull(objectKeyName); Preconditions.checkNotNull(objectKeyName);
if (!objectKeyName.startsWith(OzoneConsts.OPEN_KEY_PREFIX)) { // TODO: Fix this in later patches.
throw new IllegalArgumentException("Invalid key name,"
+ " the name should be the key name with open key prefix");
}
// Simply removes the entry from OM DB.
metadataManager.writeLock().lock();
try {
byte[] openKey = DFSUtil.string2Bytes(objectKeyName);
byte[] delKeyValue = metadataManager.get(openKey);
if (delKeyValue == null) {
throw new IOException("Failed to delete key " + objectKeyName
+ " because it is not found in DB");
}
metadataManager.delete(openKey);
} finally {
metadataManager.writeLock().unlock();
}
} }
} }

View File

@ -17,12 +17,12 @@
package org.apache.hadoop.ozone.om; package org.apache.hadoop.ozone.om;
import com.google.common.annotations.VisibleForTesting; import com.google.common.annotations.VisibleForTesting;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.common.BlockGroup; import org.apache.hadoop.ozone.common.BlockGroup;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo; import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
import org.apache.hadoop.ozone.om.helpers.OmVolumeArgs; import org.apache.hadoop.ozone.om.helpers.OmVolumeArgs;
import org.apache.hadoop.utils.BatchOperation; import org.apache.hadoop.utils.db.DBStore;
import org.apache.hadoop.utils.MetadataStore; import org.apache.hadoop.utils.db.Table;
import java.io.IOException; import java.io.IOException;
import java.util.List; import java.util.List;
@ -40,68 +40,47 @@ public interface OMMetadataManager {
/** /**
* Stop metadata manager. * Stop metadata manager.
*/ */
void stop() throws IOException; void stop() throws Exception;
/** /**
* Get metadata store. * Get metadata store.
*
* @return metadata store. * @return metadata store.
*/ */
@VisibleForTesting @VisibleForTesting
MetadataStore getStore(); DBStore getStore();
/** /**
* Returns the read lock used on Metadata DB. * Returns the read lock used on Metadata DB.
*
* @return readLock * @return readLock
*/ */
Lock readLock(); Lock readLock();
/** /**
* Returns the write lock used on Metadata DB. * Returns the write lock used on Metadata DB.
*
* @return writeLock * @return writeLock
*/ */
Lock writeLock(); Lock writeLock();
/**
* Returns the value associated with this key.
* @param key - key
* @return value
*/
byte[] get(byte[] key) throws IOException;
/**
* Puts a Key into Metadata DB.
* @param key - key
* @param value - value
*/
void put(byte[] key, byte[] value) throws IOException;
/**
* Deletes a Key from Metadata DB.
* @param key - key
*/
void delete(byte[] key) throws IOException;
/**
* Atomic write a batch of operations.
* @param batch
* @throws IOException
*/
void writeBatch(BatchOperation batch) throws IOException;
/** /**
* Given a volume return the corresponding DB key. * Given a volume return the corresponding DB key.
*
* @param volume - Volume name * @param volume - Volume name
*/ */
byte[] getVolumeKey(String volume); byte[] getVolumeKey(String volume);
/** /**
* Given a user return the corresponding DB key. * Given a user return the corresponding DB key.
*
* @param user - User name * @param user - User name
*/ */
byte[] getUserKey(String user); byte[] getUserKey(String user);
/** /**
* Given a volume and bucket, return the corresponding DB key. * Given a volume and bucket, return the corresponding DB key.
*
* @param volume - User name * @param volume - User name
* @param bucket - Bucket name * @param bucket - Bucket name
*/ */
@ -109,53 +88,37 @@ public interface OMMetadataManager {
/** /**
* Given a volume, bucket and a key, return the corresponding DB key. * Given a volume, bucket and a key, return the corresponding DB key.
*
* @param volume - volume name * @param volume - volume name
* @param bucket - bucket name * @param bucket - bucket name
* @param key - key name * @param key - key name
* @return bytes of DB key. * @return bytes of DB key.
*/ */
byte[] getDBKeyBytes(String volume, String bucket, String key); byte[] getOzoneKeyBytes(String volume, String bucket, String key);
/** /**
* Returns the DB key name of a deleted key in OM metadata store. * Returns the DB key name of a open key in OM metadata store. Should be
* The name for a deleted key has prefix #deleting# followed by * #open# prefix followed by actual key name.
* the actual key name. *
* @param keyName - key name * @param volume - volume name
* @return bytes of DB key. * @param bucket - bucket name
*/ * @param key - key name
byte[] getDeletedKeyName(byte[] keyName);
/**
* Returns the DB key name of a open key in OM metadata store.
* Should be #open# prefix followed by actual key name.
* @param keyName - key name
* @param id - the id for this open * @param id - the id for this open
* @return bytes of DB key. * @return bytes of DB key.
*/ */
byte[] getOpenKeyNameBytes(String keyName, int id); byte[] getOpenKeyBytes(String volume, String bucket, String key, long id);
/** /**
* Returns the full name of a key given volume name, bucket name and key name. * Given a volume, check if it is empty, i.e there are no buckets inside it.
* Generally done by padding certain delimiters.
* *
* @param volumeName - volume name
* @param bucketName - bucket name
* @param keyName - key name
* @return the full key name.
*/
String getKeyWithDBPrefix(String volumeName, String bucketName,
String keyName);
/**
* Given a volume, check if it is empty,
* i.e there are no buckets inside it.
* @param volume - Volume name * @param volume - Volume name
*/ */
boolean isVolumeEmpty(String volume) throws IOException; boolean isVolumeEmpty(String volume) throws IOException;
/** /**
* Given a volume/bucket, check if it is empty, * Given a volume/bucket, check if it is empty, i.e there are no keys inside
* i.e there are no keys inside it. * it.
*
* @param volume - Volume name * @param volume - Volume name
* @param bucket - Bucket name * @param bucket - Bucket name
* @return true if the bucket is empty * @return true if the bucket is empty
@ -163,46 +126,38 @@ String getKeyWithDBPrefix(String volumeName, String bucketName,
boolean isBucketEmpty(String volume, String bucket) throws IOException; boolean isBucketEmpty(String volume, String bucket) throws IOException;
/** /**
* Returns a list of buckets represented by {@link OmBucketInfo} * Returns a list of buckets represented by {@link OmBucketInfo} in the given
* in the given volume. * volume.
* *
* @param volumeName * @param volumeName the name of the volume. This argument is required, this
* the name of the volume. This argument is required, * method returns buckets in this given volume.
* this method returns buckets in this given volume. * @param startBucket the start bucket name. Only the buckets whose name is
* @param startBucket * after this value will be included in the result. This key is excluded from
* the start bucket name. Only the buckets whose name is * the result.
* after this value will be included in the result. * @param bucketPrefix bucket name prefix. Only the buckets whose name has
* This key is excluded from the result.
* @param bucketPrefix
* bucket name prefix. Only the buckets whose name has
* this prefix will be included in the result. * this prefix will be included in the result.
* @param maxNumOfBuckets * @param maxNumOfBuckets the maximum number of buckets to return. It ensures
* the maximum number of buckets to return. It ensures
* the size of the result will not exceed this limit. * the size of the result will not exceed this limit.
* @return a list of buckets. * @return a list of buckets.
* @throws IOException * @throws IOException
*/ */
List<OmBucketInfo> listBuckets(String volumeName, String startBucket, List<OmBucketInfo> listBuckets(String volumeName, String startBucket,
String bucketPrefix, int maxNumOfBuckets) throws IOException; String bucketPrefix, int maxNumOfBuckets)
throws IOException;
/** /**
* Returns a list of keys represented by {@link OmKeyInfo} * Returns a list of keys represented by {@link OmKeyInfo} in the given
* in the given bucket. * bucket.
* *
* @param volumeName * @param volumeName the name of the volume.
* the name of the volume. * @param bucketName the name of the bucket.
* @param bucketName * @param startKey the start key name, only the keys whose name is after this
* the name of the bucket. * value will be included in the result. This key is excluded from the
* @param startKey * result.
* the start key name, only the keys whose name is * @param keyPrefix key name prefix, only the keys whose name has this prefix
* after this value will be included in the result. * will be included in the result.
* This key is excluded from the result. * @param maxKeys the maximum number of keys to return. It ensures the size of
* @param keyPrefix * the result will not exceed this limit.
* key name prefix, only the keys whose name has
* this prefix will be included in the result.
* @param maxKeys
* the maximum number of keys to return. It ensures
* the size of the result will not exceed this limit.
* @return a list of keys. * @return a list of keys.
* @throws IOException * @throws IOException
*/ */
@ -211,18 +166,14 @@ List<OmKeyInfo> listKeys(String volumeName,
throws IOException; throws IOException;
/** /**
* Returns a list of volumes owned by a given user; if user is null, * Returns a list of volumes owned by a given user; if user is null, returns
* returns all volumes. * all volumes.
* *
* @param userName * @param userName volume owner
* volume owner * @param prefix the volume prefix used to filter the listing result.
* @param prefix * @param startKey the start volume name determines where to start listing
* the volume prefix used to filter the listing result. * from, this key is excluded from the result.
* @param startKey * @param maxKeys the maximum number of volumes to return.
* the start volume name determines where to start listing from,
* this key is excluded from the result.
* @param maxKeys
* the maximum number of volumes to return.
* @return a list of {@link OmVolumeArgs} * @return a list of {@link OmVolumeArgs}
* @throws IOException * @throws IOException
*/ */
@ -231,9 +182,9 @@ List<OmVolumeArgs> listVolumes(String userName, String prefix,
/** /**
* Returns a list of pending deletion key info that ups to the given count. * Returns a list of pending deletion key info that ups to the given count.
* Each entry is a {@link BlockGroup}, which contains the info about the * Each entry is a {@link BlockGroup}, which contains the info about the key
* key name and all its associated block IDs. A pending deletion key is * name and all its associated block IDs. A pending deletion key is stored
* stored with #deleting# prefix in OM DB. * with #deleting# prefix in OM DB.
* *
* @param count max number of keys to return. * @param count max number of keys to return.
* @return a list of {@link BlockGroup} represent keys and blocks. * @return a list of {@link BlockGroup} represent keys and blocks.
@ -250,4 +201,47 @@ List<OmVolumeArgs> listVolumes(String userName, String prefix,
* @throws IOException * @throws IOException
*/ */
List<BlockGroup> getExpiredOpenKeys() throws IOException; List<BlockGroup> getExpiredOpenKeys() throws IOException;
/**
* Returns the user Table.
*
* @return UserTable.
*/
Table getUserTable();
/**
* Returns the Volume Table.
*
* @return VolumeTable.
*/
Table getVolumeTable();
/**
* Returns the BucketTable.
*
* @return BucketTable.
*/
Table getBucketTable();
/**
* Returns the KeyTable.
*
* @return KeyTable.
*/
Table getKeyTable();
/**
* Get Deleted Table.
*
* @return Deleted Table.
*/
Table getDeletedTable();
/**
* Gets the OpenKeyTable.
*
* @return Table.
*/
Table getOpenKeyTable();
} }

View File

@ -19,77 +19,178 @@
import com.google.common.annotations.VisibleForTesting; import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Strings; import com.google.common.base.Strings;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import org.apache.commons.lang3.tuple.ImmutablePair; import org.apache.commons.lang3.StringUtils;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.hdds.client.BlockID; import org.apache.hadoop.hdds.client.BlockID;
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
import org.apache.hadoop.ozone.om.helpers.OmVolumeArgs;
import org.apache.hadoop.ozone.common.BlockGroup;
import org.apache.hadoop.hdds.conf.OzoneConfiguration; import org.apache.hadoop.hdds.conf.OzoneConfiguration;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.ozone.OzoneConsts; import org.apache.hadoop.ozone.OzoneConsts;
import org.apache.hadoop.ozone.common.BlockGroup;
import org.apache.hadoop.ozone.om.exceptions.OMException; import org.apache.hadoop.ozone.om.exceptions.OMException;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes; import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
import org.apache.hadoop.ozone.om.helpers.OmVolumeArgs;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.BucketInfo; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.BucketInfo;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.KeyInfo; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.KeyInfo;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.VolumeInfo; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.VolumeInfo;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.VolumeList; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.VolumeList;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
import org.apache.hadoop.utils.BatchOperation; import org.apache.hadoop.utils.db.DBStore;
import org.apache.hadoop.utils.MetadataKeyFilters; import org.apache.hadoop.utils.db.DBStoreBuilder;
import org.apache.hadoop.utils.MetadataKeyFilters.KeyPrefixFilter; import org.apache.hadoop.utils.db.Table;
import org.apache.hadoop.utils.MetadataKeyFilters.MetadataKeyFilter; import org.apache.hadoop.utils.db.TableIterator;
import org.apache.hadoop.utils.MetadataStore; import org.eclipse.jetty.util.StringUtil;
import org.apache.hadoop.utils.MetadataStoreBuilder; import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File; import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.ArrayList;
import java.util.Map; import java.util.Map;
import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock; import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock; import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import static org.apache.hadoop.hdds.server.ServerUtils.getOzoneMetaDirPath;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS; import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS_DEFAULT; import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS_DEFAULT;
import static org.apache.hadoop.ozone.OzoneConsts.DELETING_KEY_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OM_DB_NAME; import static org.apache.hadoop.ozone.OzoneConsts.OM_DB_NAME;
import static org.apache.hadoop.ozone.OzoneConsts.OPEN_KEY_ID_DELIMINATOR; import static org.apache.hadoop.ozone.OzoneConsts.OM_KEY_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OPEN_KEY_PREFIX;
import static org.apache.hadoop.ozone.om.OMConfigKeys
.OZONE_OM_DB_CACHE_SIZE_DEFAULT;
import static org.apache.hadoop.ozone.om.OMConfigKeys
.OZONE_OM_DB_CACHE_SIZE_MB;
import static org.apache.hadoop.hdds.server.ServerUtils.getOzoneMetaDirPath;
/** /**
* Ozone metadata manager interface. * Ozone metadata manager interface.
*/ */
public class OmMetadataManagerImpl implements OMMetadataManager { public class OmMetadataManagerImpl implements OMMetadataManager {
private static final Logger LOG =
LoggerFactory.getLogger(OmMetadataManagerImpl.class);
private final MetadataStore store; /**
* OM RocksDB Structure .
* <p>
* OM DB stores metadata as KV pairs in different column families.
* <p>
* OM DB Schema:
* |-------------------------------------------------------------------|
* | Column Family | VALUE |
* |-------------------------------------------------------------------|
* | userTable | user->VolumeList |
* |-------------------------------------------------------------------|
* | volumeTable | /volume->VolumeInfo |
* |-------------------------------------------------------------------|
* | bucketTable | /volume/bucket-> BucketInfo |
* |-------------------------------------------------------------------|
* | keyTable | /volumeName/bucketName/keyName->KeyInfo |
* |-------------------------------------------------------------------|
* | deletedTable | /volumeName/bucketName/keyName->KeyInfo |
* |-------------------------------------------------------------------|
* | openKey | /volumeName/bucketName/keyName/id->KeyInfo |
* |-------------------------------------------------------------------|
*/
private static final String USER_TABLE = "userTable";
private static final String VOLUME_TABLE = "volumeTable";
private static final String BUCKET_TABLE = "bucketTable";
private static final String KEY_TABLE = "keyTable";
private static final String DELETED_TABLE = "deletedTable";
private static final String OPEN_KEY_TABLE = "openKeyTable";
private final DBStore store;
// TODO: Make this lock move into Table instead of *ONE* lock for the whole
// DB.
private final ReadWriteLock lock; private final ReadWriteLock lock;
private final long openKeyExpireThresholdMS; private final long openKeyExpireThresholdMS;
private final Table userTable;
private final Table volumeTable;
private final Table bucketTable;
private final Table keyTable;
private final Table deletedTable;
private final Table openKeyTable;
public OmMetadataManagerImpl(OzoneConfiguration conf) throws IOException { public OmMetadataManagerImpl(OzoneConfiguration conf) throws IOException {
File metaDir = getOzoneMetaDirPath(conf); File metaDir = getOzoneMetaDirPath(conf);
final int cacheSize = conf.getInt(OZONE_OM_DB_CACHE_SIZE_MB,
OZONE_OM_DB_CACHE_SIZE_DEFAULT);
File omDBFile = new File(metaDir.getPath(), OM_DB_NAME);
this.store = MetadataStoreBuilder.newBuilder()
.setConf(conf)
.setDbFile(omDBFile)
.setCacheSize(cacheSize * OzoneConsts.MB)
.build();
this.lock = new ReentrantReadWriteLock(); this.lock = new ReentrantReadWriteLock();
this.openKeyExpireThresholdMS = 1000 * conf.getInt( this.openKeyExpireThresholdMS = 1000 * conf.getInt(
OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS, OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS,
OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS_DEFAULT); OZONE_OPEN_KEY_EXPIRE_THRESHOLD_SECONDS_DEFAULT);
this.store = DBStoreBuilder.newBuilder(conf)
.setName(OM_DB_NAME)
.setPath(Paths.get(metaDir.getPath()))
.addTable(USER_TABLE)
.addTable(VOLUME_TABLE)
.addTable(BUCKET_TABLE)
.addTable(KEY_TABLE)
.addTable(DELETED_TABLE)
.addTable(OPEN_KEY_TABLE)
.build();
userTable = this.store.getTable(USER_TABLE);
checkTableStatus(userTable, USER_TABLE);
volumeTable = this.store.getTable(VOLUME_TABLE);
checkTableStatus(volumeTable, VOLUME_TABLE);
bucketTable = this.store.getTable(BUCKET_TABLE);
checkTableStatus(bucketTable, BUCKET_TABLE);
keyTable = this.store.getTable(KEY_TABLE);
checkTableStatus(keyTable, KEY_TABLE);
deletedTable = this.store.getTable(DELETED_TABLE);
checkTableStatus(deletedTable, DELETED_TABLE);
openKeyTable = this.store.getTable(OPEN_KEY_TABLE);
checkTableStatus(openKeyTable, OPEN_KEY_TABLE);
}
@Override
public Table getUserTable() {
return userTable;
}
@Override
public Table getVolumeTable() {
return volumeTable;
}
@Override
public Table getBucketTable() {
return bucketTable;
}
@Override
public Table getKeyTable() {
return keyTable;
}
@Override
public Table getDeletedTable() {
return deletedTable;
}
@Override
public Table getOpenKeyTable() {
return openKeyTable;
}
private void checkTableStatus(Table table, String name) throws IOException {
String logMessage = "Unable to get a reference to %s table. Cannot " +
"continue.";
String errMsg = "Inconsistent DB state, Table - %s. Please check the logs" +
"for more info.";
if (table == null) {
LOG.error(String.format(logMessage, name));
throw new IOException(String.format(errMsg, name));
}
} }
/** /**
@ -104,7 +205,7 @@ public void start() {
* Stop metadata manager. * Stop metadata manager.
*/ */
@Override @Override
public void stop() throws IOException { public void stop() throws Exception {
if (store != null) { if (store != null) {
store.close(); store.close();
} }
@ -112,86 +213,75 @@ public void stop() throws IOException {
/** /**
* Get metadata store. * Get metadata store.
*
* @return store - metadata store. * @return store - metadata store.
*/ */
@VisibleForTesting @VisibleForTesting
@Override @Override
public MetadataStore getStore() { public DBStore getStore() {
return store; return store;
} }
/** /**
* Given a volume return the corresponding DB key. * Given a volume return the corresponding DB key.
*
* @param volume - Volume name * @param volume - Volume name
*/ */
@Override
public byte[] getVolumeKey(String volume) { public byte[] getVolumeKey(String volume) {
String dbVolumeName = OzoneConsts.OM_VOLUME_PREFIX + volume; return DFSUtil.string2Bytes(OzoneConsts.OM_KEY_PREFIX + volume);
return DFSUtil.string2Bytes(dbVolumeName);
} }
/** /**
* Given a user return the corresponding DB key. * Given a user return the corresponding DB key.
*
* @param user - User name * @param user - User name
*/ */
@Override
public byte[] getUserKey(String user) { public byte[] getUserKey(String user) {
String dbUserName = OzoneConsts.OM_USER_PREFIX + user; return DFSUtil.string2Bytes(user);
return DFSUtil.string2Bytes(dbUserName);
} }
/** /**
* Given a volume and bucket, return the corresponding DB key. * Given a volume and bucket, return the corresponding DB key.
*
* @param volume - User name * @param volume - User name
* @param bucket - Bucket name * @param bucket - Bucket name
*/ */
@Override
public byte[] getBucketKey(String volume, String bucket) { public byte[] getBucketKey(String volume, String bucket) {
String bucketKeyString = OzoneConsts.OM_VOLUME_PREFIX + volume StringBuilder builder =
+ OzoneConsts.OM_BUCKET_PREFIX + bucket; new StringBuilder().append(OM_KEY_PREFIX).append(volume);
return DFSUtil.string2Bytes(bucketKeyString);
}
/** if (StringUtils.isNotBlank(bucket)) {
* @param volume builder.append(OM_KEY_PREFIX).append(bucket);
* @param bucket
* @return
*/
private String getBucketWithDBPrefix(String volume, String bucket) {
StringBuffer sb = new StringBuffer();
sb.append(OzoneConsts.OM_VOLUME_PREFIX)
.append(volume)
.append(OzoneConsts.OM_BUCKET_PREFIX);
if (!Strings.isNullOrEmpty(bucket)) {
sb.append(bucket);
} }
return sb.toString(); return DFSUtil.string2Bytes(builder.toString());
} }
@Override @Override
public String getKeyWithDBPrefix(String volume, String bucket, String key) { public byte[] getOzoneKeyBytes(String volume, String bucket, String key) {
String keyVB = OzoneConsts.OM_KEY_PREFIX + volume StringBuilder builder = new StringBuilder()
+ OzoneConsts.OM_KEY_PREFIX + bucket .append(OM_KEY_PREFIX).append(volume);
+ OzoneConsts.OM_KEY_PREFIX; // TODO : Throw if the Bucket is null?
return Strings.isNullOrEmpty(key) ? keyVB : keyVB + key; builder.append(OM_KEY_PREFIX).append(bucket);
if (StringUtil.isNotBlank(key)) {
builder.append(OM_KEY_PREFIX).append(key);
}
return DFSUtil.string2Bytes(builder.toString());
} }
@Override @Override
public byte[] getDBKeyBytes(String volume, String bucket, String key) { public byte[] getOpenKeyBytes(String volume, String bucket,
return DFSUtil.string2Bytes(getKeyWithDBPrefix(volume, bucket, key)); String key, long id) {
} String openKey = OM_KEY_PREFIX + volume + OM_KEY_PREFIX + bucket +
OM_KEY_PREFIX + key + OM_KEY_PREFIX + id;
@Override return DFSUtil.string2Bytes(openKey);
public byte[] getDeletedKeyName(byte[] keyName) {
return DFSUtil.string2Bytes(
DELETING_KEY_PREFIX + DFSUtil.bytes2String(keyName));
}
@Override
public byte[] getOpenKeyNameBytes(String keyName, int id) {
return DFSUtil.string2Bytes(OPEN_KEY_PREFIX + id +
OPEN_KEY_ID_DELIMINATOR + keyName);
} }
/** /**
* Returns the read lock used on Metadata DB. * Returns the read lock used on Metadata DB.
*
* @return readLock * @return readLock
*/ */
@Override @Override
@ -201,6 +291,7 @@ public Lock readLock() {
/** /**
* Returns the write lock used on Metadata DB. * Returns the write lock used on Metadata DB.
*
* @return writeLock * @return writeLock
*/ */
@Override @Override
@ -209,71 +300,79 @@ public Lock writeLock() {
} }
/** /**
* Returns the value associated with this key. * Returns true if the firstArray startsWith the bytes of secondArray.
* @param key - key *
* @return value * @param firstArray - Byte array
* @param secondArray - Byte array
* @return true if the first array bytes match the bytes in the second array.
*/ */
@Override private boolean startsWith(byte[] firstArray, byte[] secondArray) {
public byte[] get(byte[] key) throws IOException {
return store.get(key); if (firstArray == null) {
// if both are null, then the arrays match, else if first is null and
// second is not, then this function returns false.
return secondArray == null;
} }
/**
* Puts a Key into Metadata DB. if (secondArray != null) {
* @param key - key // If the second array is longer then first array cannot be starting with
* @param value - value // the bytes of second array.
*/ if (secondArray.length > firstArray.length) {
@Override return false;
public void put(byte[] key, byte[] value) throws IOException {
store.put(key, value);
} }
/** for (int ndx = 0; ndx < secondArray.length; ndx++) {
* Deletes a Key from Metadata DB. if (firstArray[ndx] != secondArray[ndx]) {
* @param key - key return false;
*/
public void delete(byte[] key) throws IOException {
store.delete(key);
} }
}
@Override return true; //match, return true.
public void writeBatch(BatchOperation batch) throws IOException { }
this.store.writeBatch(batch); return false; // if first is not null and second is null, we define that
// array does not start with same chars.
} }
/** /**
* Given a volume, check if it is empty, i.e there are no buckets inside it. * Given a volume, check if it is empty, i.e there are no buckets inside it.
* We iterate in the bucket table and see if there is any key that starts with
* the volume prefix. We actually look for /volume/, since if we don't have
* the trailing slash it is possible that we might match some other volume.
* <p>
* For example, vol1 and vol122 might match, to avoid that we look for /vol1/
*
* @param volume - Volume name * @param volume - Volume name
* @return true if the volume is empty * @return true if the volume is empty
*/ */
@Override
public boolean isVolumeEmpty(String volume) throws IOException { public boolean isVolumeEmpty(String volume) throws IOException {
String dbVolumeRootName = OzoneConsts.OM_VOLUME_PREFIX + volume byte[] volumePrefix = getVolumeKey(volume + OM_KEY_PREFIX);
+ OzoneConsts.OM_BUCKET_PREFIX; try (TableIterator<Table.KeyValue> bucketIter = bucketTable.iterator()) {
byte[] dbVolumeRootKey = DFSUtil.string2Bytes(dbVolumeRootName); Table.KeyValue kv = bucketIter.seek(volumePrefix);
ImmutablePair<byte[], byte[]> volumeRoot = if (kv != null && startsWith(kv.getKey(), volumePrefix)) {
store.peekAround(0, dbVolumeRootKey); return false; // we found at least one bucket with this volume prefix.
if (volumeRoot != null) { }
return !DFSUtil.bytes2String(volumeRoot.getKey())
.startsWith(dbVolumeRootName);
} }
return true; return true;
} }
/** /**
* Given a volume/bucket, check if it is empty, * Given a volume/bucket, check if it is empty, i.e there are no keys inside
* i.e there are no keys inside it. * it. Prefix is /volume/bucket/, and we lookup the keyTable.
*
* @param volume - Volume name * @param volume - Volume name
* @param bucket - Bucket name * @param bucket - Bucket name
* @return true if the bucket is empty * @return true if the bucket is empty
*/ */
@Override
public boolean isBucketEmpty(String volume, String bucket) public boolean isBucketEmpty(String volume, String bucket)
throws IOException { throws IOException {
String keyRootName = getKeyWithDBPrefix(volume, bucket, null); byte[] keyPrefix = getBucketKey(volume, bucket + OM_KEY_PREFIX);
byte[] keyRoot = DFSUtil.string2Bytes(keyRootName); try (TableIterator<Table.KeyValue> keyIter = keyTable.iterator()) {
ImmutablePair<byte[], byte[]> firstKey = store.peekAround(0, keyRoot); Table.KeyValue kv = keyIter.seek(keyPrefix);
if (firstKey != null) { if (kv != null && startsWith(kv.getKey(), keyPrefix)) {
return !DFSUtil.bytes2String(firstKey.getKey()) return false; // we found at least one key with this vol/bucket prefix.
.startsWith(keyRootName); }
} }
return true; return true;
} }
@ -292,42 +391,54 @@ public List<OmBucketInfo> listBuckets(final String volumeName,
} }
byte[] volumeNameBytes = getVolumeKey(volumeName); byte[] volumeNameBytes = getVolumeKey(volumeName);
if (store.get(volumeNameBytes) == null) { if (volumeTable.get(volumeNameBytes) == null) {
throw new OMException("Volume " + volumeName + " not found.", throw new OMException("Volume " + volumeName + " not found.",
ResultCodes.FAILED_VOLUME_NOT_FOUND); ResultCodes.FAILED_VOLUME_NOT_FOUND);
} }
// A bucket starts with /#volume/#bucket_prefix byte[] startKey;
MetadataKeyFilter filter = (preKey, currentKey, nextKey) -> { boolean skipStartKey = false;
if (currentKey != null) { if (StringUtil.isNotBlank(startBucket)) {
String bucketNamePrefix = // if the user has specified a start key, we need to seek to that key
getBucketWithDBPrefix(volumeName, bucketPrefix); // and avoid that key in the response set.
String bucket = DFSUtil.bytes2String(currentKey); startKey = getBucketKey(volumeName, startBucket);
return bucket.startsWith(bucketNamePrefix); skipStartKey = true;
}
return false;
};
List<Map.Entry<byte[], byte[]>> rangeResult;
if (!Strings.isNullOrEmpty(startBucket)) {
// Since we are excluding start key from the result,
// the maxNumOfBuckets is incremented.
rangeResult = store.getSequentialRangeKVs(
getBucketKey(volumeName, startBucket),
maxNumOfBuckets + 1, filter);
if (!rangeResult.isEmpty()) {
//Remove start key from result.
rangeResult.remove(0);
}
} else { } else {
rangeResult = store.getSequentialRangeKVs(null, maxNumOfBuckets, filter); // If the user has specified a prefix key, we need to get to the first
// of the keys with the prefix match. We can leverage RocksDB to do that.
// However, if the user has specified only a prefix, we cannot skip
// the first prefix key we see, the boolean skipStartKey allows us to
// skip the startkey or not depending on what patterns are specified.
startKey = getBucketKey(volumeName, bucketPrefix);
} }
for (Map.Entry<byte[], byte[]> entry : rangeResult) { byte[] seekPrefix;
OmBucketInfo info = OmBucketInfo.getFromProtobuf( if (StringUtil.isNotBlank(bucketPrefix)) {
BucketInfo.parseFrom(entry.getValue())); seekPrefix = getBucketKey(volumeName, bucketPrefix);
result.add(info); } else {
seekPrefix = getVolumeKey(volumeName + OM_KEY_PREFIX);
}
int currentCount = 0;
try (TableIterator<Table.KeyValue> bucketIter = bucketTable.iterator()) {
Table.KeyValue kv = bucketIter.seek(startKey);
while (currentCount < maxNumOfBuckets && bucketIter.hasNext()) {
kv = bucketIter.next();
// Skip the Start Bucket if needed.
if (kv != null && skipStartKey &&
Arrays.equals(kv.getKey(), startKey)) {
continue;
}
if (kv != null && startsWith(kv.getKey(), seekPrefix)) {
result.add(OmBucketInfo.getFromProtobuf(
BucketInfo.parseFrom(kv.getValue())));
currentCount++;
} else {
// The SeekPrefix does not match any more, we can break out of the
// loop.
break;
}
}
} }
return result; return result;
} }
@ -347,33 +458,47 @@ public List<OmKeyInfo> listKeys(String volumeName, String bucketName,
} }
byte[] bucketNameBytes = getBucketKey(volumeName, bucketName); byte[] bucketNameBytes = getBucketKey(volumeName, bucketName);
if (store.get(bucketNameBytes) == null) { if (getBucketTable().get(bucketNameBytes) == null) {
throw new OMException("Bucket " + bucketName + " not found.", throw new OMException("Bucket " + bucketName + " not found.",
ResultCodes.FAILED_BUCKET_NOT_FOUND); ResultCodes.FAILED_BUCKET_NOT_FOUND);
} }
MetadataKeyFilter filter = new KeyPrefixFilter() byte[] seekKey;
.addFilter(getKeyWithDBPrefix(volumeName, bucketName, keyPrefix)); boolean skipStartKey = false;
if (StringUtil.isNotBlank(startKey)) {
List<Map.Entry<byte[], byte[]>> rangeResult; // Seek to the specified key.
if (!Strings.isNullOrEmpty(startKey)) { seekKey = getOzoneKeyBytes(volumeName, bucketName, startKey);
//Since we are excluding start key from the result, skipStartKey = true;
// the maxNumOfBuckets is incremented.
rangeResult = store.getSequentialRangeKVs(
getDBKeyBytes(volumeName, bucketName, startKey),
maxKeys + 1, filter);
if (!rangeResult.isEmpty()) {
//Remove start key from result.
rangeResult.remove(0);
}
} else { } else {
rangeResult = store.getSequentialRangeKVs(null, maxKeys, filter); // This allows us to seek directly to the first key with the right prefix.
seekKey = getOzoneKeyBytes(volumeName, bucketName, keyPrefix);
} }
for (Map.Entry<byte[], byte[]> entry : rangeResult) { byte[] seekPrefix;
OmKeyInfo info = OmKeyInfo.getFromProtobuf( if (StringUtil.isNotBlank(keyPrefix)) {
KeyInfo.parseFrom(entry.getValue())); seekPrefix = getOzoneKeyBytes(volumeName, bucketName, keyPrefix);
result.add(info); } else {
seekPrefix = getBucketKey(volumeName, bucketName + OM_KEY_PREFIX);
}
int currentCount = 0;
try (TableIterator<Table.KeyValue> keyIter = getKeyTable().iterator()) {
Table.KeyValue kv = keyIter.seek(seekKey);
while (currentCount < maxKeys && keyIter.hasNext()) {
kv = keyIter.next();
// Skip the Start key if needed.
if (kv != null && skipStartKey && Arrays.equals(kv.getKey(), seekKey)) {
continue;
}
if (kv != null && startsWith(kv.getKey(), seekPrefix)) {
result.add(OmKeyInfo.getFromProtobuf(
KeyInfo.parseFrom(kv.getValue())));
currentCount++;
} else {
// The SeekPrefix does not match any more, we can break out of the
// loop.
break;
}
}
} }
return result; return result;
} }
@ -383,11 +508,11 @@ public List<OmVolumeArgs> listVolumes(String userName,
String prefix, String startKey, int maxKeys) throws IOException { String prefix, String startKey, int maxKeys) throws IOException {
List<OmVolumeArgs> result = Lists.newArrayList(); List<OmVolumeArgs> result = Lists.newArrayList();
VolumeList volumes; VolumeList volumes;
if (Strings.isNullOrEmpty(userName)) { if (StringUtil.isBlank(userName)) {
volumes = getAllVolumes(); throw new OMException("User name is required to list Volumes.",
} else { ResultCodes.FAILED_USER_NOT_FOUND);
volumes = getVolumesByUser(userName);
} }
volumes = getVolumesByUser(userName);
if (volumes == null || volumes.getVolumeNamesCount() == 0) { if (volumes == null || volumes.getVolumeNamesCount() == 0) {
return result; return result;
@ -406,7 +531,7 @@ public List<OmVolumeArgs> listVolumes(String userName,
continue; continue;
} }
if (startKeyFound && result.size() < maxKeys) { if (startKeyFound && result.size() < maxKeys) {
byte[] volumeInfo = store.get(this.getVolumeKey(volumeName)); byte[] volumeInfo = getVolumeTable().get(this.getVolumeKey(volumeName));
if (volumeInfo == null) { if (volumeInfo == null) {
// Could not get volume info by given volume name, // Could not get volume info by given volume name,
// since the volume name is loaded from db, // since the volume name is loaded from db,
@ -433,7 +558,7 @@ private VolumeList getVolumesByUser(byte[] userNameKey)
throws OMException { throws OMException {
VolumeList volumes = null; VolumeList volumes = null;
try { try {
byte[] volumesInBytes = store.get(userNameKey); byte[] volumesInBytes = getUserTable().get(userNameKey);
if (volumesInBytes == null) { if (volumesInBytes == null) {
// No volume found for this user, return an empty list // No volume found for this user, return an empty list
return VolumeList.newBuilder().build(); return VolumeList.newBuilder().build();
@ -447,32 +572,12 @@ private VolumeList getVolumesByUser(byte[] userNameKey)
return volumes; return volumes;
} }
private VolumeList getAllVolumes() throws IOException {
// Scan all users in database
KeyPrefixFilter filter =
new KeyPrefixFilter().addFilter(OzoneConsts.OM_USER_PREFIX);
// We are not expecting a huge number of users per cluster,
// it should be fine to scan all users in db and return us a
// list of volume names in string per user.
List<Map.Entry<byte[], byte[]>> rangeKVs = store
.getSequentialRangeKVs(null, Integer.MAX_VALUE, filter);
VolumeList.Builder builder = VolumeList.newBuilder();
for (Map.Entry<byte[], byte[]> entry : rangeKVs) {
VolumeList volumes = this.getVolumesByUser(entry.getKey());
builder.addAllVolumeNames(volumes.getVolumeNamesList());
}
return builder.build();
}
@Override @Override
public List<BlockGroup> getPendingDeletionKeys(final int count) public List<BlockGroup> getPendingDeletionKeys(final int count)
throws IOException { throws IOException {
List<BlockGroup> keyBlocksList = Lists.newArrayList(); List<BlockGroup> keyBlocksList = Lists.newArrayList();
List<Map.Entry<byte[], byte[]>> rangeResult = // TODO: Fix this later, Not part of this patch.
store.getRangeKVs(null, count, List<Map.Entry<byte[], byte[]>> rangeResult = Collections.emptyList();
MetadataKeyFilters.getDeletingKeyFilter());
for (Map.Entry<byte[], byte[]> entry : rangeResult) { for (Map.Entry<byte[], byte[]> entry : rangeResult) {
OmKeyInfo info = OmKeyInfo info =
OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(entry.getValue())); OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(entry.getValue()));
@ -497,11 +602,9 @@ public List<BlockGroup> getPendingDeletionKeys(final int count)
public List<BlockGroup> getExpiredOpenKeys() throws IOException { public List<BlockGroup> getExpiredOpenKeys() throws IOException {
List<BlockGroup> keyBlocksList = Lists.newArrayList(); List<BlockGroup> keyBlocksList = Lists.newArrayList();
long now = Time.now(); long now = Time.now();
final MetadataKeyFilter openKeyFilter = // TODO: Fix the getExpiredOpenKeys, Not part of this patch.
new KeyPrefixFilter().addFilter(OPEN_KEY_PREFIX); List<Map.Entry<byte[], byte[]>> rangeResult = Collections.emptyList();
List<Map.Entry<byte[], byte[]>> rangeResult =
store.getSequentialRangeKVs(null, Integer.MAX_VALUE,
openKeyFilter);
for (Map.Entry<byte[], byte[]> entry : rangeResult) { for (Map.Entry<byte[], byte[]> entry : rangeResult) {
OmKeyInfo info = OmKeyInfo info =
OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(entry.getValue())); OmKeyInfo.getFromProtobuf(KeyInfo.parseFrom(entry.getValue()));

View File

@ -21,14 +21,27 @@
import com.google.common.base.Preconditions; import com.google.common.base.Preconditions;
import com.google.protobuf.BlockingService; import com.google.protobuf.BlockingService;
import org.apache.hadoop.classification.InterfaceAudience; import org.apache.hadoop.classification.InterfaceAudience;
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
import org.apache.hadoop.hdds.protocol.DatanodeDetails; import org.apache.hadoop.hdds.protocol.DatanodeDetails;
import org.apache.hadoop.hdds.protocol.proto.HddsProtos;
import org.apache.hadoop.hdds.scm.ScmInfo;
import org.apache.hadoop.hdds.scm.protocol.ScmBlockLocationProtocol;
import org.apache.hadoop.hdds.scm.protocol.StorageContainerLocationProtocol;
import org.apache.hadoop.hdds.scm.protocolPB.ScmBlockLocationProtocolClientSideTranslatorPB;
import org.apache.hadoop.hdds.scm.protocolPB.ScmBlockLocationProtocolPB;
import org.apache.hadoop.hdds.scm.protocolPB.StorageContainerLocationProtocolClientSideTranslatorPB;
import org.apache.hadoop.hdds.scm.protocolPB.StorageContainerLocationProtocolPB;
import org.apache.hadoop.hdds.server.ServiceRuntimeInfoImpl;
import org.apache.hadoop.hdfs.DFSUtil; import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.ipc.Client; import org.apache.hadoop.ipc.Client;
import org.apache.hadoop.ipc.ProtobufRpcEngine; import org.apache.hadoop.ipc.ProtobufRpcEngine;
import org.apache.hadoop.ipc.RPC; import org.apache.hadoop.ipc.RPC;
import org.apache.hadoop.hdds.server.ServiceRuntimeInfoImpl; import org.apache.hadoop.metrics2.lib.DefaultMetricsSystem;
import org.apache.hadoop.metrics2.util.MBeans;
import org.apache.hadoop.net.NetUtils;
import org.apache.hadoop.ozone.common.Storage.StorageState; import org.apache.hadoop.ozone.common.Storage.StorageState;
import org.apache.hadoop.ozone.om.exceptions.OMException; import org.apache.hadoop.ozone.om.exceptions.OMException;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes;
import org.apache.hadoop.ozone.om.helpers.OmBucketArgs; import org.apache.hadoop.ozone.om.helpers.OmBucketArgs;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo; import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.om.helpers.OmKeyArgs; import org.apache.hadoop.ozone.om.helpers.OmKeyArgs;
@ -39,36 +52,12 @@
import org.apache.hadoop.ozone.om.helpers.ServiceInfo; import org.apache.hadoop.ozone.om.helpers.ServiceInfo;
import org.apache.hadoop.ozone.om.protocol.OzoneManagerProtocol; import org.apache.hadoop.ozone.om.protocol.OzoneManagerProtocol;
import org.apache.hadoop.ozone.om.protocolPB.OzoneManagerProtocolPB; import org.apache.hadoop.ozone.om.protocolPB.OzoneManagerProtocolPB;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.OzoneAclInfo;
import org.apache.hadoop.metrics2.lib.DefaultMetricsSystem; import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.ServicePort;
import org.apache.hadoop.metrics2.util.MBeans;
import org.apache.hadoop.net.NetUtils;
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
import org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos
.ServicePort;
import org.apache.hadoop.ozone.protocol.proto
.OzoneManagerProtocolProtos.OzoneAclInfo;
import org.apache.hadoop.hdds.protocol.proto.HddsProtos;
import org.apache.hadoop.ozone.protocolPB.OzoneManagerProtocolServerSideTranslatorPB; import org.apache.hadoop.ozone.protocolPB.OzoneManagerProtocolServerSideTranslatorPB;
import org.apache.hadoop.hdds.scm.ScmInfo;
import org.apache.hadoop.hdds.scm.protocol.ScmBlockLocationProtocol;
import org.apache.hadoop.hdds.scm.protocol.StorageContainerLocationProtocol;
import org.apache.hadoop.hdds.scm.protocolPB
.ScmBlockLocationProtocolClientSideTranslatorPB;
import org.apache.hadoop.hdds.scm.protocolPB.ScmBlockLocationProtocolPB;
import org.apache.hadoop.hdds.scm.protocolPB
.StorageContainerLocationProtocolClientSideTranslatorPB;
import org.apache.hadoop.hdds.scm.protocolPB.StorageContainerLocationProtocolPB;
import org.apache.hadoop.security.UserGroupInformation; import org.apache.hadoop.security.UserGroupInformation;
import org.apache.hadoop.util.GenericOptionsParser; import org.apache.hadoop.util.GenericOptionsParser;
import org.apache.hadoop.util.StringUtils; import org.apache.hadoop.util.StringUtils;
import static org.apache.hadoop.hdds.HddsUtils.getScmAddressForBlockClients;
import static org.apache.hadoop.hdds.HddsUtils.getScmAddressForClients;
import static org.apache.hadoop.hdds.HddsUtils.isHddsEnabled;
import static org.apache.hadoop.ozone.OmUtils.getOmAddress;
import static org.apache.hadoop.hdds.server.ServerUtils
.updateRPCListenAddress;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -81,18 +70,17 @@
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import static org.apache.hadoop.hdds.HddsUtils.getScmAddressForBlockClients;
import static org.apache.hadoop.hdds.HddsUtils.getScmAddressForClients;
import static org.apache.hadoop.hdds.HddsUtils.isHddsEnabled;
import static org.apache.hadoop.hdds.protocol.proto.HddsProtos.NodeState.HEALTHY;
import static org.apache.hadoop.hdds.server.ServerUtils.updateRPCListenAddress;
import static org.apache.hadoop.ozone.OmUtils.getOmAddress;
import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_ENABLED; import static org.apache.hadoop.ozone.OzoneConfigKeys.OZONE_ENABLED;
import static org.apache.hadoop.ozone.om.OMConfigKeys import static org.apache.hadoop.ozone.om.OMConfigKeys.OZONE_OM_ADDRESS_KEY;
.OZONE_OM_ADDRESS_KEY; import static org.apache.hadoop.ozone.om.OMConfigKeys.OZONE_OM_HANDLER_COUNT_DEFAULT;
import static org.apache.hadoop.ozone.om.OMConfigKeys import static org.apache.hadoop.ozone.om.OMConfigKeys.OZONE_OM_HANDLER_COUNT_KEY;
.OZONE_OM_HANDLER_COUNT_DEFAULT; import static org.apache.hadoop.ozone.protocol.proto.OzoneManagerProtocolProtos.OzoneManagerService.newReflectiveBlockingService;
import static org.apache.hadoop.ozone.om.OMConfigKeys
.OZONE_OM_HANDLER_COUNT_KEY;
import static org.apache.hadoop.ozone.protocol.proto
.OzoneManagerProtocolProtos.OzoneManagerService
.newReflectiveBlockingService;
import static org.apache.hadoop.hdds.protocol.proto.HddsProtos
.NodeState.HEALTHY;
import static org.apache.hadoop.util.ExitUtil.terminate; import static org.apache.hadoop.util.ExitUtil.terminate;
/** /**
@ -108,33 +96,6 @@ public final class OzoneManager extends ServiceRuntimeInfoImpl
"Usage: \n ozone om [genericOptions] " + "[ " "Usage: \n ozone om [genericOptions] " + "[ "
+ StartupOption.CREATEOBJECTSTORE.getName() + " ]\n " + "ozone om [ " + StartupOption.CREATEOBJECTSTORE.getName() + " ]\n " + "ozone om [ "
+ StartupOption.HELP.getName() + " ]\n"; + StartupOption.HELP.getName() + " ]\n";
/** Startup options. */
public enum StartupOption {
CREATEOBJECTSTORE("-createObjectStore"),
HELP("-help"),
REGULAR("-regular");
private final String name;
StartupOption(String arg) {
this.name = arg;
}
public String getName() {
return name;
}
public static StartupOption parse(String value) {
for (StartupOption option : StartupOption.values()) {
if (option.name.equalsIgnoreCase(value)) {
return option;
}
}
return null;
}
}
private final OzoneConfiguration configuration; private final OzoneConfiguration configuration;
private final RPC.Server omRpcServer; private final RPC.Server omRpcServer;
private final InetSocketAddress omRpcAddress; private final InetSocketAddress omRpcAddress;
@ -238,20 +199,6 @@ private static StorageContainerLocationProtocol getScmContainerClient(
return scmContainerClient; return scmContainerClient;
} }
@VisibleForTesting
public KeyManager getKeyManager() {
return keyManager;
}
@VisibleForTesting
public ScmInfo getScmInfo() throws IOException {
return scmBlockClient.getScmInfo();
}
@VisibleForTesting
public OMStorage getOmStorage() {
return omStorage;
}
/** /**
* Starts an RPC server, if configured. * Starts an RPC server, if configured.
* *
@ -260,7 +207,6 @@ public OMStorage getOmStorage() {
* @param protocol RPC protocol provided by RPC server * @param protocol RPC protocol provided by RPC server
* @param instance RPC protocol implementation instance * @param instance RPC protocol implementation instance
* @param handlerCount RPC server handler count * @param handlerCount RPC server handler count
*
* @return RPC server * @return RPC server
* @throws IOException if there is an I/O error while creating RPC server * @throws IOException if there is an I/O error while creating RPC server
*/ */
@ -281,18 +227,6 @@ private static RPC.Server startRpcServer(OzoneConfiguration conf,
return rpcServer; return rpcServer;
} }
/**
* Get metadata manager.
* @return metadata manager.
*/
public OMMetadataManager getMetadataManager() {
return metadataManager;
}
public OMMetrics getMetrics() {
return metrics;
}
/** /**
* Main entry point for starting OzoneManager. * Main entry point for starting OzoneManager.
* *
@ -329,6 +263,7 @@ private static void printUsage(PrintStream out) {
/** /**
* Constructs OM instance based on command line arguments. * Constructs OM instance based on command line arguments.
*
* @param argv Command line arguments * @param argv Command line arguments
* @param conf OzoneConfiguration * @param conf OzoneConfiguration
* @return OM instance * @return OM instance
@ -363,9 +298,11 @@ public static OzoneManager createOm(String[] argv,
/** /**
* Initializes the OM instance. * Initializes the OM instance.
*
* @param conf OzoneConfiguration * @param conf OzoneConfiguration
* @return true if OM initialization succeeds, false otherwise * @return true if OM initialization succeeds, false otherwise
* @throws IOException in case ozone metadata directory path is not accessible * @throws IOException in case ozone metadata directory path is not
* accessible
*/ */
private static boolean omInit(OzoneConfiguration conf) throws IOException { private static boolean omInit(OzoneConfiguration conf) throws IOException {
@ -406,15 +343,18 @@ private static boolean omInit(OzoneConfiguration conf) throws IOException {
/** /**
* Parses the command line options for OM initialization. * Parses the command line options for OM initialization.
*
* @param args command line arguments * @param args command line arguments
* @return StartupOption if options are valid, null otherwise * @return StartupOption if options are valid, null otherwise
*/ */
private static StartupOption parseArguments(String[] args) { private static StartupOption parseArguments(String[] args) {
if (args == null || args.length == 0) { if (args == null || args.length == 0) {
return StartupOption.REGULAR; return StartupOption.REGULAR;
} else if (args.length == 1) { } else {
if (args.length == 1) {
return StartupOption.parse(args[0]); return StartupOption.parse(args[0]);
} }
}
return null; return null;
} }
@ -432,6 +372,34 @@ private static String buildRpcServerStartMessage(String description,
String.format("%s not started", description); String.format("%s not started", description);
} }
@VisibleForTesting
public KeyManager getKeyManager() {
return keyManager;
}
@VisibleForTesting
public ScmInfo getScmInfo() throws IOException {
return scmBlockClient.getScmInfo();
}
@VisibleForTesting
public OMStorage getOmStorage() {
return omStorage;
}
/**
* Get metadata manager.
*
* @return metadata manager.
*/
public OMMetadataManager getMetadataManager() {
return metadataManager;
}
public OMMetrics getMetrics() {
return metrics;
}
/** /**
* Start service. * Start service.
*/ */
@ -533,8 +501,8 @@ public void setQuota(String volume, long quota) throws IOException {
* *
* @param volume - volume * @param volume - volume
* @param userAcl - user acls which needs to be checked for access * @param userAcl - user acls which needs to be checked for access
* @return true if the user has required access for the volume, * @return true if the user has required access for the volume, false
* false otherwise * otherwise
* @throws IOException * @throws IOException
*/ */
@Override @Override
@ -702,7 +670,7 @@ public OpenKeySession openKey(OmKeyArgs args) throws IOException {
} }
@Override @Override
public void commitKey(OmKeyArgs args, int clientID) public void commitKey(OmKeyArgs args, long clientID)
throws IOException { throws IOException {
try { try {
metrics.incNumKeyCommits(); metrics.incNumKeyCommits();
@ -714,7 +682,7 @@ public void commitKey(OmKeyArgs args, int clientID)
} }
@Override @Override
public OmKeyLocationInfo allocateBlock(OmKeyArgs args, int clientID) public OmKeyLocationInfo allocateBlock(OmKeyArgs args, long clientID)
throws IOException { throws IOException {
try { try {
metrics.incNumBlockAllocateCalls(); metrics.incNumBlockAllocateCalls();
@ -786,6 +754,7 @@ public List<OmKeyInfo> listKeys(String volumeName, String bucketName,
/** /**
* Sets bucket property from args. * Sets bucket property from args.
*
* @param args - BucketArgs. * @param args - BucketArgs.
* @throws IOException * @throws IOException
*/ */
@ -801,9 +770,9 @@ public void setBucketProperty(OmBucketArgs args)
} }
} }
/** /**
* Deletes an existing empty bucket from volume. * Deletes an existing empty bucket from volume.
*
* @param volume - Name of the volume. * @param volume - Name of the volume.
* @param bucket - Name of the bucket. * @param bucket - Name of the bucket.
* @throws IOException * @throws IOException
@ -908,4 +877,32 @@ public List<ServiceInfo> getServiceList() throws IOException {
// metrics.incNumGetServiceListFails() // metrics.incNumGetServiceListFails()
return services; return services;
} }
/**
* Startup options.
*/
public enum StartupOption {
CREATEOBJECTSTORE("-createObjectStore"),
HELP("-help"),
REGULAR("-regular");
private final String name;
StartupOption(String arg) {
this.name = arg;
}
public static StartupOption parse(String value) {
for (StartupOption option : StartupOption.values()) {
if (option.name.equalsIgnoreCase(value)) {
return option;
}
}
return null;
}
public String getName() {
return name;
}
}
} }

View File

@ -28,7 +28,9 @@
.OzoneManagerProtocolProtos.VolumeInfo; .OzoneManagerProtocolProtos.VolumeInfo;
import org.apache.hadoop.hdds.protocol.proto.HddsProtos; import org.apache.hadoop.hdds.protocol.proto.HddsProtos;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
import org.apache.hadoop.utils.BatchOperation; import org.apache.hadoop.utils.RocksDBStore;
import org.rocksdb.RocksDBException;
import org.rocksdb.WriteBatch;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -67,10 +69,10 @@ public VolumeManagerImpl(OMMetadataManager metadataManager,
// Helpers to add and delete volume from user list // Helpers to add and delete volume from user list
private void addVolumeToOwnerList(String volume, String owner, private void addVolumeToOwnerList(String volume, String owner,
BatchOperation batchOperation) throws IOException { WriteBatch batchOperation) throws RocksDBException, IOException {
// Get the volume list // Get the volume list
byte[] dbUserKey = metadataManager.getUserKey(owner); byte[] dbUserKey = metadataManager.getUserKey(owner);
byte[] volumeList = metadataManager.get(dbUserKey); byte[] volumeList = metadataManager.getUserTable().get(dbUserKey);
List<String> prevVolList = new LinkedList<>(); List<String> prevVolList = new LinkedList<>();
if (volumeList != null) { if (volumeList != null) {
VolumeList vlist = VolumeList.parseFrom(volumeList); VolumeList vlist = VolumeList.parseFrom(volumeList);
@ -87,15 +89,15 @@ private void addVolumeToOwnerList(String volume, String owner,
prevVolList.add(volume); prevVolList.add(volume);
VolumeList newVolList = VolumeList.newBuilder() VolumeList newVolList = VolumeList.newBuilder()
.addAllVolumeNames(prevVolList).build(); .addAllVolumeNames(prevVolList).build();
batchOperation.put(dbUserKey, newVolList.toByteArray()); batchOperation.put(metadataManager.getUserTable().getHandle(),
dbUserKey, newVolList.toByteArray());
} }
private void delVolumeFromOwnerList(String volume, String owner, private void delVolumeFromOwnerList(String volume, String owner,
BatchOperation batchOperation) WriteBatch batch) throws RocksDBException, IOException {
throws IOException {
// Get the volume list // Get the volume list
byte[] dbUserKey = metadataManager.getUserKey(owner); byte[] dbUserKey = metadataManager.getUserKey(owner);
byte[] volumeList = metadataManager.get(dbUserKey); byte[] volumeList = metadataManager.getUserTable().get(dbUserKey);
List<String> prevVolList = new LinkedList<>(); List<String> prevVolList = new LinkedList<>();
if (volumeList != null) { if (volumeList != null) {
VolumeList vlist = VolumeList.parseFrom(volumeList); VolumeList vlist = VolumeList.parseFrom(volumeList);
@ -108,11 +110,12 @@ private void delVolumeFromOwnerList(String volume, String owner,
// Remove the volume from the list // Remove the volume from the list
prevVolList.remove(volume); prevVolList.remove(volume);
if (prevVolList.size() == 0) { if (prevVolList.size() == 0) {
batchOperation.delete(dbUserKey); batch.delete(dbUserKey);
} else { } else {
VolumeList newVolList = VolumeList.newBuilder() VolumeList newVolList = VolumeList.newBuilder()
.addAllVolumeNames(prevVolList).build(); .addAllVolumeNames(prevVolList).build();
batchOperation.put(dbUserKey, newVolList.toByteArray()); batch.put(metadataManager.getUserTable().getHandle(),
dbUserKey, newVolList.toByteArray());
} }
} }
@ -126,7 +129,7 @@ public void createVolume(OmVolumeArgs args) throws IOException {
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
byte[] dbVolumeKey = metadataManager.getVolumeKey(args.getVolume()); byte[] dbVolumeKey = metadataManager.getVolumeKey(args.getVolume());
byte[] volumeInfo = metadataManager.get(dbVolumeKey); byte[] volumeInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
// Check of the volume already exists // Check of the volume already exists
if (volumeInfo != null) { if (volumeInfo != null) {
@ -134,10 +137,11 @@ public void createVolume(OmVolumeArgs args) throws IOException {
throw new OMException(ResultCodes.FAILED_VOLUME_ALREADY_EXISTS); throw new OMException(ResultCodes.FAILED_VOLUME_ALREADY_EXISTS);
} }
BatchOperation batch = new BatchOperation(); try(WriteBatch batch = new WriteBatch()) {
// Write the vol info // Write the vol info
List<HddsProtos.KeyValue> metadataList = new LinkedList<>(); List<HddsProtos.KeyValue> metadataList = new LinkedList<>();
for (Map.Entry<String, String> entry : args.getKeyValueMap().entrySet()) { for (Map.Entry<String, String> entry :
args.getKeyValueMap().entrySet()) {
metadataList.add(HddsProtos.KeyValue.newBuilder() metadataList.add(HddsProtos.KeyValue.newBuilder()
.setKey(entry.getKey()).setValue(entry.getValue()).build()); .setKey(entry.getKey()).setValue(entry.getValue()).build());
} }
@ -152,19 +156,26 @@ public void createVolume(OmVolumeArgs args) throws IOException {
.addAllVolumeAcls(aclList) .addAllVolumeAcls(aclList)
.setCreationTime(Time.now()) .setCreationTime(Time.now())
.build(); .build();
batch.put(dbVolumeKey, newVolumeInfo.toByteArray()); batch.put(metadataManager.getVolumeTable().getHandle(),
dbVolumeKey, newVolumeInfo.toByteArray());
// Add volume to user list // Add volume to user list
addVolumeToOwnerList(args.getVolume(), args.getOwnerName(), batch); addVolumeToOwnerList(args.getVolume(), args.getOwnerName(), batch);
metadataManager.writeBatch(batch); metadataManager.getStore().write(batch);
}
LOG.debug("created volume:{} user:{}", args.getVolume(), LOG.debug("created volume:{} user:{}", args.getVolume(),
args.getOwnerName()); args.getOwnerName());
} catch (IOException ex) { } catch (RocksDBException | IOException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
LOG.error("Volume creation failed for user:{} volume:{}", LOG.error("Volume creation failed for user:{} volume:{}",
args.getOwnerName(), args.getVolume(), ex); args.getOwnerName(), args.getVolume(), ex);
} }
throw ex; if(ex instanceof RocksDBException) {
throw RocksDBStore.toIOException("Volume creation failed.",
(RocksDBException) ex);
} else {
throw (IOException) ex;
}
} finally { } finally {
metadataManager.writeLock().unlock(); metadataManager.writeLock().unlock();
} }
@ -184,7 +195,7 @@ public void setOwner(String volume, String owner) throws IOException {
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
byte[] dbVolumeKey = metadataManager.getVolumeKey(volume); byte[] dbVolumeKey = metadataManager.getVolumeKey(volume);
byte[] volInfo = metadataManager.get(dbVolumeKey); byte[] volInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
if (volInfo == null) { if (volInfo == null) {
LOG.debug("Changing volume ownership failed for user:{} volume:{}", LOG.debug("Changing volume ownership failed for user:{} volume:{}",
owner, volume); owner, volume);
@ -195,7 +206,7 @@ public void setOwner(String volume, String owner) throws IOException {
OmVolumeArgs volumeArgs = OmVolumeArgs.getFromProtobuf(volumeInfo); OmVolumeArgs volumeArgs = OmVolumeArgs.getFromProtobuf(volumeInfo);
Preconditions.checkState(volume.equals(volumeInfo.getVolume())); Preconditions.checkState(volume.equals(volumeInfo.getVolume()));
BatchOperation batch = new BatchOperation(); try(WriteBatch batch = new WriteBatch()) {
delVolumeFromOwnerList(volume, volumeArgs.getOwnerName(), batch); delVolumeFromOwnerList(volume, volumeArgs.getOwnerName(), batch);
addVolumeToOwnerList(volume, owner, batch); addVolumeToOwnerList(volume, owner, batch);
@ -208,15 +219,21 @@ public void setOwner(String volume, String owner) throws IOException {
.build(); .build();
VolumeInfo newVolumeInfo = newVolumeArgs.getProtobuf(); VolumeInfo newVolumeInfo = newVolumeArgs.getProtobuf();
batch.put(dbVolumeKey, newVolumeInfo.toByteArray()); batch.put(metadataManager.getVolumeTable().getHandle(),
dbVolumeKey, newVolumeInfo.toByteArray());
metadataManager.writeBatch(batch); metadataManager.getStore().write(batch);
} catch (IOException ex) { }
} catch (RocksDBException | IOException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
LOG.error("Changing volume ownership failed for user:{} volume:{}", LOG.error("Changing volume ownership failed for user:{} volume:{}",
owner, volume, ex); owner, volume, ex);
} }
throw ex; if(ex instanceof RocksDBException) {
throw RocksDBStore.toIOException("Volume creation failed.",
(RocksDBException) ex);
} else {
throw (IOException) ex;
}
} finally { } finally {
metadataManager.writeLock().unlock(); metadataManager.writeLock().unlock();
} }
@ -234,7 +251,7 @@ public void setQuota(String volume, long quota) throws IOException {
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
byte[] dbVolumeKey = metadataManager.getVolumeKey(volume); byte[] dbVolumeKey = metadataManager.getVolumeKey(volume);
byte[] volInfo = metadataManager.get(dbVolumeKey); byte[] volInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
if (volInfo == null) { if (volInfo == null) {
LOG.debug("volume:{} does not exist", volume); LOG.debug("volume:{} does not exist", volume);
throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND); throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND);
@ -253,7 +270,8 @@ public void setQuota(String volume, long quota) throws IOException {
.setCreationTime(volumeArgs.getCreationTime()).build(); .setCreationTime(volumeArgs.getCreationTime()).build();
VolumeInfo newVolumeInfo = newVolumeArgs.getProtobuf(); VolumeInfo newVolumeInfo = newVolumeArgs.getProtobuf();
metadataManager.put(dbVolumeKey, newVolumeInfo.toByteArray()); metadataManager.getVolumeTable().put(dbVolumeKey,
newVolumeInfo.toByteArray());
} catch (IOException ex) { } catch (IOException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
LOG.error("Changing volume quota failed for volume:{} quota:{}", volume, LOG.error("Changing volume quota failed for volume:{} quota:{}", volume,
@ -276,7 +294,7 @@ public OmVolumeArgs getVolumeInfo(String volume) throws IOException {
metadataManager.readLock().lock(); metadataManager.readLock().lock();
try { try {
byte[] dbVolumeKey = metadataManager.getVolumeKey(volume); byte[] dbVolumeKey = metadataManager.getVolumeKey(volume);
byte[] volInfo = metadataManager.get(dbVolumeKey); byte[] volInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
if (volInfo == null) { if (volInfo == null) {
LOG.debug("volume:{} does not exist", volume); LOG.debug("volume:{} does not exist", volume);
throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND); throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND);
@ -307,9 +325,9 @@ public void deleteVolume(String volume) throws IOException {
Preconditions.checkNotNull(volume); Preconditions.checkNotNull(volume);
metadataManager.writeLock().lock(); metadataManager.writeLock().lock();
try { try {
BatchOperation batch = new BatchOperation();
byte[] dbVolumeKey = metadataManager.getVolumeKey(volume); byte[] dbVolumeKey = metadataManager.getVolumeKey(volume);
byte[] volInfo = metadataManager.get(dbVolumeKey); byte[] volInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
if (volInfo == null) { if (volInfo == null) {
LOG.debug("volume:{} does not exist", volume); LOG.debug("volume:{} does not exist", volume);
throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND); throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND);
@ -324,14 +342,22 @@ public void deleteVolume(String volume) throws IOException {
Preconditions.checkState(volume.equals(volumeInfo.getVolume())); Preconditions.checkState(volume.equals(volumeInfo.getVolume()));
// delete the volume from the owner list // delete the volume from the owner list
// as well as delete the volume entry // as well as delete the volume entry
try(WriteBatch batch = new WriteBatch()) {
delVolumeFromOwnerList(volume, volumeInfo.getOwnerName(), batch); delVolumeFromOwnerList(volume, volumeInfo.getOwnerName(), batch);
batch.delete(dbVolumeKey); batch.delete(metadataManager.getVolumeTable().getHandle(),
metadataManager.writeBatch(batch); dbVolumeKey);
} catch (IOException ex) { metadataManager.getStore().write(batch);
}
} catch (RocksDBException| IOException ex) {
if (!(ex instanceof OMException)) { if (!(ex instanceof OMException)) {
LOG.error("Delete volume failed for volume:{}", volume, ex); LOG.error("Delete volume failed for volume:{}", volume, ex);
} }
throw ex; if(ex instanceof RocksDBException) {
throw RocksDBStore.toIOException("Volume creation failed.",
(RocksDBException) ex);
} else {
throw (IOException) ex;
}
} finally { } finally {
metadataManager.writeLock().unlock(); metadataManager.writeLock().unlock();
} }
@ -352,7 +378,7 @@ public boolean checkVolumeAccess(String volume, OzoneAclInfo userAcl)
metadataManager.readLock().lock(); metadataManager.readLock().lock();
try { try {
byte[] dbVolumeKey = metadataManager.getVolumeKey(volume); byte[] dbVolumeKey = metadataManager.getVolumeKey(volume);
byte[] volInfo = metadataManager.get(dbVolumeKey); byte[] volInfo = metadataManager.getVolumeTable().get(dbVolumeKey);
if (volInfo == null) { if (volInfo == null) {
LOG.debug("volume:{} does not exist", volume); LOG.debug("volume:{} does not exist", volume);
throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND); throw new OMException(ResultCodes.FAILED_VOLUME_NOT_FOUND);

View File

@ -526,8 +526,7 @@ public CommitKeyResponse commitKey(RpcController controller,
.setFactor(factor) .setFactor(factor)
.setDataSize(keyArgs.getDataSize()) .setDataSize(keyArgs.getDataSize())
.build(); .build();
int id = request.getClientID(); impl.commitKey(omKeyArgs, request.getClientID());
impl.commitKey(omKeyArgs, id);
resp.setStatus(Status.OK); resp.setStatus(Status.OK);
} catch (IOException e) { } catch (IOException e) {
resp.setStatus(exceptionToResponseStatus(e)); resp.setStatus(exceptionToResponseStatus(e));
@ -547,8 +546,8 @@ public AllocateBlockResponse allocateBlock(RpcController controller,
.setBucketName(keyArgs.getBucketName()) .setBucketName(keyArgs.getBucketName())
.setKeyName(keyArgs.getKeyName()) .setKeyName(keyArgs.getKeyName())
.build(); .build();
int id = request.getClientID(); OmKeyLocationInfo newLocation = impl.allocateBlock(omKeyArgs,
OmKeyLocationInfo newLocation = impl.allocateBlock(omKeyArgs, id); request.getClientID());
resp.setKeyLocation(newLocation.getProtobuf()); resp.setKeyLocation(newLocation.getProtobuf());
resp.setStatus(Status.OK); resp.setStatus(Status.OK);
} catch (IOException e) { } catch (IOException e) {

View File

@ -17,33 +17,26 @@
package org.apache.hadoop.ozone.om; package org.apache.hadoop.ozone.om;
import org.apache.hadoop.fs.StorageType; import org.apache.hadoop.fs.StorageType;
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
import org.apache.hadoop.hdds.server.ServerUtils;
import org.apache.hadoop.hdfs.DFSUtil; import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.ozone.om.helpers.OmBucketArgs; import org.apache.hadoop.ozone.OzoneAcl;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.apache.hadoop.ozone.OzoneConsts;
import org.apache.hadoop.ozone.om.exceptions.OMException; import org.apache.hadoop.ozone.om.exceptions.OMException;
import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes; import org.apache.hadoop.ozone.om.exceptions.OMException.ResultCodes;
import org.apache.hadoop.ozone.OzoneAcl; import org.apache.hadoop.ozone.om.helpers.OmBucketArgs;
import org.apache.hadoop.ozone.om.helpers.OmBucketInfo;
import org.junit.Assert; import org.junit.Assert;
import org.junit.Rule; import org.junit.Rule;
import org.junit.Test; import org.junit.Test;
import org.junit.rules.ExpectedException; import org.junit.rules.ExpectedException;
import org.junit.rules.TemporaryFolder;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import org.mockito.Mockito;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.runners.MockitoJUnitRunner; import org.mockito.runners.MockitoJUnitRunner;
import org.mockito.stubbing.Answer;
import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.util.Map;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.concurrent.locks.ReadWriteLock; import java.util.List;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import static org.mockito.Mockito.any;
/** /**
* Tests BucketManagerImpl, mocks OMMetadataManager for testing. * Tests BucketManagerImpl, mocks OMMetadataManager for testing.
@ -53,86 +46,35 @@ public class TestBucketManagerImpl {
@Rule @Rule
public ExpectedException thrown = ExpectedException.none(); public ExpectedException thrown = ExpectedException.none();
private OMMetadataManager getMetadataManagerMock(String... volumesToCreate) @Rule
throws IOException { public TemporaryFolder folder = new TemporaryFolder();
OMMetadataManager metadataManager = Mockito.mock(OMMetadataManager.class);
Map<String, byte[]> metadataDB = new HashMap<>();
ReadWriteLock lock = new ReentrantReadWriteLock();
Mockito.when(metadataManager.writeLock()).thenReturn(lock.writeLock()); private OzoneConfiguration createNewTestPath() throws IOException {
Mockito.when(metadataManager.readLock()).thenReturn(lock.readLock()); OzoneConfiguration conf = new OzoneConfiguration();
Mockito.when(metadataManager.getVolumeKey(any(String.class))).thenAnswer( File newFolder = folder.newFolder();
(InvocationOnMock invocation) -> if (!newFolder.exists()) {
DFSUtil.string2Bytes( Assert.assertTrue(newFolder.mkdirs());
OzoneConsts.OM_VOLUME_PREFIX + invocation.getArguments()[0])); }
Mockito.when(metadataManager ServerUtils.setOzoneMetaDirPath(conf, newFolder.toString());
.getBucketKey(any(String.class), any(String.class))).thenAnswer( return conf;
(InvocationOnMock invocation) -> }
DFSUtil.string2Bytes(
OzoneConsts.OM_VOLUME_PREFIX
+ invocation.getArguments()[0]
+ OzoneConsts.OM_BUCKET_PREFIX
+ invocation.getArguments()[1]));
Mockito.doAnswer( private OmMetadataManagerImpl createSampleVol() throws IOException {
new Answer<Boolean>() { OzoneConfiguration conf = createNewTestPath();
@Override OmMetadataManagerImpl metaMgr = new OmMetadataManagerImpl(conf);
public Boolean answer(InvocationOnMock invocation) byte[] volumeKey = metaMgr.getVolumeKey("sampleVol");
throws Throwable { // This is a simple hack for testing, we just test if the volume via a
String keyRootName = OzoneConsts.OM_KEY_PREFIX // null check, do not parse the value part. So just write some dummy value.
+ invocation.getArguments()[0] metaMgr.getVolumeTable().put(volumeKey, volumeKey);
+ OzoneConsts.OM_KEY_PREFIX return metaMgr;
+ invocation.getArguments()[1]
+ OzoneConsts.OM_KEY_PREFIX;
Iterator<String> keyIterator = metadataDB.keySet().iterator();
while(keyIterator.hasNext()) {
if(keyIterator.next().startsWith(keyRootName)) {
return false;
}
}
return true;
}
}).when(metadataManager).isBucketEmpty(any(String.class),
any(String.class));
Mockito.doAnswer(
new Answer<Void>() {
@Override
public Void answer(InvocationOnMock invocation) throws Throwable {
metadataDB.put(DFSUtil.bytes2String(
(byte[])invocation.getArguments()[0]),
(byte[])invocation.getArguments()[1]);
return null;
}
}).when(metadataManager).put(any(byte[].class), any(byte[].class));
Mockito.when(metadataManager.get(any(byte[].class))).thenAnswer(
(InvocationOnMock invocation) ->
metadataDB.get(DFSUtil.bytes2String(
(byte[])invocation.getArguments()[0]))
);
Mockito.doAnswer(
new Answer<Void>() {
@Override
public Void answer(InvocationOnMock invocation) throws Throwable {
metadataDB.remove(DFSUtil.bytes2String(
(byte[])invocation.getArguments()[0]));
return null;
}
}).when(metadataManager).delete(any(byte[].class));
for(String volumeName : volumesToCreate) {
byte[] dummyVolumeInfo = DFSUtil.string2Bytes(volumeName);
metadataDB.put(OzoneConsts.OM_VOLUME_PREFIX + volumeName,
dummyVolumeInfo);
}
return metadataManager;
} }
@Test @Test
public void testCreateBucketWithoutVolume() throws IOException { public void testCreateBucketWithoutVolume() throws Exception {
thrown.expectMessage("Volume doesn't exist"); thrown.expectMessage("Volume doesn't exist");
OMMetadataManager metaMgr = getMetadataManagerMock(); OzoneConfiguration conf = createNewTestPath();
OmMetadataManagerImpl metaMgr =
new OmMetadataManagerImpl(conf);
try { try {
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
@ -144,25 +86,31 @@ public void testCreateBucketWithoutVolume() throws IOException {
Assert.assertEquals(ResultCodes.FAILED_VOLUME_NOT_FOUND, Assert.assertEquals(ResultCodes.FAILED_VOLUME_NOT_FOUND,
omEx.getResult()); omEx.getResult());
throw omEx; throw omEx;
} finally {
metaMgr.getStore().close();
} }
} }
@Test @Test
public void testCreateBucket() throws IOException { public void testCreateBucket() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
.setVolumeName("sampleVol") .setVolumeName("sampleVol")
.setBucketName("bucketOne") .setBucketName("bucketOne")
.build(); .build();
bucketManager.createBucket(bucketInfo); bucketManager.createBucket(bucketInfo);
Assert.assertNotNull(bucketManager.getBucketInfo("sampleVol", "bucketOne")); Assert.assertNotNull(bucketManager.getBucketInfo("sampleVol",
"bucketOne"));
metaMgr.getStore().close();
} }
@Test @Test
public void testCreateAlreadyExistingBucket() throws IOException { public void testCreateAlreadyExistingBucket() throws Exception {
thrown.expectMessage("Bucket already exist"); thrown.expectMessage("Bucket already exist");
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
try { try {
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
@ -175,26 +123,33 @@ public void testCreateAlreadyExistingBucket() throws IOException {
Assert.assertEquals(ResultCodes.FAILED_BUCKET_ALREADY_EXISTS, Assert.assertEquals(ResultCodes.FAILED_BUCKET_ALREADY_EXISTS,
omEx.getResult()); omEx.getResult());
throw omEx; throw omEx;
} finally {
metaMgr.getStore().close();
} }
} }
@Test @Test
public void testGetBucketInfoForInvalidBucket() throws IOException { public void testGetBucketInfoForInvalidBucket() throws Exception {
thrown.expectMessage("Bucket not found"); thrown.expectMessage("Bucket not found");
OmMetadataManagerImpl metaMgr = createSampleVol();
try { try {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol");
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
bucketManager.getBucketInfo("sampleVol", "bucketOne"); bucketManager.getBucketInfo("sampleVol", "bucketOne");
} catch (OMException omEx) { } catch (OMException omEx) {
Assert.assertEquals(ResultCodes.FAILED_BUCKET_NOT_FOUND, Assert.assertEquals(ResultCodes.FAILED_BUCKET_NOT_FOUND,
omEx.getResult()); omEx.getResult());
throw omEx; throw omEx;
} finally {
metaMgr.getStore().close();
} }
} }
@Test @Test
public void testGetBucketInfo() throws IOException { public void testGetBucketInfo() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
.setVolumeName("sampleVol") .setVolumeName("sampleVol")
@ -210,11 +165,13 @@ public void testGetBucketInfo() throws IOException {
Assert.assertEquals(StorageType.DISK, Assert.assertEquals(StorageType.DISK,
result.getStorageType()); result.getStorageType());
Assert.assertEquals(false, result.getIsVersionEnabled()); Assert.assertEquals(false, result.getIsVersionEnabled());
metaMgr.getStore().close();
} }
@Test @Test
public void testSetBucketPropertyAddACL() throws IOException { public void testSetBucketPropertyAddACL() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
List<OzoneAcl> acls = new LinkedList<>(); List<OzoneAcl> acls = new LinkedList<>();
OzoneAcl ozoneAcl = new OzoneAcl(OzoneAcl.OzoneACLType.USER, OzoneAcl ozoneAcl = new OzoneAcl(OzoneAcl.OzoneACLType.USER,
"root", OzoneAcl.OzoneACLRights.READ); "root", OzoneAcl.OzoneACLRights.READ);
@ -247,11 +204,13 @@ public void testSetBucketPropertyAddACL() throws IOException {
"sampleVol", "bucketOne"); "sampleVol", "bucketOne");
Assert.assertEquals(2, updatedResult.getAcls().size()); Assert.assertEquals(2, updatedResult.getAcls().size());
Assert.assertTrue(updatedResult.getAcls().contains(newAcl)); Assert.assertTrue(updatedResult.getAcls().contains(newAcl));
metaMgr.getStore().close();
} }
@Test @Test
public void testSetBucketPropertyRemoveACL() throws IOException { public void testSetBucketPropertyRemoveACL() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
List<OzoneAcl> acls = new LinkedList<>(); List<OzoneAcl> acls = new LinkedList<>();
OzoneAcl aclOne = new OzoneAcl(OzoneAcl.OzoneACLType.USER, OzoneAcl aclOne = new OzoneAcl(OzoneAcl.OzoneACLType.USER,
"root", OzoneAcl.OzoneACLRights.READ); "root", OzoneAcl.OzoneACLRights.READ);
@ -283,11 +242,13 @@ public void testSetBucketPropertyRemoveACL() throws IOException {
"sampleVol", "bucketOne"); "sampleVol", "bucketOne");
Assert.assertEquals(1, updatedResult.getAcls().size()); Assert.assertEquals(1, updatedResult.getAcls().size());
Assert.assertFalse(updatedResult.getAcls().contains(aclTwo)); Assert.assertFalse(updatedResult.getAcls().contains(aclTwo));
metaMgr.getStore().close();
} }
@Test @Test
public void testSetBucketPropertyChangeStorageType() throws IOException { public void testSetBucketPropertyChangeStorageType() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
.setVolumeName("sampleVol") .setVolumeName("sampleVol")
@ -309,11 +270,13 @@ public void testSetBucketPropertyChangeStorageType() throws IOException {
"sampleVol", "bucketOne"); "sampleVol", "bucketOne");
Assert.assertEquals(StorageType.SSD, Assert.assertEquals(StorageType.SSD,
updatedResult.getStorageType()); updatedResult.getStorageType());
metaMgr.getStore().close();
} }
@Test @Test
public void testSetBucketPropertyChangeVersioning() throws IOException { public void testSetBucketPropertyChangeVersioning() throws Exception {
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
.setVolumeName("sampleVol") .setVolumeName("sampleVol")
@ -333,12 +296,13 @@ public void testSetBucketPropertyChangeVersioning() throws IOException {
OmBucketInfo updatedResult = bucketManager.getBucketInfo( OmBucketInfo updatedResult = bucketManager.getBucketInfo(
"sampleVol", "bucketOne"); "sampleVol", "bucketOne");
Assert.assertTrue(updatedResult.getIsVersionEnabled()); Assert.assertTrue(updatedResult.getIsVersionEnabled());
metaMgr.getStore().close();
} }
@Test @Test
public void testDeleteBucket() throws IOException { public void testDeleteBucket() throws Exception {
thrown.expectMessage("Bucket not found"); thrown.expectMessage("Bucket not found");
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
for (int i = 0; i < 5; i++) { for (int i = 0; i < 5; i++) {
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
@ -366,12 +330,13 @@ public void testDeleteBucket() throws IOException {
omEx.getResult()); omEx.getResult());
throw omEx; throw omEx;
} }
metaMgr.getStore().close();
} }
@Test @Test
public void testDeleteNonEmptyBucket() throws IOException { public void testDeleteNonEmptyBucket() throws Exception {
thrown.expectMessage("Bucket is not empty"); thrown.expectMessage("Bucket is not empty");
OMMetadataManager metaMgr = getMetadataManagerMock("sampleVol"); OmMetadataManagerImpl metaMgr = createSampleVol();
BucketManager bucketManager = new BucketManagerImpl(metaMgr); BucketManager bucketManager = new BucketManagerImpl(metaMgr);
OmBucketInfo bucketInfo = OmBucketInfo.newBuilder() OmBucketInfo bucketInfo = OmBucketInfo.newBuilder()
.setVolumeName("sampleVol") .setVolumeName("sampleVol")
@ -379,9 +344,11 @@ public void testDeleteNonEmptyBucket() throws IOException {
.build(); .build();
bucketManager.createBucket(bucketInfo); bucketManager.createBucket(bucketInfo);
//Create keys in bucket //Create keys in bucket
metaMgr.put(DFSUtil.string2Bytes("/sampleVol/bucketOne/key_one"), metaMgr.getKeyTable().put(DFSUtil.string2Bytes("/sampleVol/bucketOne" +
"/key_one"),
DFSUtil.string2Bytes("value_one")); DFSUtil.string2Bytes("value_one"));
metaMgr.put(DFSUtil.string2Bytes("/sampleVol/bucketOne/key_two"), metaMgr.getKeyTable().put(DFSUtil.string2Bytes("/sampleVol/bucketOne" +
"/key_two"),
DFSUtil.string2Bytes("value_two")); DFSUtil.string2Bytes("value_two"));
try { try {
bucketManager.deleteBucket("sampleVol", "bucketOne"); bucketManager.deleteBucket("sampleVol", "bucketOne");
@ -390,5 +357,6 @@ public void testDeleteNonEmptyBucket() throws IOException {
omEx.getResult()); omEx.getResult());
throw omEx; throw omEx;
} }
metaMgr.getStore().close();
} }
} }

View File

@ -57,9 +57,8 @@
import static org.apache.hadoop.ozone.OzoneConsts.CONTAINER_DB_SUFFIX; import static org.apache.hadoop.ozone.OzoneConsts.CONTAINER_DB_SUFFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OM_DB_NAME; import static org.apache.hadoop.ozone.OzoneConsts.OM_DB_NAME;
import static org.apache.hadoop.ozone.OzoneConsts.OM_KEY_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OM_USER_PREFIX; import static org.apache.hadoop.ozone.OzoneConsts.OM_USER_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OM_BUCKET_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OM_VOLUME_PREFIX;
import static org.apache.hadoop.ozone.OzoneConsts.OPEN_CONTAINERS_DB; import static org.apache.hadoop.ozone.OzoneConsts.OPEN_CONTAINERS_DB;
/** /**
@ -412,12 +411,15 @@ private void insertOMDB(Connection conn, KeyType type, String keyName,
} }
} }
// TODO: This has to be fixed.
// we don't have prefix anymore. now each key is written into different
// table. The logic has to be changed.
private KeyType getKeyType(String key) { private KeyType getKeyType(String key) {
if (key.startsWith(OM_USER_PREFIX)) { if (key.startsWith(OM_USER_PREFIX)) {
return KeyType.USER; return KeyType.USER;
} else if (key.startsWith(OM_VOLUME_PREFIX)) { } else if (key.startsWith(OM_KEY_PREFIX)) {
return key.replaceFirst(OM_VOLUME_PREFIX, "") return key.replaceFirst(OM_KEY_PREFIX, "")
.contains(OM_BUCKET_PREFIX) ? KeyType.BUCKET : KeyType.VOLUME; .contains(OM_KEY_PREFIX) ? KeyType.BUCKET : KeyType.VOLUME;
}else { }else {
return KeyType.KEY; return KeyType.KEY;
} }