HADOOP-15710. ABFS checkException to map 403 to AccessDeniedException. (#2648)
When 403 is returned from an ABFS HTTP call, an AccessDeniedException is raised. The exception text is unchanged, for any application string matching on the getMessage() contents. Contributed by Steve Loughran. Change-Id: I519d50ccd657968fd8ee72d132518099de901e15
This commit is contained in:
parent
e0f8462b39
commit
99337a4dd0
|
@ -26,6 +26,7 @@ import java.io.OutputStream;
|
||||||
import java.net.HttpURLConnection;
|
import java.net.HttpURLConnection;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.net.URISyntaxException;
|
import java.net.URISyntaxException;
|
||||||
|
import java.nio.file.AccessDeniedException;
|
||||||
import java.util.Hashtable;
|
import java.util.Hashtable;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
@ -1116,7 +1117,8 @@ public class AzureBlobFileSystem extends FileSystem {
|
||||||
* @param allowedErrorCodesList varargs list of error codes.
|
* @param allowedErrorCodesList varargs list of error codes.
|
||||||
* @throws IOException if the exception error code is not on the allowed list.
|
* @throws IOException if the exception error code is not on the allowed list.
|
||||||
*/
|
*/
|
||||||
private void checkException(final Path path,
|
@VisibleForTesting
|
||||||
|
static void checkException(final Path path,
|
||||||
final AzureBlobFileSystemException exception,
|
final AzureBlobFileSystemException exception,
|
||||||
final AzureServiceErrorCode... allowedErrorCodesList) throws IOException {
|
final AzureServiceErrorCode... allowedErrorCodesList) throws IOException {
|
||||||
if (exception instanceof AbfsRestOperationException) {
|
if (exception instanceof AbfsRestOperationException) {
|
||||||
|
@ -1125,16 +1127,21 @@ public class AzureBlobFileSystem extends FileSystem {
|
||||||
if (ArrayUtils.contains(allowedErrorCodesList, ere.getErrorCode())) {
|
if (ArrayUtils.contains(allowedErrorCodesList, ere.getErrorCode())) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
int statusCode = ere.getStatusCode();
|
|
||||||
|
|
||||||
//AbfsRestOperationException.getMessage() contains full error info including path/uri.
|
//AbfsRestOperationException.getMessage() contains full error info including path/uri.
|
||||||
if (statusCode == HttpURLConnection.HTTP_NOT_FOUND) {
|
String message = ere.getMessage();
|
||||||
throw (IOException) new FileNotFoundException(ere.getMessage())
|
|
||||||
|
switch (ere.getStatusCode()) {
|
||||||
|
case HttpURLConnection.HTTP_NOT_FOUND:
|
||||||
|
throw (IOException) new FileNotFoundException(message)
|
||||||
.initCause(exception);
|
.initCause(exception);
|
||||||
} else if (statusCode == HttpURLConnection.HTTP_CONFLICT) {
|
case HttpURLConnection.HTTP_CONFLICT:
|
||||||
throw (IOException) new FileAlreadyExistsException(ere.getMessage())
|
throw (IOException) new FileAlreadyExistsException(message)
|
||||||
.initCause(exception);
|
.initCause(exception);
|
||||||
} else {
|
case HttpURLConnection.HTTP_FORBIDDEN:
|
||||||
|
case HttpURLConnection.HTTP_UNAUTHORIZED:
|
||||||
|
throw (IOException) new AccessDeniedException(message)
|
||||||
|
.initCause(exception);
|
||||||
|
default:
|
||||||
throw ere;
|
throw ere;
|
||||||
}
|
}
|
||||||
} else if (exception instanceof SASTokenProviderException) {
|
} else if (exception instanceof SASTokenProviderException) {
|
||||||
|
|
|
@ -20,13 +20,12 @@ package org.apache.hadoop.fs.azurebfs;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.nio.file.AccessDeniedException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
import org.apache.hadoop.fs.azurebfs.contracts.exceptions.AbfsRestOperationException;
|
|
||||||
import org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode;
|
|
||||||
import org.assertj.core.api.Assertions;
|
import org.assertj.core.api.Assertions;
|
||||||
import org.junit.Assume;
|
import org.junit.Assume;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -52,6 +51,7 @@ import org.apache.hadoop.fs.permission.FsPermission;
|
||||||
import org.apache.hadoop.security.AccessControlException;
|
import org.apache.hadoop.security.AccessControlException;
|
||||||
|
|
||||||
import static org.apache.hadoop.fs.azurebfs.constants.ConfigurationKeys.FS_AZURE_SAS_TOKEN_PROVIDER_TYPE;
|
import static org.apache.hadoop.fs.azurebfs.constants.ConfigurationKeys.FS_AZURE_SAS_TOKEN_PROVIDER_TYPE;
|
||||||
|
import static org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode.AUTHORIZATION_PERMISSION_MISS_MATCH;
|
||||||
import static org.apache.hadoop.fs.azurebfs.utils.AclTestHelpers.aclEntry;
|
import static org.apache.hadoop.fs.azurebfs.utils.AclTestHelpers.aclEntry;
|
||||||
import static org.apache.hadoop.fs.permission.AclEntryScope.ACCESS;
|
import static org.apache.hadoop.fs.permission.AclEntryScope.ACCESS;
|
||||||
import static org.apache.hadoop.fs.permission.AclEntryScope.DEFAULT;
|
import static org.apache.hadoop.fs.permission.AclEntryScope.DEFAULT;
|
||||||
|
@ -432,15 +432,12 @@ public class ITestAzureBlobFileSystemDelegationSAS extends AbstractAbfsIntegrati
|
||||||
rootStatus.getOwner());
|
rootStatus.getOwner());
|
||||||
|
|
||||||
// Attempt to set permission without being the owner.
|
// Attempt to set permission without being the owner.
|
||||||
try {
|
intercept(AccessDeniedException.class,
|
||||||
|
AUTHORIZATION_PERMISSION_MISS_MATCH.getErrorCode(), () -> {
|
||||||
fs.setPermission(rootPath, new FsPermission(FsAction.ALL,
|
fs.setPermission(rootPath, new FsPermission(FsAction.ALL,
|
||||||
FsAction.READ_EXECUTE, FsAction.EXECUTE));
|
FsAction.READ_EXECUTE, FsAction.EXECUTE));
|
||||||
assertTrue("Set permission should fail because saoid is not the owner.", false);
|
return "Set permission should fail because saoid is not the owner.";
|
||||||
} catch (AbfsRestOperationException ex) {
|
});
|
||||||
// Should fail with permission mismatch
|
|
||||||
assertEquals(AzureServiceErrorCode.AUTHORIZATION_PERMISSION_MISS_MATCH,
|
|
||||||
ex.getErrorCode());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Attempt to set permission as the owner.
|
// Attempt to set permission as the owner.
|
||||||
fs.setOwner(rootPath, MockDelegationSASTokenProvider.TEST_OWNER, null);
|
fs.setOwner(rootPath, MockDelegationSASTokenProvider.TEST_OWNER, null);
|
||||||
|
|
|
@ -0,0 +1,92 @@
|
||||||
|
/*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* 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.apache.hadoop.fs.azurebfs;
|
||||||
|
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.net.HttpURLConnection;
|
||||||
|
import java.nio.file.AccessDeniedException;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.apache.hadoop.fs.FileAlreadyExistsException;
|
||||||
|
import org.apache.hadoop.fs.Path;
|
||||||
|
import org.apache.hadoop.fs.azurebfs.contracts.exceptions.AbfsRestOperationException;
|
||||||
|
import org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode;
|
||||||
|
import org.apache.hadoop.test.AbstractHadoopTestBase;
|
||||||
|
|
||||||
|
import static org.apache.hadoop.fs.azurebfs.AzureBlobFileSystem.checkException;
|
||||||
|
import static org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode.AUTHORIZATION_PERMISSION_MISS_MATCH;
|
||||||
|
import static org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode.PATH_ALREADY_EXISTS;
|
||||||
|
import static org.apache.hadoop.fs.azurebfs.contracts.services.AzureServiceErrorCode.PATH_NOT_FOUND;
|
||||||
|
import static org.apache.hadoop.test.LambdaTestUtils.intercept;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test suite to verify exception conversion, filtering etc.
|
||||||
|
*/
|
||||||
|
public class TestAbfsErrorTranslation extends AbstractHadoopTestBase {
|
||||||
|
|
||||||
|
public static final Path PATH = new Path("abfs//store/path");
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConvert403ToAccessDenied() throws Throwable {
|
||||||
|
assertTranslated(HttpURLConnection.HTTP_FORBIDDEN,
|
||||||
|
AUTHORIZATION_PERMISSION_MISS_MATCH,
|
||||||
|
AccessDeniedException.class,
|
||||||
|
AUTHORIZATION_PERMISSION_MISS_MATCH.getErrorCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConvert404ToFNFE() throws Throwable {
|
||||||
|
assertTranslated(HttpURLConnection.HTTP_NOT_FOUND,
|
||||||
|
PATH_NOT_FOUND,
|
||||||
|
FileNotFoundException.class,
|
||||||
|
PATH_NOT_FOUND.getErrorCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConvert409ToFileAlreadyExistsException() throws Throwable {
|
||||||
|
assertTranslated(HttpURLConnection.HTTP_CONFLICT,
|
||||||
|
PATH_ALREADY_EXISTS,
|
||||||
|
FileAlreadyExistsException.class,
|
||||||
|
PATH_ALREADY_EXISTS.getErrorCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assert that for a given status code and AzureServiceErrorCode, a specific
|
||||||
|
* exception class is raised.
|
||||||
|
* @param <E> type of exception
|
||||||
|
* @param httpStatus http status code
|
||||||
|
* @param exitCode AzureServiceErrorCode
|
||||||
|
* @param clazz class of raised exception
|
||||||
|
* @param expectedText text to expect in the exception
|
||||||
|
* @throws Exception any other exception than the one expected
|
||||||
|
*/
|
||||||
|
private <E extends Throwable> void assertTranslated(
|
||||||
|
int httpStatus, AzureServiceErrorCode exitCode,
|
||||||
|
Class<E> clazz, String expectedText) throws Exception {
|
||||||
|
AbfsRestOperationException ex =
|
||||||
|
new AbfsRestOperationException(httpStatus, exitCode.getErrorCode(),
|
||||||
|
"", null);
|
||||||
|
intercept(clazz, expectedText, () -> {
|
||||||
|
checkException(PATH, ex);
|
||||||
|
return "expected exception translation from " + ex;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue