blob: a9f5c738187de71c832b96911fea8b675a7eb690 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hdfs.server.aliasmap;
import org.apache.commons.io.FileUtils;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hdfs.DFSConfigKeys;
import org.apache.hadoop.hdfs.protocol.Block;
import org.apache.hadoop.hdfs.protocol.ProvidedStorageLocation;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.util.Arrays;
import java.util.Optional;
/**
* ITestInMemoryAliasMap is an integration test that writes and reads to
* an AliasMap. This is an integration test because it can't be run in parallel
* like normal unit tests since there is conflict over the port being in use.
*/
public class ITestInMemoryAliasMap {
private InMemoryAliasMap aliasMap;
private File tempDirectory;
private static String bpid = "bpid-0";
@Before
public void setUp() throws Exception {
Configuration conf = new Configuration();
File temp = Files.createTempDirectory("seagull").toFile();
tempDirectory = new File(temp, bpid);
tempDirectory.mkdirs();
conf.set(DFSConfigKeys.DFS_PROVIDED_ALIASMAP_INMEMORY_LEVELDB_DIR,
temp.getAbsolutePath());
aliasMap = InMemoryAliasMap.init(conf, bpid);
}
@After
public void tearDown() throws Exception {
aliasMap.close();
FileUtils.deleteDirectory(tempDirectory);
}
@Test
public void testReadNotFoundReturnsNothing() throws IOException {
Block block = new Block(42, 43, 44);
Optional<ProvidedStorageLocation> actualProvidedStorageLocationOpt
= aliasMap.read(block);
assertFalse(actualProvidedStorageLocationOpt.isPresent());
}
@Test
public void testReadWriteRemove() throws Exception {
Block block = new Block(42, 43, 44);
Path path = new Path("eagle", "mouse");
long offset = 47;
long length = 48;
int nonceSize = 4;
byte[] nonce = new byte[nonceSize];
Arrays.fill(nonce, 0, (nonceSize - 1), Byte.parseByte("0011", 2));
ProvidedStorageLocation expectedProvidedStorageLocation =
new ProvidedStorageLocation(path, offset, length, nonce);
aliasMap.write(block, expectedProvidedStorageLocation);
Optional<ProvidedStorageLocation> actualProvidedStorageLocationOpt
= aliasMap.read(block);
assertTrue(actualProvidedStorageLocationOpt.isPresent());
assertEquals(expectedProvidedStorageLocation,
actualProvidedStorageLocationOpt.get());
aliasMap.remove(block);
actualProvidedStorageLocationOpt = aliasMap.read(block);
assertFalse(actualProvidedStorageLocationOpt.isPresent());
}
@Test
public void testIteration() throws IOException {
Block block1 = new Block(42, 43, 44);
Block block2 = new Block(43, 44, 45);
Block block3 = new Block(44, 45, 46);
Path path = new Path("eagle", "mouse");
int nonceSize = 4;
byte[] nonce = new byte[nonceSize];
Arrays.fill(nonce, 0, (nonceSize - 1), Byte.parseByte("0011", 2));
ProvidedStorageLocation expectedProvidedStorageLocation1 =
new ProvidedStorageLocation(path, 47, 48, nonce);
ProvidedStorageLocation expectedProvidedStorageLocation2 =
new ProvidedStorageLocation(path, 48, 49, nonce);
ProvidedStorageLocation expectedProvidedStorageLocation3 =
new ProvidedStorageLocation(path, 49, 50, nonce);
aliasMap.write(block1, expectedProvidedStorageLocation1);
aliasMap.write(block2, expectedProvidedStorageLocation2);
aliasMap.write(block3, expectedProvidedStorageLocation3);
InMemoryAliasMap.IterationResult list = aliasMap.list(Optional.empty());
// we should have 3 results
assertEquals(3, list.getFileRegions().size());
// no more results expected
assertFalse(list.getNextBlock().isPresent());
}
@Test
public void testSnapshot() throws Exception {
Block block1 = new Block(100);
Block block2 = new Block(200);
Path path = new Path("users", "alice");
ProvidedStorageLocation remoteLocation =
new ProvidedStorageLocation(path, 0, 1000, new byte[0]);
// write the first block
aliasMap.write(block1, remoteLocation);
// create snapshot
File snapshotFile = InMemoryAliasMap.createSnapshot(aliasMap);
// write the 2nd block after the snapshot
aliasMap.write(block2, remoteLocation);
// creata a new aliasmap object from the snapshot
InMemoryAliasMap snapshotAliasMap = null;
Configuration newConf = new Configuration();
newConf.set(DFSConfigKeys.DFS_PROVIDED_ALIASMAP_INMEMORY_LEVELDB_DIR,
snapshotFile.getAbsolutePath());
try {
snapshotAliasMap = InMemoryAliasMap.init(newConf, bpid);
// now the snapshot should have the first block but not the second one.
assertTrue(snapshotAliasMap.read(block1).isPresent());
assertFalse(snapshotAliasMap.read(block2).isPresent());
} finally {
if (snapshotAliasMap != null) {
snapshotAliasMap.close();
}
}
}
}