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

com.ikelin.S3Wagon Maven / Gradle / Ivy

package com.ikelin;

import com.amazonaws.services.s3.AmazonS3;
import com.amazonaws.services.s3.AmazonS3ClientBuilder;
import com.amazonaws.services.s3.model.ObjectMetadata;
import com.amazonaws.services.s3.model.PutObjectRequest;
import com.amazonaws.services.s3.model.S3Object;
import com.amazonaws.services.s3.transfer.TransferManager;
import com.amazonaws.services.s3.transfer.TransferManagerBuilder;
import com.amazonaws.services.s3.transfer.Upload;
import org.apache.maven.wagon.AbstractWagon;
import org.apache.maven.wagon.ConnectionException;
import org.apache.maven.wagon.InputData;
import org.apache.maven.wagon.OutputData;
import org.apache.maven.wagon.ResourceDoesNotExistException;
import org.apache.maven.wagon.StreamWagon;
import org.apache.maven.wagon.TransferFailedException;
import org.apache.maven.wagon.resource.Resource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
import java.util.Iterator;
import java.util.concurrent.ConcurrentHashMap;

/**
 * An Apache Maven Wagon Provider for Amazon S3.
 */
public class S3Wagon extends StreamWagon {

  private static final Logger LOGGER = LoggerFactory.getLogger(S3Wagon.class);

  private final AmazonS3 amazonS3;
  private final TransferManager transferManager;
  private final ConcurrentHashMap uploads;

  /**
   * Creates a S3 Wagon using default {@code AmazonS3} and default {@code TransferManager}.
   */
  public S3Wagon() {
    this(AmazonS3ClientBuilder.defaultClient(), TransferManagerBuilder.defaultTransferManager(),
        new ConcurrentHashMap<>());
  }

  /**
   * Creates a S3 Wagon using the provided {@code AmazonS3}, {@code TransferManager}, and {@code
   * ConcurrentHashMap} of {@code Upload} and its {@code OutputStream} in progress.
   *
   * @param amazonS3 a {@code AmazonS3} used to download artifacts
   * @param transferManager a {@code TransferManager} used to upload artifacts
   * @param uploads a {@code ConcurrentHashMap} map that holds {@code Upload} and its {@code
   *     OutputStream}
   */
  protected S3Wagon(final AmazonS3 amazonS3, final TransferManager transferManager,
      final ConcurrentHashMap uploads) {
    super();
    this.amazonS3 = amazonS3;
    this.transferManager = transferManager;
    this.uploads = uploads;
  }

  @Override
  public void fillInputData(final InputData inputData) throws ResourceDoesNotExistException {
    Resource resource = inputData.getResource();
    String bucket = getBucket();
    String key = getKey(resource);

    if (!amazonS3.doesObjectExist(bucket, key)) {
      throw new ResourceDoesNotExistException(
          "File at S3 bucket=" + bucket + " key=" + key + " does not exist");
    }

    S3Object s3Object = amazonS3.getObject(bucket, key);
    InputStream inputStream = new BufferedInputStream(s3Object.getObjectContent());
    inputData.setInputStream(inputStream);
    ObjectMetadata meta = s3Object.getObjectMetadata();
    resource.setContentLength(meta.getContentLength());
    resource.setLastModified(meta.getLastModified().getTime());
  }

  @Override
  public void fillOutputData(final OutputData outputData) throws TransferFailedException {
    Resource resource = outputData.getResource();
    String bucket = getBucket();
    String key = getKey(resource);

    // connect input stream to transfer manager to output data's output stream
    PipedInputStream inputStream = new PipedInputStream(AbstractWagon.DEFAULT_BUFFER_SIZE);
    PipedOutputStream outputStream;
    try {
      outputStream = new PipedOutputStream(inputStream);
    } catch (IOException ioe) {
      throw new TransferFailedException("Unable to put to S3 bucket=" + bucket + " key=" + key,
          ioe);
    }

    ObjectMetadata meta = new ObjectMetadata();
    meta.setContentLength(resource.getContentLength());

    PutObjectRequest request = new PutObjectRequest(bucket, key, inputStream, meta);
    Upload upload = transferManager.upload(request);
    uploads.put(upload, outputStream);

    outputData.setOutputStream(outputStream);
  }

  @Override
  protected void openConnectionInternal() {
  }

  /**
   * Waits for all {@code Upload} to finish and then closes all {@code OutputStream}.
   */
  @Override
  public void closeConnection() throws ConnectionException {
    Iterator iterator = uploads.keySet().iterator();
    while (iterator.hasNext()) {
      // wait for upload to finish
      Upload upload = iterator.next();
      try {
        upload.waitForUploadResult();
      } catch (InterruptedException ie) {
        LOGGER.error("Failed to wait for upload to S3 to complete", ie);
        Thread.currentThread().interrupt();
      }

      // close output stream
      OutputStream outputStream = uploads.get(upload);
      try {
        outputStream.close();
      } catch (IOException ioe) {
        throw new ConnectionException(ioe.getMessage(), ioe);
      }
      iterator.remove();
    }
  }

  private String getBucket() {
    return getRepository().getHost();
  }

  private String getKey(final Resource resource) {
    return getRepository().getBasedir().substring(1) + "/" + resource.getName();
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy