/** * 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.hbase.io.asyncfs; import com.google.common.base.Throwables; import com.google.common.util.concurrent.ThreadFactoryBuilder; import io.netty.channel.EventLoop; import java.io.IOException; import java.io.InterruptedIOException; import java.nio.ByteBuffer; import java.util.concurrent.CompletableFuture; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import org.apache.hadoop.fs.CommonConfigurationKeysPublic; import org.apache.hadoop.fs.FSDataOutputStream; import org.apache.hadoop.fs.FileSystem; import org.apache.hadoop.fs.Path; import org.apache.hadoop.hbase.classification.InterfaceAudience; import org.apache.hadoop.hbase.io.ByteArrayOutputStream; import org.apache.hadoop.hbase.util.CancelableProgressable; import org.apache.hadoop.hdfs.DistributedFileSystem; import org.apache.hadoop.hdfs.protocol.DatanodeInfo; /** * Helper class for creating AsyncFSOutput. */ @InterfaceAudience.Private public final class AsyncFSOutputHelper { private AsyncFSOutputHelper() { } /** * Create {@link FanOutOneBlockAsyncDFSOutput} for {@link DistributedFileSystem}, and a simple * implementation for other {@link FileSystem} which wraps around a {@link FSDataOutputStream}. */ public static AsyncFSOutput createOutput(FileSystem fs, Path f, boolean overwrite, boolean createParent, short replication, long blockSize, final EventLoop eventLoop) throws IOException { if (fs instanceof DistributedFileSystem) { return FanOutOneBlockAsyncDFSOutputHelper.createOutput((DistributedFileSystem) fs, f, overwrite, createParent, replication, blockSize, eventLoop); } final FSDataOutputStream fsOut; int bufferSize = fs.getConf().getInt(CommonConfigurationKeysPublic.IO_FILE_BUFFER_SIZE_KEY, CommonConfigurationKeysPublic.IO_FILE_BUFFER_SIZE_DEFAULT); if (createParent) { fsOut = fs.create(f, overwrite, bufferSize, replication, blockSize, null); } else { fsOut = fs.createNonRecursive(f, overwrite, bufferSize, replication, blockSize, null); } final ExecutorService flushExecutor = Executors.newSingleThreadExecutor(new ThreadFactoryBuilder().setDaemon(true) .setNameFormat("AsyncFSOutputFlusher-" + f.toString().replace("%", "%%")).build()); return new AsyncFSOutput() { private final ByteArrayOutputStream out = new ByteArrayOutputStream(); @Override public void write(final byte[] b, final int off, final int len) { if (eventLoop.inEventLoop()) { out.write(b, off, len); } else { eventLoop.submit(() -> out.write(b, off, len)).syncUninterruptibly(); } } @Override public void write(byte[] b) { write(b, 0, b.length); } @Override public void recoverAndClose(CancelableProgressable reporter) throws IOException { fsOut.close(); } @Override public DatanodeInfo[] getPipeline() { return new DatanodeInfo[0]; } private void flush0(CompletableFuture<Long> future, boolean sync) { try { synchronized (out) { fsOut.write(out.getBuffer(), 0, out.size()); out.reset(); } } catch (IOException e) { eventLoop.execute(() -> future.completeExceptionally(e)); return; } try { if (sync) { fsOut.hsync(); } else { fsOut.hflush(); } long pos = fsOut.getPos(); eventLoop.execute(() -> future.complete(pos)); } catch (IOException e) { eventLoop.execute(() -> future.completeExceptionally(e)); } } @Override public CompletableFuture<Long> flush(boolean sync) { CompletableFuture<Long> future = new CompletableFuture<>(); flushExecutor.execute(() -> flush0(future, sync)); return future; } @Override public void close() throws IOException { try { flushExecutor.submit(() -> { synchronized (out) { fsOut.write(out.getBuffer(), 0, out.size()); out.reset(); } return null; }).get(); } catch (InterruptedException e) { throw new InterruptedIOException(); } catch (ExecutionException e) { Throwables.propagateIfPossible(e.getCause(), IOException.class); throw new IOException(e.getCause()); } finally { flushExecutor.shutdown(); } fsOut.close(); } @Override public int buffered() { return out.size(); } @Override public void writeInt(int i) { out.writeInt(i); } @Override public void write(ByteBuffer bb) { out.write(bb, bb.position(), bb.remaining()); } }; } }