/* * Copyright (C) 2008, Shawn O. Pearce * and other copyright owners as documented in the project's IP log. * * This program and the accompanying materials are made available * under the terms of the Eclipse Distribution License v1.0 which * accompanies this distribution, is reproduced below, and is * available at http://www.eclipse.org/org/documents/edl-v10.php * * All rights reserved. * * Redistribution and use in source and binary forms, with or * without modification, are permitted provided that the following * conditions are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * - Neither the name of the Eclipse Foundation, Inc. nor the * names of its contributors may be used to endorse or promote * products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package org.eclipse.jgit.transport; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.HttpURLConnection; import java.net.Proxy; import java.net.ProxySelector; import java.net.URL; import java.net.URLConnection; import java.security.DigestOutputStream; import java.security.InvalidKeyException; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; import java.security.spec.InvalidKeySpecException; import java.text.MessageFormat; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Collections; import java.util.Date; import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Properties; import java.util.Set; import java.util.SortedMap; import java.util.TimeZone; import java.util.TreeMap; import javax.crypto.Mac; import javax.crypto.spec.SecretKeySpec; import org.eclipse.jgit.internal.JGitText; import org.eclipse.jgit.lib.Constants; import org.eclipse.jgit.lib.NullProgressMonitor; import org.eclipse.jgit.lib.ProgressMonitor; import org.eclipse.jgit.util.Base64; import org.eclipse.jgit.util.HttpSupport; import org.eclipse.jgit.util.StringUtils; import org.eclipse.jgit.util.TemporaryBuffer; import org.xml.sax.Attributes; import org.xml.sax.InputSource; import org.xml.sax.SAXException; import org.xml.sax.XMLReader; import org.xml.sax.helpers.DefaultHandler; import org.xml.sax.helpers.XMLReaderFactory; /** * A simple HTTP REST client for the Amazon S3 service. *

* This client uses the REST API to communicate with the Amazon S3 servers and * read or write content through a bucket that the user has access to. It is a * very lightweight implementation of the S3 API and therefore does not have all * of the bells and whistles of popular client implementations. *

* Authentication is always performed using the user's AWSAccessKeyId and their * private AWSSecretAccessKey. *

* Optional client-side encryption may be enabled if requested. The format is * compatible with jets3t, * a popular Java based Amazon S3 client library. Enabling encryption can hide * sensitive data from the operators of the S3 service. */ public class AmazonS3 { private static final Set SIGNED_HEADERS; private static final String HMAC = "HmacSHA1"; //$NON-NLS-1$ private static final String X_AMZ_ACL = "x-amz-acl"; //$NON-NLS-1$ private static final String X_AMZ_META = "x-amz-meta-"; //$NON-NLS-1$ static { SIGNED_HEADERS = new HashSet(); SIGNED_HEADERS.add("content-type"); //$NON-NLS-1$ SIGNED_HEADERS.add("content-md5"); //$NON-NLS-1$ SIGNED_HEADERS.add("date"); //$NON-NLS-1$ } private static boolean isSignedHeader(final String name) { final String nameLC = StringUtils.toLowerCase(name); return SIGNED_HEADERS.contains(nameLC) || nameLC.startsWith("x-amz-"); //$NON-NLS-1$ } private static String toCleanString(final List list) { final StringBuilder s = new StringBuilder(); for (final String v : list) { if (s.length() > 0) s.append(','); s.append(v.replaceAll("\n", "").trim()); //$NON-NLS-1$ //$NON-NLS-2$ } return s.toString(); } private static String remove(final Map m, final String k) { final String r = m.remove(k); return r != null ? r : ""; //$NON-NLS-1$ } private static String httpNow() { final String tz = "GMT"; //$NON-NLS-1$ final SimpleDateFormat fmt; fmt = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss", Locale.US); //$NON-NLS-1$ fmt.setTimeZone(TimeZone.getTimeZone(tz)); return fmt.format(new Date()) + " " + tz; //$NON-NLS-1$ } private static MessageDigest newMD5() { try { return MessageDigest.getInstance("MD5"); //$NON-NLS-1$ } catch (NoSuchAlgorithmException e) { throw new RuntimeException(JGitText.get().JRELacksMD5Implementation, e); } } /** AWSAccessKeyId, public string that identifies the user's account. */ private final String publicKey; /** Decoded form of the private AWSSecretAccessKey, to sign requests. */ private final SecretKeySpec privateKey; /** Our HTTP proxy support, in case we are behind a firewall. */ private final ProxySelector proxySelector; /** ACL to apply to created objects. */ private final String acl; /** Maximum number of times to try an operation. */ private final int maxAttempts; /** Encryption algorithm, may be a null instance that provides pass-through. */ private final WalkEncryption encryption; /** Directory for locally buffered content. */ private final File tmpDir; /** S3 Bucket Domain. */ private final String domain; /** * Create a new S3 client for the supplied user information. *

* The connection properties are a subset of those supported by the popular * jets3t library. * For example: * *

	 * # AWS Access and Secret Keys (required)
	 * accesskey: <YourAWSAccessKey>
	 * secretkey: <YourAWSSecretKey>
	 *
	 * # Access Control List setting to apply to uploads, must be one of:
	 * # PRIVATE, PUBLIC_READ (defaults to PRIVATE).
	 * acl: PRIVATE
	 *
	 * # S3 Domain
	 * # AWS S3 Region Domain (defaults to s3.amazonaws.com)
	 * domain: s3.amazonaws.com
	 *
	 * # Number of times to retry after internal error from S3.
	 * httpclient.retry-max: 3
	 *
	 * # End-to-end encryption (hides content from S3 owners)
	 * password: <encryption pass-phrase>
	 * crypto.algorithm: PBEWithMD5AndDES
	 * 
* * @param props * connection properties. * */ public AmazonS3(final Properties props) { domain = props.getProperty("domain", "s3.amazonaws.com"); //$NON-NLS-1$ //$NON-NLS-2$ publicKey = props.getProperty("accesskey"); //$NON-NLS-1$ if (publicKey == null) throw new IllegalArgumentException(JGitText.get().missingAccesskey); final String secret = props.getProperty("secretkey"); //$NON-NLS-1$ if (secret == null) throw new IllegalArgumentException(JGitText.get().missingSecretkey); privateKey = new SecretKeySpec(Constants.encodeASCII(secret), HMAC); final String pacl = props.getProperty("acl", "PRIVATE"); //$NON-NLS-1$ //$NON-NLS-2$ if (StringUtils.equalsIgnoreCase("PRIVATE", pacl)) //$NON-NLS-1$ acl = "private"; //$NON-NLS-1$ else if (StringUtils.equalsIgnoreCase("PUBLIC", pacl)) //$NON-NLS-1$ acl = "public-read"; //$NON-NLS-1$ else if (StringUtils.equalsIgnoreCase("PUBLIC-READ", pacl)) //$NON-NLS-1$ acl = "public-read"; //$NON-NLS-1$ else if (StringUtils.equalsIgnoreCase("PUBLIC_READ", pacl)) //$NON-NLS-1$ acl = "public-read"; //$NON-NLS-1$ else throw new IllegalArgumentException("Invalid acl: " + pacl); //$NON-NLS-1$ try { final String cPas = props.getProperty("password"); //$NON-NLS-1$ if (cPas != null) { String cAlg = props.getProperty("crypto.algorithm"); //$NON-NLS-1$ if (cAlg == null) cAlg = "PBEWithMD5AndDES"; //$NON-NLS-1$ encryption = new WalkEncryption.ObjectEncryptionV2(cAlg, cPas); } else { encryption = WalkEncryption.NONE; } } catch (InvalidKeySpecException e) { throw new IllegalArgumentException(JGitText.get().invalidEncryption, e); } catch (NoSuchAlgorithmException e) { throw new IllegalArgumentException(JGitText.get().invalidEncryption, e); } maxAttempts = Integer.parseInt(props.getProperty( "httpclient.retry-max", "3")); //$NON-NLS-1$ //$NON-NLS-2$ proxySelector = ProxySelector.getDefault(); String tmp = props.getProperty("tmpdir"); //$NON-NLS-1$ tmpDir = tmp != null && tmp.length() > 0 ? new File(tmp) : null; } /** * Get the content of a bucket object. * * @param bucket * name of the bucket storing the object. * @param key * key of the object within its bucket. * @return connection to stream the content of the object. The request * properties of the connection may not be modified by the caller as * the request parameters have already been signed. * @throws IOException * sending the request was not possible. */ public URLConnection get(final String bucket, final String key) throws IOException { for (int curAttempt = 0; curAttempt < maxAttempts; curAttempt++) { final HttpURLConnection c = open("GET", bucket, key); //$NON-NLS-1$ authorize(c); switch (HttpSupport.response(c)) { case HttpURLConnection.HTTP_OK: encryption.validate(c, X_AMZ_META); return c; case HttpURLConnection.HTTP_NOT_FOUND: throw new FileNotFoundException(key); case HttpURLConnection.HTTP_INTERNAL_ERROR: continue; default: throw error(JGitText.get().s3ActionReading, key, c); } } throw maxAttempts(JGitText.get().s3ActionReading, key); } /** * Decrypt an input stream from {@link #get(String, String)}. * * @param u * connection previously created by {@link #get(String, String)}}. * @return stream to read plain text from. * @throws IOException * decryption could not be configured. */ public InputStream decrypt(final URLConnection u) throws IOException { return encryption.decrypt(u.getInputStream()); } /** * List the names of keys available within a bucket. *

* This method is primarily meant for obtaining a "recursive directory * listing" rooted under the specified bucket and prefix location. * * @param bucket * name of the bucket whose objects should be listed. * @param prefix * common prefix to filter the results by. Must not be null. * Supplying the empty string will list all keys in the bucket. * Supplying a non-empty string will act as though a trailing '/' * appears in prefix, even if it does not. * @return list of keys starting with prefix, after removing * prefix (or prefix + "/")from all * of them. * @throws IOException * sending the request was not possible, or the response XML * document could not be parsed properly. */ public List list(final String bucket, String prefix) throws IOException { if (prefix.length() > 0 && !prefix.endsWith("/")) //$NON-NLS-1$ prefix += "/"; //$NON-NLS-1$ final ListParser lp = new ListParser(bucket, prefix); do { lp.list(); } while (lp.truncated); return lp.entries; } /** * Delete a single object. *

* Deletion always succeeds, even if the object does not exist. * * @param bucket * name of the bucket storing the object. * @param key * key of the object within its bucket. * @throws IOException * deletion failed due to communications error. */ public void delete(final String bucket, final String key) throws IOException { for (int curAttempt = 0; curAttempt < maxAttempts; curAttempt++) { final HttpURLConnection c = open("DELETE", bucket, key); //$NON-NLS-1$ authorize(c); switch (HttpSupport.response(c)) { case HttpURLConnection.HTTP_NO_CONTENT: return; case HttpURLConnection.HTTP_INTERNAL_ERROR: continue; default: throw error(JGitText.get().s3ActionDeletion, key, c); } } throw maxAttempts(JGitText.get().s3ActionDeletion, key); } /** * Atomically create or replace a single small object. *

* This form is only suitable for smaller contents, where the caller can * reasonable fit the entire thing into memory. *

* End-to-end data integrity is assured by internally computing the MD5 * checksum of the supplied data and transmitting the checksum along with * the data itself. * * @param bucket * name of the bucket storing the object. * @param key * key of the object within its bucket. * @param data * new data content for the object. Must not be null. Zero length * array will create a zero length object. * @throws IOException * creation/updating failed due to communications error. */ public void put(final String bucket, final String key, final byte[] data) throws IOException { if (encryption != WalkEncryption.NONE) { // We have to copy to produce the cipher text anyway so use // the large object code path as it supports that behavior. // final OutputStream os = beginPut(bucket, key, null, null); os.write(data); os.close(); return; } final String md5str = Base64.encodeBytes(newMD5().digest(data)); final String lenstr = String.valueOf(data.length); for (int curAttempt = 0; curAttempt < maxAttempts; curAttempt++) { final HttpURLConnection c = open("PUT", bucket, key); //$NON-NLS-1$ c.setRequestProperty("Content-Length", lenstr); //$NON-NLS-1$ c.setRequestProperty("Content-MD5", md5str); //$NON-NLS-1$ c.setRequestProperty(X_AMZ_ACL, acl); authorize(c); c.setDoOutput(true); c.setFixedLengthStreamingMode(data.length); final OutputStream os = c.getOutputStream(); try { os.write(data); } finally { os.close(); } switch (HttpSupport.response(c)) { case HttpURLConnection.HTTP_OK: return; case HttpURLConnection.HTTP_INTERNAL_ERROR: continue; default: throw error(JGitText.get().s3ActionWriting, key, c); } } throw maxAttempts(JGitText.get().s3ActionWriting, key); } /** * Atomically create or replace a single large object. *

* Initially the returned output stream buffers data into memory, but if the * total number of written bytes starts to exceed an internal limit the data * is spooled to a temporary file on the local drive. *

* Network transmission is attempted only when close() gets * called at the end of output. Closing the returned stream can therefore * take significant time, especially if the written content is very large. *

* End-to-end data integrity is assured by internally computing the MD5 * checksum of the supplied data and transmitting the checksum along with * the data itself. * * @param bucket * name of the bucket storing the object. * @param key * key of the object within its bucket. * @param monitor * (optional) progress monitor to post upload completion to * during the stream's close method. * @param monitorTask * (optional) task name to display during the close method. * @return a stream which accepts the new data, and transmits once closed. * @throws IOException * if encryption was enabled it could not be configured. */ public OutputStream beginPut(final String bucket, final String key, final ProgressMonitor monitor, final String monitorTask) throws IOException { final MessageDigest md5 = newMD5(); final TemporaryBuffer buffer = new TemporaryBuffer.LocalFile(tmpDir) { @Override public void close() throws IOException { super.close(); try { putImpl(bucket, key, md5.digest(), this, monitor, monitorTask); } finally { destroy(); } } }; return encryption.encrypt(new DigestOutputStream(buffer, md5)); } private void putImpl(final String bucket, final String key, final byte[] csum, final TemporaryBuffer buf, ProgressMonitor monitor, String monitorTask) throws IOException { if (monitor == null) monitor = NullProgressMonitor.INSTANCE; if (monitorTask == null) monitorTask = MessageFormat.format(JGitText.get().progressMonUploading, key); final String md5str = Base64.encodeBytes(csum); final long len = buf.length(); for (int curAttempt = 0; curAttempt < maxAttempts; curAttempt++) { final HttpURLConnection c = open("PUT", bucket, key); //$NON-NLS-1$ c.setFixedLengthStreamingMode(len); c.setRequestProperty("Content-MD5", md5str); //$NON-NLS-1$ c.setRequestProperty(X_AMZ_ACL, acl); encryption.request(c, X_AMZ_META); authorize(c); c.setDoOutput(true); monitor.beginTask(monitorTask, (int) (len / 1024)); final OutputStream os = c.getOutputStream(); try { buf.writeTo(os, monitor); } finally { monitor.endTask(); os.close(); } switch (HttpSupport.response(c)) { case HttpURLConnection.HTTP_OK: return; case HttpURLConnection.HTTP_INTERNAL_ERROR: continue; default: throw error(JGitText.get().s3ActionWriting, key, c); } } throw maxAttempts(JGitText.get().s3ActionWriting, key); } private IOException error(final String action, final String key, final HttpURLConnection c) throws IOException { final IOException err = new IOException(MessageFormat.format( JGitText.get().amazonS3ActionFailed, action, key, Integer.valueOf(HttpSupport.response(c)), c.getResponseMessage())); final InputStream errorStream = c.getErrorStream(); if (errorStream == null) return err; final ByteArrayOutputStream b = new ByteArrayOutputStream(); byte[] buf = new byte[2048]; for (;;) { final int n = errorStream.read(buf); if (n < 0) break; if (n > 0) b.write(buf, 0, n); } buf = b.toByteArray(); if (buf.length > 0) err.initCause(new IOException("\n" + new String(buf))); //$NON-NLS-1$ return err; } private IOException maxAttempts(final String action, final String key) { return new IOException(MessageFormat.format( JGitText.get().amazonS3ActionFailedGivingUp, action, key, Integer.valueOf(maxAttempts))); } private HttpURLConnection open(final String method, final String bucket, final String key) throws IOException { final Map noArgs = Collections.emptyMap(); return open(method, bucket, key, noArgs); } private HttpURLConnection open(final String method, final String bucket, final String key, final Map args) throws IOException { final StringBuilder urlstr = new StringBuilder(); urlstr.append("http://"); //$NON-NLS-1$ urlstr.append(bucket); urlstr.append('.'); urlstr.append(domain); urlstr.append('/'); if (key.length() > 0) HttpSupport.encode(urlstr, key); if (!args.isEmpty()) { final Iterator> i; urlstr.append('?'); i = args.entrySet().iterator(); while (i.hasNext()) { final Map.Entry e = i.next(); urlstr.append(e.getKey()); urlstr.append('='); HttpSupport.encode(urlstr, e.getValue()); if (i.hasNext()) urlstr.append('&'); } } final URL url = new URL(urlstr.toString()); final Proxy proxy = HttpSupport.proxyFor(proxySelector, url); final HttpURLConnection c; c = (HttpURLConnection) url.openConnection(proxy); c.setRequestMethod(method); c.setRequestProperty("User-Agent", "jgit/1.0"); //$NON-NLS-1$ //$NON-NLS-2$ c.setRequestProperty("Date", httpNow()); //$NON-NLS-1$ return c; } private void authorize(final HttpURLConnection c) throws IOException { final Map> reqHdr = c.getRequestProperties(); final SortedMap sigHdr = new TreeMap(); for (final Map.Entry> entry : reqHdr.entrySet()) { final String hdr = entry.getKey(); if (isSignedHeader(hdr)) sigHdr.put(StringUtils.toLowerCase(hdr), toCleanString(entry.getValue())); } final StringBuilder s = new StringBuilder(); s.append(c.getRequestMethod()); s.append('\n'); s.append(remove(sigHdr, "content-md5")); //$NON-NLS-1$ s.append('\n'); s.append(remove(sigHdr, "content-type")); //$NON-NLS-1$ s.append('\n'); s.append(remove(sigHdr, "date")); //$NON-NLS-1$ s.append('\n'); for (final Map.Entry e : sigHdr.entrySet()) { s.append(e.getKey()); s.append(':'); s.append(e.getValue()); s.append('\n'); } final String host = c.getURL().getHost(); s.append('/'); s.append(host.substring(0, host.length() - domain.length() - 1)); s.append(c.getURL().getPath()); final String sec; try { final Mac m = Mac.getInstance(HMAC); m.init(privateKey); sec = Base64.encodeBytes(m.doFinal(s.toString().getBytes("UTF-8"))); //$NON-NLS-1$ } catch (NoSuchAlgorithmException e) { throw new IOException(MessageFormat.format(JGitText.get().noHMACsupport, HMAC, e.getMessage())); } catch (InvalidKeyException e) { throw new IOException(MessageFormat.format(JGitText.get().invalidKey, e.getMessage())); } c.setRequestProperty("Authorization", "AWS " + publicKey + ":" + sec); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ } static Properties properties(final File authFile) throws FileNotFoundException, IOException { final Properties p = new Properties(); final FileInputStream in = new FileInputStream(authFile); try { p.load(in); } finally { in.close(); } return p; } private final class ListParser extends DefaultHandler { final List entries = new ArrayList(); private final String bucket; private final String prefix; boolean truncated; private StringBuilder data; ListParser(final String bn, final String p) { bucket = bn; prefix = p; } void list() throws IOException { final Map args = new TreeMap(); if (prefix.length() > 0) args.put("prefix", prefix); //$NON-NLS-1$ if (!entries.isEmpty()) args.put("marker", prefix + entries.get(entries.size() - 1)); //$NON-NLS-1$ for (int curAttempt = 0; curAttempt < maxAttempts; curAttempt++) { final HttpURLConnection c = open("GET", bucket, "", args); //$NON-NLS-1$ //$NON-NLS-2$ authorize(c); switch (HttpSupport.response(c)) { case HttpURLConnection.HTTP_OK: truncated = false; data = null; final XMLReader xr; try { xr = XMLReaderFactory.createXMLReader(); } catch (SAXException e) { throw new IOException(JGitText.get().noXMLParserAvailable); } xr.setContentHandler(this); final InputStream in = c.getInputStream(); try { xr.parse(new InputSource(in)); } catch (SAXException parsingError) { final IOException p; p = new IOException(MessageFormat.format(JGitText.get().errorListing, prefix)); p.initCause(parsingError); throw p; } finally { in.close(); } return; case HttpURLConnection.HTTP_INTERNAL_ERROR: continue; default: throw AmazonS3.this.error("Listing", prefix, c); //$NON-NLS-1$ } } throw maxAttempts("Listing", prefix); //$NON-NLS-1$ } @Override public void startElement(final String uri, final String name, final String qName, final Attributes attributes) throws SAXException { if ("Key".equals(name) || "IsTruncated".equals(name)) //$NON-NLS-1$ //$NON-NLS-2$ data = new StringBuilder(); } @Override public void ignorableWhitespace(final char[] ch, final int s, final int n) throws SAXException { if (data != null) data.append(ch, s, n); } @Override public void characters(final char[] ch, final int s, final int n) throws SAXException { if (data != null) data.append(ch, s, n); } @Override public void endElement(final String uri, final String name, final String qName) throws SAXException { if ("Key".equals(name)) //$NON-NLS-1$ entries.add(data.toString().substring(prefix.length())); else if ("IsTruncated".equals(name)) //$NON-NLS-1$ truncated = StringUtils.equalsIgnoreCase("true", data.toString()); //$NON-NLS-1$ data = null; } } }