com.google.cloud.hadoop.fs.gcs.GoogleHadoopOutputStream Maven / Gradle / Ivy
/*
* Copyright 2013 Google Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.cloud.hadoop.fs.gcs;
import com.google.cloud.hadoop.gcsio.CreateFileOptions;
import com.google.cloud.hadoop.gcsio.GoogleCloudStorageFileSystem;
import com.google.cloud.hadoop.gcsio.GoogleCloudStorageOptions;
import com.google.cloud.hadoop.util.GoogleCloudStorageEventBus;
import com.google.cloud.hadoop.util.ITraceFactory;
import com.google.common.flogger.GoogleLogger;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URI;
import java.nio.channels.Channels;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.WritableByteChannel;
import org.apache.hadoop.fs.FileAlreadyExistsException;
import org.apache.hadoop.fs.FileSystem;
/** A buffered output stream that allows writing to a GCS object. */
class GoogleHadoopOutputStream extends OutputStream {
private static final GoogleLogger logger = GoogleLogger.forEnclosingClass();
private final GhfsStorageStatistics storageStatistics;
private final ITraceFactory traceFactory;
// All store IO access goes through this.
private WritableByteChannel channel;
// Output stream corresponding to channel.
private OutputStream out;
// Path of the file to write to.
private final URI gcsPath;
// Statistics tracker provided by the parent GoogleHadoopFileSystemBase for recording
// numbers of bytes written.
private final FileSystem.Statistics statistics;
private final GhfsStreamStats streamStats;
/**
* Constructs an instance of GoogleHadoopOutputStream object.
*
* @param ghfs Instance of GoogleHadoopFileSystemBase.
* @param gcsPath Path of the file to write to.
* @param statistics File system statistics object.
* @throws IOException if an IO error occurs.
*/
GoogleHadoopOutputStream(
GoogleHadoopFileSystemBase ghfs,
URI gcsPath,
FileSystem.Statistics statistics,
CreateFileOptions createFileOptions)
throws IOException {
logger.atFiner().log(
"GoogleHadoopOutputStream(gcsPath: %s, createFileOptions: %s)", gcsPath, createFileOptions);
this.gcsPath = gcsPath;
this.statistics = statistics;
GoogleCloudStorageFileSystem gcsfs = ghfs.getGcsFs();
this.channel = createChannel(gcsfs, gcsPath, createFileOptions);
this.out = createOutputStream(this.channel, gcsfs.getOptions().getCloudStorageOptions());
this.storageStatistics = ghfs.getStorageStatistics();
this.streamStats =
new GhfsStreamStats(storageStatistics, GhfsStatistic.STREAM_WRITE_OPERATIONS, gcsPath);
this.traceFactory = ghfs.getTraceFactory();
}
private static WritableByteChannel createChannel(
GoogleCloudStorageFileSystem gcsfs, URI gcsPath, CreateFileOptions options)
throws IOException {
try {
return gcsfs.create(gcsPath, options);
} catch (java.nio.file.FileAlreadyExistsException e) {
GoogleCloudStorageEventBus.postOnException();
throw (FileAlreadyExistsException)
new FileAlreadyExistsException(String.format("'%s' already exists", gcsPath))
.initCause(e);
}
}
private static OutputStream createOutputStream(
WritableByteChannel channel, GoogleCloudStorageOptions gcsOptions) {
OutputStream out = Channels.newOutputStream(channel);
int bufferSize = gcsOptions.getWriteChannelOptions().getBufferSize();
return bufferSize > 0 ? new BufferedOutputStream(out, bufferSize) : out;
}
/** Writes the specified byte to this output stream. */
@Override
public void write(int b) throws IOException {
long start = System.nanoTime();
throwIfNotOpen();
out.write(b);
statistics.incrementBytesWritten(1);
statistics.incrementWriteOps(1);
// Using a lightweight implementation to update instrumentation. This method can be called quite
// frequently and need to be lightweight.
streamStats.updateWriteStreamStats(1, start);
}
/**
* Writes to this output stream 'len' bytes of the specified buffer starting at the given offset.
*/
@Override
public void write(byte[] b, int offset, int len) throws IOException {
long start = System.nanoTime();
throwIfNotOpen();
out.write(b, offset, len);
statistics.incrementBytesWritten(len);
statistics.incrementWriteOps(1);
streamStats.updateWriteStreamStats(len, start);
}
/** Closes this output stream and releases any system resources associated with this stream. */
@Override
public void close() throws IOException {
GhfsStorageStatistics.trackDuration(
storageStatistics,
GhfsStatistic.STREAM_WRITE_CLOSE_OPERATIONS,
gcsPath,
traceFactory,
() -> {
logger.atFiner().log("close(%s)", gcsPath);
try {
if (out != null) {
out.close();
}
} finally {
out = null;
channel = null;
streamStats.close();
}
return null;
});
}
private boolean isOpen() {
return out != null;
}
private void throwIfNotOpen() throws IOException {
if (!isOpen()) {
GoogleCloudStorageEventBus.postOnException();
throw new ClosedChannelException();
}
}
WritableByteChannel getInternalChannel() {
return channel;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy