io.kestra.storage.gcs.GcsStorage Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of storage-gcs Show documentation
Show all versions of storage-gcs Show documentation
Google Cloud Storage storage plugin for Kestra
package io.kestra.storage.gcs;
import com.google.api.gax.paging.Page;
import com.google.cloud.WriteChannel;
import com.google.cloud.storage.*;
import io.kestra.core.storages.FileAttributes;
import io.micronaut.core.annotation.Introspected;
import io.kestra.core.storages.StorageInterface;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.nio.ByteBuffer;
import java.nio.channels.Channels;
import java.nio.channels.ReadableByteChannel;
import java.util.*;
import java.util.stream.Collectors;
import jakarta.inject.Inject;
import jakarta.inject.Singleton;
import org.jetbrains.annotations.NotNull;
import static io.kestra.core.utils.Rethrow.throwFunction;
@Singleton
@GcsStorageEnabled
@Introspected
public class GcsStorage implements StorageInterface {
@Inject
GcsClientFactory factory;
@Inject
GcsConfig config;
private Storage client() {
return factory.of(config);
}
private BlobId blob(String tenantId, URI uri) {
String path = getPath(tenantId, uri);
return blob(path);
}
@NotNull
private BlobId blob(String path) {
return BlobId.of(this.config.getBucket(), path);
}
@NotNull
private String getPath(String tenantId, URI uri) {
if (uri == null) {
uri = URI.create("/");
}
parentTraversalGuard(uri);
String path = uri.getPath();
if (!path.startsWith("/")) {
path = "/" + path;
}
if (tenantId == null) {
return path;
}
return "/" + tenantId + path;
}
// Traversal does not work with gcs but it just return empty objects so throwing is more explicit
private void parentTraversalGuard(URI uri) {
if (uri.toString().contains("..")) {
throw new IllegalArgumentException("File should be accessed with their full path and not using relative '..' path.");
}
}
@Override
public InputStream get(String tenantId, URI uri) throws IOException {
try {
Blob blob = this.client().get(this.blob(tenantId, URI.create(uri.getPath())));
if (blob == null || !blob.exists()) {
throw new FileNotFoundException(uri + " (File not found)");
}
ReadableByteChannel reader = blob.reader();
return Channels.newInputStream(reader);
} catch (StorageException e) {
throw new IOException(e);
}
}
@Override
public List list(String tenantId, URI uri) throws IOException {
String path = getPath(tenantId, uri);
String prefix = (path.endsWith("/")) ? path : path + "/";
Page blobs = this.client().list(config.bucket, Storage.BlobListOption.prefix(prefix),
Storage.BlobListOption.currentDirectory());
List list = blobs.streamAll()
.filter(blob -> {
String key = blob.getName().substring(prefix.length());
// Remove recursive result and requested dir
return !key.isEmpty() && !Objects.equals(key, prefix) && new File(key).getParent() == null;
})
.map(throwFunction(this::getGcsFileAttributes))
.toList();
if(list.isEmpty()) {
// this will throw FileNotFound if there is no directory
this.getAttributes(tenantId, uri);
}
return list;
}
@Override
public boolean exists(String tenantId, URI uri) {
try {
Blob blob = this.client().get(this.blob(tenantId, URI.create(uri.getPath())));
return blob != null && blob.exists();
} catch (StorageException e) {
return false;
}
}
@Override
public Long size(String tenantId,URI uri) throws IOException {
try {
Blob blob = this.client().get(this.blob(tenantId, URI.create(uri.getPath())));
if (blob == null || !blob.exists()) {
throw new FileNotFoundException(uri + " (File not found)");
}
return blob.getSize();
} catch (StorageException e) {
throw new IOException(e);
}
}
@Override
public Long lastModifiedTime(String tenantId,URI uri) throws IOException {
try {
Blob blob = this.client().get(this.blob(tenantId, URI.create(uri.getPath())));
if (blob == null || !blob.exists()) {
throw new FileNotFoundException(uri + " (File not found)");
}
return blob.getUpdateTimeOffsetDateTime().toInstant().toEpochMilli();
} catch (StorageException e) {
throw new IOException(e);
}
}
@Override
public FileAttributes getAttributes(String tenantId, URI uri) throws IOException {
String path = getPath(tenantId, uri);
if (!exists(tenantId, uri)) {
path = path + "/";
}
Blob blob = this.client().get(this.blob(path));
if (blob == null) {
throw new FileNotFoundException("%s not found.".formatted(uri));
}
return getGcsFileAttributes(blob);
}
private FileAttributes getGcsFileAttributes(Blob blob) {
GcsFileAttributes.GcsFileAttributesBuilder builder = GcsFileAttributes.builder()
.fileName(new File(blob.getName()).getName())
.blobInfo(blob.asBlobInfo());
if (blob.getName().endsWith("/")) {
builder.isDirectory(true);
}
return builder.build();
}
@Override
public URI put(String tenantId, URI uri, InputStream data) throws IOException {
try {
String path = getPath(tenantId, uri);
mkdirs(path);
BlobInfo blobInfo = BlobInfo
.newBuilder(this.blob(tenantId, uri))
.build();
try (WriteChannel writer = this.client().writer(blobInfo)) {
byte[] buffer = new byte[10_240];
int limit;
while ((limit = data.read(buffer)) >= 0) {
writer.write(ByteBuffer.wrap(buffer, 0, limit));
}
}
data.close();
return URI.create("kestra://" + uri.getPath());
} catch (StorageException e) {
throw new IOException(e);
}
}
private void mkdirs(String path) {
path = path.replaceAll("^/*", "");
String[] directories = path.split("/");
StringBuilder aggregatedPath = new StringBuilder("/");
// perform 1 put request per parent directory in the path
for (int i = 0; i <= directories.length - (path.endsWith("/") ? 1 : 2); i++) {
aggregatedPath.append(directories[i]).append("/");
BlobInfo blobInfo = BlobInfo
.newBuilder(this.blob(aggregatedPath.toString()))
.build();
this.client().create(blobInfo);
}
}
public boolean delete(String tenantId, URI uri) throws IOException {
return !deleteByPrefix(tenantId, uri).isEmpty();
}
@Override
public URI createDirectory(String tenantId, URI uri) {
String path = getPath(tenantId, uri);
if (!path.endsWith("/")) {
path = path + "/";
}
mkdirs(path);
return createUri(uri.getPath());
}
@Override
public URI move(String tenantId, URI from, URI to) throws IOException {
String path = getPath(tenantId, from);
StorageBatch batch = this.client().batch();
if (getAttributes(tenantId, from).getType() == FileAttributes.FileType.File) {
// move just a file
BlobId source = blob(path);
BlobId target = blob(tenantId, to);
moveFile(source, target, batch);
} else {
// move directories
String prefix = (!path.endsWith("/")) ? path + "/" : path;
Page list = client().list(config.bucket, Storage.BlobListOption.prefix(prefix));
list.streamAll().forEach(blob -> {
BlobId target = blob(getPath(tenantId, to) + "/" + blob.getName().substring(prefix.length()));
moveFile(blob.getBlobId(), target, batch);
});
}
batch.submit();
return createUri(to.getPath());
}
private void moveFile(BlobId source, BlobId target, StorageBatch batch) {
client().copy(Storage.CopyRequest.newBuilder().setSource(source).setTarget(target).build());
batch.delete(source);
}
@Override
public List deleteByPrefix(String tenantId, URI storagePrefix) throws IOException {
try {
StorageBatch batch = this.client().batch();
Map> results = new HashMap<>();
String prefix = getPath(tenantId, storagePrefix);
Page blobs = this.client()
.list(this.config.getBucket(),
Storage.BlobListOption.prefix(prefix)
);
for (Blob blob : blobs.iterateAll()) {
results.put(URI.create("kestra://" + blob.getBlobId().getName().replace(tenantId + "/", "").replaceAll("/$", "")), batch.delete(blob.getBlobId()));
}
if (results.isEmpty()) {
return List.of();
}
batch.submit();
if (!results.entrySet().stream().allMatch(r -> r.getValue() != null && r.getValue().get())) {
throw new IOException("Unable to delete all files, failed on [" +
results
.entrySet()
.stream()
.filter(r -> r.getValue() == null || !r.getValue().get())
.map(r -> r.getKey().getPath())
.collect(Collectors.joining(", ")) +
"]");
}
return new ArrayList<>(results.keySet());
} catch (StorageException e) {
throw new IOException(e);
}
}
private static URI createUri(String key) {
return URI.create("kestra://%s".formatted(key));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy