All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.kestra.storage.gcs.GcsStorage Maven / Gradle / Ivy

There is a newer version: 0.20.0
Show newest version
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