io.trino.filesystem.gcs.GcsFileSystem Maven / Gradle / Ivy
/*
* 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 io.trino.filesystem.gcs;
import com.google.api.gax.paging.Page;
import com.google.cloud.storage.Blob;
import com.google.cloud.storage.BlobId;
import com.google.cloud.storage.BlobInfo;
import com.google.cloud.storage.Storage;
import com.google.cloud.storage.Storage.BlobListOption;
import com.google.cloud.storage.StorageBatch;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterators;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import io.airlift.units.Duration;
import io.trino.filesystem.FileIterator;
import io.trino.filesystem.Location;
import io.trino.filesystem.TrinoFileSystem;
import io.trino.filesystem.TrinoFileSystemException;
import io.trino.filesystem.TrinoInputFile;
import io.trino.filesystem.TrinoOutputFile;
import io.trino.filesystem.UriLocation;
import io.trino.filesystem.encryption.EncryptionKey;
import java.io.IOException;
import java.net.URISyntaxException;
import java.net.URL;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.Set;
import static com.google.cloud.storage.Storage.BlobListOption.currentDirectory;
import static com.google.cloud.storage.Storage.BlobListOption.matchGlob;
import static com.google.cloud.storage.Storage.BlobListOption.pageSize;
import static com.google.cloud.storage.Storage.SignUrlOption.withExtHeaders;
import static com.google.cloud.storage.Storage.SignUrlOption.withV4Signature;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.collect.ImmutableMap.toImmutableMap;
import static com.google.common.collect.Iterables.partition;
import static io.airlift.concurrent.MoreFutures.getFutureValue;
import static io.trino.filesystem.gcs.GcsUtils.encodedKey;
import static io.trino.filesystem.gcs.GcsUtils.getBlob;
import static io.trino.filesystem.gcs.GcsUtils.handleGcsException;
import static io.trino.filesystem.gcs.GcsUtils.keySha256Checksum;
import static java.util.Objects.requireNonNull;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
public class GcsFileSystem
implements TrinoFileSystem
{
private final ListeningExecutorService executorService;
private final Storage storage;
private final int readBlockSizeBytes;
private final long writeBlockSizeBytes;
private final int pageSize;
private final int batchSize;
public GcsFileSystem(ListeningExecutorService executorService, Storage storage, int readBlockSizeBytes, long writeBlockSizeBytes, int pageSize, int batchSize)
{
this.executorService = requireNonNull(executorService, "executorService is null");
this.storage = requireNonNull(storage, "storage is null");
this.readBlockSizeBytes = readBlockSizeBytes;
this.writeBlockSizeBytes = writeBlockSizeBytes;
this.pageSize = pageSize;
this.batchSize = batchSize;
}
@Override
public TrinoInputFile newInputFile(Location location)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.empty(), Optional.empty(), Optional.empty());
}
@Override
public TrinoInputFile newEncryptedInputFile(Location location, EncryptionKey key)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.empty(), Optional.empty(), Optional.of(key));
}
@Override
public TrinoInputFile newInputFile(Location location, long length)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.of(length), Optional.empty(), Optional.empty());
}
@Override
public TrinoInputFile newEncryptedInputFile(Location location, long length, EncryptionKey key)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.of(length), Optional.empty(), Optional.of(key));
}
@Override
public TrinoInputFile newInputFile(Location location, long length, Instant lastModified)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.of(length), Optional.of(lastModified), Optional.empty());
}
@Override
public TrinoInputFile newEncryptedInputFile(Location location, long length, Instant lastModified, EncryptionKey key)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsInputFile(gcsLocation, storage, readBlockSizeBytes, OptionalLong.of(length), Optional.of(lastModified), Optional.of(key));
}
@Override
public TrinoOutputFile newOutputFile(Location location)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsOutputFile(gcsLocation, storage, writeBlockSizeBytes, Optional.empty());
}
@Override
public TrinoOutputFile newEncryptedOutputFile(Location location, EncryptionKey key)
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
return new GcsOutputFile(gcsLocation, storage, writeBlockSizeBytes, Optional.of(key));
}
@Override
public void deleteFile(Location location)
throws IOException
{
GcsLocation gcsLocation = new GcsLocation(location);
checkIsValidFile(gcsLocation);
getBlob(storage, gcsLocation).ifPresent(Blob::delete);
}
@Override
public void deleteFiles(Collection locations)
throws IOException
{
List> batchFutures = new ArrayList<>();
try {
for (List locationBatch : partition(locations, batchSize)) {
StorageBatch batch = storage.batch();
for (Location location : locationBatch) {
GcsLocation gcsLocation = new GcsLocation(location);
batch.delete(BlobId.of(gcsLocation.bucket(), gcsLocation.path()));
}
batchFutures.add(executorService.submit(batch::submit));
}
getFutureValue(Futures.allAsList(batchFutures));
}
catch (RuntimeException e) {
throw handleGcsException(e, "delete files", locations);
}
}
@Override
public void deleteDirectory(Location location)
throws IOException
{
GcsLocation gcsLocation = new GcsLocation(normalizeToDirectory(location));
try {
List> batchFutures = new ArrayList<>();
for (List blobBatch : partition(getPage(gcsLocation).iterateAll(), batchSize)) {
StorageBatch batch = storage.batch();
for (Blob blob : blobBatch) {
batch.delete(blob.getBlobId());
}
batchFutures.add(executorService.submit(batch::submit));
}
getFutureValue(Futures.allAsList(batchFutures));
}
catch (RuntimeException e) {
throw handleGcsException(e, "deleting directory", gcsLocation);
}
}
@Override
public void renameFile(Location source, Location target)
throws IOException
{
throw new TrinoFileSystemException("GCS does not support renames");
}
@Override
public FileIterator listFiles(Location location)
throws IOException
{
GcsLocation gcsLocation = new GcsLocation(normalizeToDirectory(location));
try {
return new GcsFileIterator(gcsLocation, getPage(gcsLocation));
}
catch (RuntimeException e) {
throw handleGcsException(e, "listing files", gcsLocation);
}
}
private static Location normalizeToDirectory(Location location)
{
String path = location.path();
if (!path.isEmpty() && !path.endsWith("/")) {
return location.appendSuffix("/");
}
return location;
}
private static void checkIsValidFile(GcsLocation gcsLocation)
{
checkState(!gcsLocation.path().isEmpty(), "Location path is empty: %s", gcsLocation);
checkState(!gcsLocation.path().endsWith("/"), "Location path ends with a slash: %s", gcsLocation);
}
private Page getPage(GcsLocation location, BlobListOption... blobListOptions)
{
List optionsBuilder = new ArrayList<>();
if (!location.path().isEmpty()) {
optionsBuilder.add(BlobListOption.prefix(location.path()));
}
optionsBuilder.addAll(Arrays.asList(blobListOptions));
optionsBuilder.add(pageSize(this.pageSize));
return storage.list(location.bucket(), optionsBuilder.toArray(BlobListOption[]::new));
}
@Override
public Optional directoryExists(Location location)
throws IOException
{
// Notes:
// GCS is not hierarchical, there is no way to determine the difference
// between an empty blob and a directory: for this case Optional.empty() will be returned per the super
// method spec.
//
// Note on blob.isDirectory: The isDirectory() method returns false unless invoked via storage.list()
// with currentDirectory() enabled for empty blobs intended to be used as directories.
// The only time blob.isDirectory() is true is when an object was created that introduced the path:
//
// Example 1: createBlob("bucket", "dir") creates an empty blob intended to be used as a "directory"
// you can then create a file "bucket", "dir/file")
// Invoking blob.isDirectory() on "dir" returns false even after the "dir/file" object is created.
//
// Example 2: createBlob("bucket", "dir2/file") when "dir2" does not exist will return true for isDirectory()
// when invoked on the "dir2/" path. Also note that the blob name has a trailing slash.
// This behavior is only enabled with BlobListOption.currentDirectory() and isDirectory() is only true when the blob
// is returned from a storage.list operation.
GcsLocation gcsLocation = new GcsLocation(location);
if (gcsLocation.path().isEmpty()) {
return Optional.of(bucketExists(gcsLocation.bucket()));
}
if (listFiles(location).hasNext()) {
return Optional.of(true);
}
return Optional.empty();
}
private boolean bucketExists(String bucket)
{
return storage.get(bucket) != null;
}
@Override
public void createDirectory(Location location)
throws IOException
{
validateGcsLocation(location);
}
@Override
public void renameDirectory(Location source, Location target)
throws IOException
{
throw new TrinoFileSystemException("GCS does not support directory renames");
}
@Override
public Set listDirectories(Location location)
throws IOException
{
GcsLocation gcsLocation = new GcsLocation(normalizeToDirectory(location));
try {
Page page = getPage(gcsLocation, currentDirectory(), matchGlob(gcsLocation.path() + "*/"));
Iterator blobIterator = Iterators.filter(page.iterateAll().iterator(), BlobInfo::isDirectory);
ImmutableSet.Builder locationBuilder = ImmutableSet.builder();
while (blobIterator.hasNext()) {
locationBuilder.add(Location.of(gcsLocation.getBase() + blobIterator.next().getName()));
}
return locationBuilder.build();
}
catch (RuntimeException e) {
throw handleGcsException(e, "listing directories", gcsLocation);
}
}
@Override
public Optional createTemporaryDirectory(Location targetPath, String temporaryPrefix, String relativePrefix)
{
validateGcsLocation(targetPath);
// GCS does not have directories
return Optional.empty();
}
@Override
public Optional preSignedUri(Location location, Duration ttl)
throws IOException
{
return preSignedUri(location, ttl, Optional.empty());
}
@Override
public Optional encryptedPreSignedUri(Location location, Duration ttl, EncryptionKey key)
throws IOException
{
return preSignedUri(location, ttl, Optional.of(key));
}
private Optional preSignedUri(Location location, Duration ttl, Optional key)
throws IOException
{
GcsLocation gcsLocation = new GcsLocation(location);
BlobInfo blobInfo = BlobInfo
.newBuilder(BlobId.of(gcsLocation.bucket(), gcsLocation.path()))
.build();
Map extHeaders = preSignedHeaders(key);
URL url = storage.signUrl(blobInfo, ttl.toMillis(), MILLISECONDS, withV4Signature(), withExtHeaders(extHeaders));
try {
return Optional.of(new UriLocation(url.toURI(), toMultiMap(extHeaders)));
}
catch (URISyntaxException e) {
throw new IOException("Error creating URI for location: " + location, e);
}
}
private static Map preSignedHeaders(Optional key)
{
if (key.isEmpty()) {
return ImmutableMap.of();
}
EncryptionKey encryption = key.get();
ImmutableMap.Builder headers = ImmutableMap.builderWithExpectedSize(3);
headers.put("x-goog-encryption-algorithm", encryption.algorithm());
headers.put("x-goog-encryption-key", encodedKey(encryption));
headers.put("x-goog-encryption-key-sha256", keySha256Checksum(encryption));
return headers.buildOrThrow();
}
private Map> toMultiMap(Map extHeaders)
{
return extHeaders.entrySet().stream()
.map(entry -> Map.entry(entry.getKey(), ImmutableList.of(entry.getValue())))
.collect(toImmutableMap(Map.Entry::getKey, Map.Entry::getValue));
}
@SuppressWarnings("ResultOfObjectAllocationIgnored")
private static void validateGcsLocation(Location location)
{
new GcsLocation(location);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy