blob: 769375c3065f4afca3354b2ee55a2b096f183d96 [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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
* <p>
* http://www.apache.org/licenses/LICENSE-2.0
* <p>
* 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.apache.hadoop.ozone.s3.signature;
import java.io.UnsupportedEncodingException;
import java.time.ZonedDateTime;
import java.util.Map;
import org.apache.hadoop.ozone.s3.signature.SignatureInfo.Version;
import com.google.common.annotations.VisibleForTesting;
import static java.time.temporal.ChronoUnit.SECONDS;
import static org.apache.hadoop.ozone.s3.util.S3Utils.urlDecode;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Parser for getting auth info from query parameters.
* <p>
* See: https://docs.aws.amazon
* .com/AmazonS3/latest/API/sigv4-query-string-auth.html
*/
public class AuthorizationV4QueryParser implements SignatureParser {
private static final Logger LOG =
LoggerFactory.getLogger(AuthorizationV4QueryParser.class);
private final Map<String, String> queryParameters;
public AuthorizationV4QueryParser(
Map<String, String> queryParameters
) {
this.queryParameters = queryParameters;
}
@Override
public SignatureInfo parseSignature() throws MalformedResourceException {
if (!queryParameters.containsKey("X-Amz-Signature")) {
return null;
}
validateDateAndExpires();
final String rawCredential = queryParameters.get("X-Amz-Credential");
Credential credential =
null;
try {
credential = new Credential(urlDecode(rawCredential));
} catch (UnsupportedEncodingException e) {
throw new IllegalArgumentException(
"X-Amz-Credential is not proper URL encoded");
}
return new SignatureInfo(
Version.V4,
credential.getDate(),
queryParameters.get("X-Amz-Date"),
credential.getAccessKeyID(),
queryParameters.get("X-Amz-Signature"),
queryParameters.get("X-Amz-SignedHeaders"),
credential.createScope(),
queryParameters.get("X-Amz-Algorithm"),
false
);
}
@VisibleForTesting
protected void validateDateAndExpires() {
final String dateString = queryParameters.get("X-Amz-Date");
final String expiresString = queryParameters.get("X-Amz-Expires");
if (expiresString != null && expiresString.length() > 0) {
final Long expires = Long.valueOf(expiresString);
if (ZonedDateTime.parse(dateString, StringToSignProducer.TIME_FORMATTER)
.plus(expires, SECONDS).isBefore(ZonedDateTime.now())) {
throw new IllegalArgumentException("Pre-signed S3 url is expired");
}
}
}
}