HDFS-16748. RBF: DFSClient should uniquely identify writing files by namespace id and iNodeId via RBF (#4813). Contributed by ZanderXu.
Reviewed-by: He Xiaoqiao <hexiaoqiao@apache.org> Signed-off-by: Ayush Saxena <ayushsaxena@apache.org>
This commit is contained in:
parent
ac42519ade
commit
be4c638e4c
|
@ -275,7 +275,7 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
* that are currently being written by this client.
|
* that are currently being written by this client.
|
||||||
* Note that a file can only be written by a single client.
|
* Note that a file can only be written by a single client.
|
||||||
*/
|
*/
|
||||||
private final Map<Long, DFSOutputStream> filesBeingWritten = new HashMap<>();
|
private final Map<String, DFSOutputStream> filesBeingWritten = new HashMap<>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Same as this(NameNode.getNNAddress(conf), conf);
|
* Same as this(NameNode.getNNAddress(conf), conf);
|
||||||
|
@ -502,9 +502,9 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get a lease and start automatic renewal */
|
/** Get a lease and start automatic renewal */
|
||||||
private void beginFileLease(final long inodeId, final DFSOutputStream out) {
|
private void beginFileLease(final String key, final DFSOutputStream out) {
|
||||||
synchronized (filesBeingWritten) {
|
synchronized (filesBeingWritten) {
|
||||||
putFileBeingWritten(inodeId, out);
|
putFileBeingWritten(key, out);
|
||||||
LeaseRenewer renewer = getLeaseRenewer();
|
LeaseRenewer renewer = getLeaseRenewer();
|
||||||
boolean result = renewer.put(this);
|
boolean result = renewer.put(this);
|
||||||
if (!result) {
|
if (!result) {
|
||||||
|
@ -518,9 +518,9 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Stop renewal of lease for the file. */
|
/** Stop renewal of lease for the file. */
|
||||||
void endFileLease(final long inodeId) {
|
void endFileLease(final String renewLeaseKey) {
|
||||||
synchronized (filesBeingWritten) {
|
synchronized (filesBeingWritten) {
|
||||||
removeFileBeingWritten(inodeId);
|
removeFileBeingWritten(renewLeaseKey);
|
||||||
// remove client from renewer if no files are open
|
// remove client from renewer if no files are open
|
||||||
if (filesBeingWritten.isEmpty()) {
|
if (filesBeingWritten.isEmpty()) {
|
||||||
getLeaseRenewer().closeClient(this);
|
getLeaseRenewer().closeClient(this);
|
||||||
|
@ -532,10 +532,10 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
* enforced to consistently update its local dfsclients array and
|
* enforced to consistently update its local dfsclients array and
|
||||||
* client's filesBeingWritten map.
|
* client's filesBeingWritten map.
|
||||||
*/
|
*/
|
||||||
public void putFileBeingWritten(final long inodeId,
|
public void putFileBeingWritten(final String key,
|
||||||
final DFSOutputStream out) {
|
final DFSOutputStream out) {
|
||||||
synchronized(filesBeingWritten) {
|
synchronized(filesBeingWritten) {
|
||||||
filesBeingWritten.put(inodeId, out);
|
filesBeingWritten.put(key, out);
|
||||||
// update the last lease renewal time only when there was no
|
// update the last lease renewal time only when there was no
|
||||||
// writes. once there is one write stream open, the lease renewer
|
// writes. once there is one write stream open, the lease renewer
|
||||||
// thread keeps it updated well with in anyone's expiration time.
|
// thread keeps it updated well with in anyone's expiration time.
|
||||||
|
@ -546,9 +546,9 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Remove a file. Only called from LeaseRenewer. */
|
/** Remove a file. Only called from LeaseRenewer. */
|
||||||
public void removeFileBeingWritten(final long inodeId) {
|
public void removeFileBeingWritten(final String key) {
|
||||||
synchronized(filesBeingWritten) {
|
synchronized(filesBeingWritten) {
|
||||||
filesBeingWritten.remove(inodeId);
|
filesBeingWritten.remove(key);
|
||||||
if (filesBeingWritten.isEmpty()) {
|
if (filesBeingWritten.isEmpty()) {
|
||||||
lastLeaseRenewal = 0;
|
lastLeaseRenewal = 0;
|
||||||
}
|
}
|
||||||
|
@ -580,6 +580,13 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public int getNumOfFilesBeingWritten() {
|
||||||
|
synchronized (filesBeingWritten) {
|
||||||
|
return filesBeingWritten.size();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get all namespaces of DFSOutputStreams.
|
* Get all namespaces of DFSOutputStreams.
|
||||||
*/
|
*/
|
||||||
|
@ -640,14 +647,14 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
/** Close/abort all files being written. */
|
/** Close/abort all files being written. */
|
||||||
public void closeAllFilesBeingWritten(final boolean abort) {
|
public void closeAllFilesBeingWritten(final boolean abort) {
|
||||||
for(;;) {
|
for(;;) {
|
||||||
final long inodeId;
|
final String key;
|
||||||
final DFSOutputStream out;
|
final DFSOutputStream out;
|
||||||
synchronized(filesBeingWritten) {
|
synchronized(filesBeingWritten) {
|
||||||
if (filesBeingWritten.isEmpty()) {
|
if (filesBeingWritten.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
inodeId = filesBeingWritten.keySet().iterator().next();
|
key = filesBeingWritten.keySet().iterator().next();
|
||||||
out = filesBeingWritten.remove(inodeId);
|
out = filesBeingWritten.remove(key);
|
||||||
}
|
}
|
||||||
if (out != null) {
|
if (out != null) {
|
||||||
try {
|
try {
|
||||||
|
@ -658,7 +665,7 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
} catch(IOException ie) {
|
} catch(IOException ie) {
|
||||||
LOG.error("Failed to " + (abort ? "abort" : "close") + " file: "
|
LOG.error("Failed to " + (abort ? "abort" : "close") + " file: "
|
||||||
+ out.getSrc() + " with inode: " + inodeId, ie);
|
+ out.getSrc() + " with renewLeaseKey: " + key, ie);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1297,7 +1304,7 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
src, masked, flag, createParent, replication, blockSize, progress,
|
src, masked, flag, createParent, replication, blockSize, progress,
|
||||||
dfsClientConf.createChecksum(checksumOpt),
|
dfsClientConf.createChecksum(checksumOpt),
|
||||||
getFavoredNodesStr(favoredNodes), ecPolicyName, storagePolicy);
|
getFavoredNodesStr(favoredNodes), ecPolicyName, storagePolicy);
|
||||||
beginFileLease(result.getFileId(), result);
|
beginFileLease(result.getUniqKey(), result);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1352,7 +1359,7 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
flag, createParent, replication, blockSize, progress, checksum,
|
flag, createParent, replication, blockSize, progress, checksum,
|
||||||
null, null, null);
|
null, null, null);
|
||||||
}
|
}
|
||||||
beginFileLease(result.getFileId(), result);
|
beginFileLease(result.getUniqKey(), result);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1497,7 +1504,7 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
checkOpen();
|
checkOpen();
|
||||||
final DFSOutputStream result = callAppend(src, flag, progress,
|
final DFSOutputStream result = callAppend(src, flag, progress,
|
||||||
favoredNodes);
|
favoredNodes);
|
||||||
beginFileLease(result.getFileId(), result);
|
beginFileLease(result.getUniqKey(), result);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2418,8 +2425,8 @@ public class DFSClient implements java.io.Closeable, RemotePeerFactory,
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
ExtendedBlock getPreviousBlock(long fileId) {
|
ExtendedBlock getPreviousBlock(String key) {
|
||||||
return filesBeingWritten.get(fileId).getBlock();
|
return filesBeingWritten.get(key).getBlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -114,6 +114,7 @@ public class DFSOutputStream extends FSOutputSummer
|
||||||
protected final String src;
|
protected final String src;
|
||||||
protected final long fileId;
|
protected final long fileId;
|
||||||
private final String namespace;
|
private final String namespace;
|
||||||
|
private final String uniqKey;
|
||||||
protected final long blockSize;
|
protected final long blockSize;
|
||||||
protected final int bytesPerChecksum;
|
protected final int bytesPerChecksum;
|
||||||
|
|
||||||
|
@ -197,6 +198,14 @@ public class DFSOutputStream extends FSOutputSummer
|
||||||
this.src = src;
|
this.src = src;
|
||||||
this.fileId = stat.getFileId();
|
this.fileId = stat.getFileId();
|
||||||
this.namespace = stat.getNamespace();
|
this.namespace = stat.getNamespace();
|
||||||
|
if (this.namespace == null) {
|
||||||
|
String defaultKey = dfsClient.getConfiguration().get(
|
||||||
|
HdfsClientConfigKeys.DFS_OUTPUT_STREAM_UNIQ_DEFAULT_KEY,
|
||||||
|
HdfsClientConfigKeys.DFS_OUTPUT_STREAM_UNIQ_DEFAULT_KEY_DEFAULT);
|
||||||
|
this.uniqKey = defaultKey + "_" + this.fileId;
|
||||||
|
} else {
|
||||||
|
this.uniqKey = this.namespace + "_" + this.fileId;
|
||||||
|
}
|
||||||
this.blockSize = stat.getBlockSize();
|
this.blockSize = stat.getBlockSize();
|
||||||
this.blockReplication = stat.getReplication();
|
this.blockReplication = stat.getReplication();
|
||||||
this.fileEncryptionInfo = stat.getFileEncryptionInfo();
|
this.fileEncryptionInfo = stat.getFileEncryptionInfo();
|
||||||
|
@ -820,7 +829,7 @@ public class DFSOutputStream extends FSOutputSummer
|
||||||
|
|
||||||
void setClosed() {
|
void setClosed() {
|
||||||
closed = true;
|
closed = true;
|
||||||
dfsClient.endFileLease(fileId);
|
dfsClient.endFileLease(getUniqKey());
|
||||||
getStreamer().release();
|
getStreamer().release();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -923,7 +932,7 @@ public class DFSOutputStream extends FSOutputSummer
|
||||||
protected void recoverLease(boolean recoverLeaseOnCloseException) {
|
protected void recoverLease(boolean recoverLeaseOnCloseException) {
|
||||||
if (recoverLeaseOnCloseException) {
|
if (recoverLeaseOnCloseException) {
|
||||||
try {
|
try {
|
||||||
dfsClient.endFileLease(fileId);
|
dfsClient.endFileLease(getUniqKey());
|
||||||
dfsClient.recoverLease(src);
|
dfsClient.recoverLease(src);
|
||||||
leaseRecovered = true;
|
leaseRecovered = true;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
@ -1091,6 +1100,11 @@ public class DFSOutputStream extends FSOutputSummer
|
||||||
return namespace;
|
return namespace;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public String getUniqKey() {
|
||||||
|
return this.uniqKey;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the source of stream.
|
* Return the source of stream.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -1055,7 +1055,7 @@ public class DFSStripedOutputStream extends DFSOutputStream
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dfsClient.endFileLease(fileId);
|
dfsClient.endFileLease(getUniqKey());
|
||||||
final IOException ioe = b.build();
|
final IOException ioe = b.build();
|
||||||
if (ioe != null) {
|
if (ioe != null) {
|
||||||
throw ioe;
|
throw ioe;
|
||||||
|
|
|
@ -281,6 +281,10 @@ public interface HdfsClientConfigKeys {
|
||||||
"dfs.client.fsck.read.timeout";
|
"dfs.client.fsck.read.timeout";
|
||||||
int DFS_CLIENT_FSCK_READ_TIMEOUT_DEFAULT = 60 * 1000;
|
int DFS_CLIENT_FSCK_READ_TIMEOUT_DEFAULT = 60 * 1000;
|
||||||
|
|
||||||
|
String DFS_OUTPUT_STREAM_UNIQ_DEFAULT_KEY =
|
||||||
|
"dfs.client.output.stream.uniq.default.key";
|
||||||
|
String DFS_OUTPUT_STREAM_UNIQ_DEFAULT_KEY_DEFAULT = "DEFAULT";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* These are deprecated config keys to client code.
|
* These are deprecated config keys to client code.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -0,0 +1,106 @@
|
||||||
|
/**
|
||||||
|
* 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.federation.router;
|
||||||
|
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.fs.FSDataOutputStream;
|
||||||
|
import org.apache.hadoop.fs.Path;
|
||||||
|
import org.apache.hadoop.hdfs.DistributedFileSystem;
|
||||||
|
import org.apache.hadoop.hdfs.protocol.HdfsFileStatus;
|
||||||
|
import org.apache.hadoop.hdfs.server.federation.MiniRouterDFSCluster;
|
||||||
|
import org.apache.hadoop.hdfs.server.federation.MockResolver;
|
||||||
|
import org.apache.hadoop.hdfs.server.federation.RouterConfigBuilder;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Testing DFSClient renewLease with same INodeId.
|
||||||
|
*/
|
||||||
|
public class TestRenewLeaseWithSameINodeId {
|
||||||
|
|
||||||
|
/** Federated HDFS cluster. */
|
||||||
|
private static MiniRouterDFSCluster cluster;
|
||||||
|
|
||||||
|
/** The first Router Context for this federated cluster. */
|
||||||
|
private static MiniRouterDFSCluster.RouterContext routerContext;
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void globalSetUp() throws Exception {
|
||||||
|
cluster = new MiniRouterDFSCluster(false, 2);
|
||||||
|
cluster.setNumDatanodesPerNameservice(3);
|
||||||
|
cluster.startCluster();
|
||||||
|
|
||||||
|
Configuration routerConf = new RouterConfigBuilder()
|
||||||
|
.metrics()
|
||||||
|
.rpc()
|
||||||
|
.quota()
|
||||||
|
.build();
|
||||||
|
cluster.addRouterOverrides(routerConf);
|
||||||
|
cluster.startRouters();
|
||||||
|
|
||||||
|
// Register and verify all NNs with all routers
|
||||||
|
cluster.registerNamenodes();
|
||||||
|
cluster.waitNamenodeRegistration();
|
||||||
|
|
||||||
|
routerContext = cluster.getRouters().get(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void tearDown() throws Exception {
|
||||||
|
cluster.shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Testing case:
|
||||||
|
* 1. One Router DFSClient writing multi files from different namespace with same iNodeId.
|
||||||
|
* 2. DFSClient Lease Renewer should work well.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testRenewLeaseWithSameINodeId() throws IOException {
|
||||||
|
// Add mount point "/ns0" and "/ns1"
|
||||||
|
Router router = cluster.getRouters().get(0).getRouter();
|
||||||
|
MockResolver resolver = (MockResolver) router.getSubclusterResolver();
|
||||||
|
resolver.addLocation("/ns0", cluster.getNameservices().get(0), "/ns0");
|
||||||
|
resolver.addLocation("/ns1", cluster.getNameservices().get(1), "/ns1");
|
||||||
|
|
||||||
|
DistributedFileSystem fs = (DistributedFileSystem) routerContext.getFileSystem();
|
||||||
|
|
||||||
|
Path path1 = new Path("/ns0/file");
|
||||||
|
Path path2 = new Path("/ns1/file");
|
||||||
|
|
||||||
|
try (FSDataOutputStream ignored1 = fs.create(path1);
|
||||||
|
FSDataOutputStream ignored2 = fs.create(path2)) {
|
||||||
|
HdfsFileStatus fileStatus1 = fs.getClient().getFileInfo(path1.toUri().getPath());
|
||||||
|
HdfsFileStatus fileStatus2 = fs.getClient().getFileInfo(path2.toUri().getPath());
|
||||||
|
|
||||||
|
// The fileId of the files from different new namespaces should be same.
|
||||||
|
assertEquals(fileStatus2.getFileId(), fileStatus1.getFileId());
|
||||||
|
|
||||||
|
// The number of fileBeingWritten of this DFSClient should be two.
|
||||||
|
assertEquals(2, fs.getClient().getNumOfFilesBeingWritten());
|
||||||
|
}
|
||||||
|
|
||||||
|
// The number of fileBeingWritten of this DFSClient should be zero.
|
||||||
|
assertEquals(0, fs.getClient().getNumOfFilesBeingWritten());
|
||||||
|
}
|
||||||
|
}
|
|
@ -6454,4 +6454,12 @@
|
||||||
frequently than this time, the client will give up waiting.
|
frequently than this time, the client will give up waiting.
|
||||||
</description>
|
</description>
|
||||||
</property>
|
</property>
|
||||||
|
<property>
|
||||||
|
<name>dfs.client.output.stream.uniq.default.key</name>
|
||||||
|
<value>DEFAULT</value>
|
||||||
|
<description>
|
||||||
|
The default prefix key to construct the uniqKey for one DFSOutputStream.
|
||||||
|
If the namespace is DEFAULT, it's best to change this conf to other value.
|
||||||
|
</description>
|
||||||
|
</property>
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|
|
@ -54,8 +54,8 @@ public class DFSClientAdapter {
|
||||||
return dfs.dfs;
|
return dfs.dfs;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static ExtendedBlock getPreviousBlock(DFSClient client, long fileId) {
|
public static ExtendedBlock getPreviousBlock(DFSClient client, String renewLeaseKey) {
|
||||||
return client.getPreviousBlock(fileId);
|
return client.getPreviousBlock(renewLeaseKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static long getFileId(DFSOutputStream out) {
|
public static long getFileId(DFSOutputStream out) {
|
||||||
|
|
|
@ -68,8 +68,9 @@ import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
import static org.mockito.ArgumentMatchers.anyBoolean;
|
import static org.mockito.ArgumentMatchers.anyBoolean;
|
||||||
import static org.mockito.ArgumentMatchers.anyLong;
|
|
||||||
import org.mockito.Mockito;
|
import org.mockito.Mockito;
|
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
import static org.mockito.Mockito.times;
|
import static org.mockito.Mockito.times;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
|
|
||||||
|
@ -433,7 +434,7 @@ public class TestDFSOutputStream {
|
||||||
EnumSet.of(CreateFlag.CREATE), (short) 3, 1024, null , 1024, null);
|
EnumSet.of(CreateFlag.CREATE), (short) 3, 1024, null , 1024, null);
|
||||||
DFSOutputStream spyDFSOutputStream = Mockito.spy(dfsOutputStream);
|
DFSOutputStream spyDFSOutputStream = Mockito.spy(dfsOutputStream);
|
||||||
spyDFSOutputStream.closeThreads(anyBoolean());
|
spyDFSOutputStream.closeThreads(anyBoolean());
|
||||||
verify(spyClient, times(1)).endFileLease(anyLong());
|
verify(spyClient, times(1)).endFileLease(anyString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
@ -852,13 +852,13 @@ public class TestHASafeMode {
|
||||||
null);
|
null);
|
||||||
create.write(testData.getBytes());
|
create.write(testData.getBytes());
|
||||||
create.hflush();
|
create.hflush();
|
||||||
long fileId = ((DFSOutputStream)create.
|
String renewLeaseKey = ((DFSOutputStream)create.
|
||||||
getWrappedStream()).getFileId();
|
getWrappedStream()).getUniqKey();
|
||||||
FileStatus fileStatus = dfs.getFileStatus(filePath);
|
FileStatus fileStatus = dfs.getFileStatus(filePath);
|
||||||
DFSClient client = DFSClientAdapter.getClient(dfs);
|
DFSClient client = DFSClientAdapter.getClient(dfs);
|
||||||
// add one dummy block at NN, but not write to DataNode
|
// add one dummy block at NN, but not write to DataNode
|
||||||
ExtendedBlock previousBlock =
|
ExtendedBlock previousBlock =
|
||||||
DFSClientAdapter.getPreviousBlock(client, fileId);
|
DFSClientAdapter.getPreviousBlock(client, renewLeaseKey);
|
||||||
DFSClientAdapter.getNamenode(client).addBlock(
|
DFSClientAdapter.getNamenode(client).addBlock(
|
||||||
pathString,
|
pathString,
|
||||||
client.getClientName(),
|
client.getClientName(),
|
||||||
|
|
Loading…
Reference in New Issue