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

org.teiid.s3.S3Connection Maven / Gradle / Ivy

The newest version!
/*
 * Copyright Red Hat, Inc. and/or its affiliates
 * and other contributors as indicated by the @author tags and
 * the COPYRIGHT.txt file distributed with this work.
 *
 * 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 org.teiid.s3;

import java.io.InputStream;
import java.util.ArrayList;
import java.util.regex.Pattern;

import org.teiid.file.VirtualFile;
import org.teiid.file.VirtualFileConnection;
import org.teiid.translator.TranslatorException;

import com.amazonaws.SdkClientException;
import com.amazonaws.services.s3.AmazonS3;
import com.amazonaws.services.s3.model.ListObjectsRequest;
import com.amazonaws.services.s3.model.ObjectListing;
import com.amazonaws.services.s3.model.ObjectMetadata;
import com.amazonaws.services.s3.model.PutObjectRequest;
import com.amazonaws.services.s3.model.S3ObjectSummary;
import com.amazonaws.services.s3.model.SSECustomerKey;

public class S3Connection implements VirtualFileConnection {

    private static final char STAR = '*';
    static final String SLASH = "/"; //$NON-NLS-1$

    private final S3Configuration s3Config;
    private final AmazonS3 s3Client;

    public S3Connection(S3Configuration s3Config, AmazonS3 s3Client) {
        this.s3Config = s3Config;
        this.s3Client = s3Client;
    }

    @Override
    public VirtualFile[] getFiles(String s) throws TranslatorException {
        if (s == null) {
            return null;
        }
        return search(s);
    }

    private VirtualFile[] search(String s) {
        //the strategy is to split the string into a prefix
        //and then apply the remaining as a pattern match
        int firstStar = s.indexOf(STAR);
        String prefix = s;
        String remainingPath = ""; //$NON-NLS-1$
        while (firstStar != -1) {
            prefix = s.substring(0, firstStar);
            remainingPath = s.substring(firstStar, s.length());
            if (remainingPath.length() < 2 || remainingPath.charAt(1) != STAR) {
                break;
            }
            prefix += STAR; //literal star match
            remainingPath = remainingPath.substring(2);
            firstStar = remainingPath.indexOf(STAR);
        }
        ArrayList s3VirtualFiles = new ArrayList<>();
        ListObjectsRequest listObjectsRequest = new ListObjectsRequest()
                .withBucketName(s3Config.getBucket())
                .withPrefix(prefix);
        ObjectListing objectListing = s3Client.listObjects(listObjectsRequest);
        Pattern pattern = convertPathToPattern(remainingPath);
        while(objectListing != null) {
            for(S3ObjectSummary s3ObjectSummary : objectListing.getObjectSummaries()) {
                String remainingKey = s3ObjectSummary.getKey().substring(prefix.length());
                //make sure the remaining path matches - special case if empty
                if ((remainingPath.isEmpty() && (prefix.endsWith(SLASH)
                        || remainingKey.startsWith(SLASH)) && remainingKey.lastIndexOf('/') < 1)
                        || pattern.matcher(remainingKey).matches()) {
                    s3VirtualFiles.add(new S3VirtualFile(s3Client, s3ObjectSummary, s3Config));
                }
            }
            if(!objectListing.isTruncated()) {
                break;
            }
            objectListing = s3Client.listNextBatchOfObjects(objectListing);
        }
        return s3VirtualFiles.toArray(new VirtualFile[s3VirtualFiles.size()]);
    }

    /**
     * Converts the path to a regex pattern.
     * @param remainingPath
     * @return
     */
    Pattern convertPathToPattern(String remainingPath) {
        StringBuilder builder = new StringBuilder();
        for (int index = 0; index < remainingPath.length(); index++) {
            int startIndex = index;
            index = remainingPath.indexOf(STAR, index);
            int endIndex = index;
            if (index == -1) {
                endIndex = remainingPath.length();
            }
            String literal = remainingPath.substring(startIndex, endIndex);
            builder.append(Pattern.quote(literal));
            if (index == -1) {
                break;
            }
            //replace the star
            if (index < remainingPath.length() - 1 && remainingPath.charAt(index + 1) == STAR) {
                index++;
                builder.append("[*]"); //$NON-NLS-1$
            } else {
                builder.append("[^/]*"); //$NON-NLS-1$
            }
        }
        //match to the end
        builder.append("$"); //$NON-NLS-1$
        Pattern pattern = Pattern.compile(builder.toString());
        return pattern;
    }

    @Override
    public void add(InputStream in, String path) throws TranslatorException {
        add(in, path, null);
    }

    @Override
    public void add(InputStream inputStream, String s, FileMetadata fileMetadata)
            throws TranslatorException {
        ObjectMetadata metadata = new ObjectMetadata();
        try{
            if (fileMetadata != null) {
                Long size = fileMetadata.size();
                if (size != null) {
                    metadata.setContentLength(size);
                }
            }
            PutObjectRequest request = new PutObjectRequest(s3Config.getBucket(), s, inputStream, metadata);
            if(s3Config.getSseKey() != null) {
                request.withSSECustomerKey(new SSECustomerKey(s3Config.getSseKey()).withAlgorithm(s3Config.getSseAlgorithm()));
            }
            s3Client.putObject(request);
        } catch (SdkClientException e){
            throw new TranslatorException(e);
        }
    }

    @Override
    public boolean remove(String s) throws TranslatorException {
        try{
            //delete does not check for existence
            if (!s3Client.doesObjectExist(s3Config.getBucket(), s)) {
                return false;
            }
            s3Client.deleteObject(s3Config.getBucket(), s);
        } catch (SdkClientException e){
            return false;
        }
        return true;
    }

    @Override
    public void close() {

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy