HADOOP-18498. ABFS: Remove unwanted ? prefix from SAS Tokens (#5136)
This commit parses SAS Tokens and removes the unwanted prefix of '?' from them, if present. At present, SAS Tokens are provided to the driver through customer implementations of the SASTokenProvider interface. The SAS token providers should not assume that the token will be the first query parameter in the URIs that communicate with the backend. However, it was observed that certain public interfaces provided by Storage to generate SAS can include the '?' as the first character of the SAS Token, which would ideally be the case when it is the first query parameter. Thus, tokens that contain this prefix will lead to an error in the driver due to a clash of query parameters. To avoid failures for use of such SAS tokens, after receiving the SAS Token from the provider, the code checks for whether any ? prefix is present or not. If yes, it is removed before further usage of the token. This way, users would not have to manually remove the prefix before passing it on as a configuration. Contributed by Sree Bhattacharya
This commit is contained in:
parent
e09e81abe4
commit
1a7acc403b
|
@ -1130,6 +1130,10 @@ public class AbfsClient implements Closeable {
|
||||||
sasToken = cachedSasToken;
|
sasToken = cachedSasToken;
|
||||||
LOG.trace("Using cached SAS token.");
|
LOG.trace("Using cached SAS token.");
|
||||||
}
|
}
|
||||||
|
// if SAS Token contains a prefix of ?, it should be removed
|
||||||
|
if (sasToken.charAt(0) == '?') {
|
||||||
|
sasToken = sasToken.substring(1);
|
||||||
|
}
|
||||||
queryBuilder.setSASToken(sasToken);
|
queryBuilder.setSASToken(sasToken);
|
||||||
LOG.trace("SAS token fetch complete for {} on {}", operation, path);
|
LOG.trace("SAS token fetch complete for {} on {}", operation, path);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
|
|
|
@ -479,4 +479,17 @@ public class ITestAzureBlobFileSystemDelegationSAS extends AbstractAbfsIntegrati
|
||||||
"r--r-----",
|
"r--r-----",
|
||||||
fileStatus.getPermission().toString());
|
fileStatus.getPermission().toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSASQuesMarkPrefix() throws Exception {
|
||||||
|
AbfsConfiguration testConfig = this.getConfiguration();
|
||||||
|
// the SAS Token Provider is changed
|
||||||
|
testConfig.set(FS_AZURE_SAS_TOKEN_PROVIDER_TYPE, "org.apache.hadoop.fs.azurebfs.extensions.MockWithPrefixSASTokenProvider");
|
||||||
|
|
||||||
|
AzureBlobFileSystem testFs = (AzureBlobFileSystem) FileSystem.newInstance(getRawConfiguration());
|
||||||
|
Path testFile = new Path("/testSASPrefixQuesMark");
|
||||||
|
|
||||||
|
// the creation of this filesystem should work correctly even when a SAS Token is generated with a ? prefix
|
||||||
|
testFs.create(testFile).close();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
/**
|
||||||
|
* 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.extensions;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class MockWithPrefixSASTokenProvider extends MockSASTokenProvider {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function to return an already generated SAS Token with a '?' prefix
|
||||||
|
* @param accountName the name of the storage account.
|
||||||
|
* @param fileSystem the name of the fileSystem.
|
||||||
|
* @param path the file or directory path.
|
||||||
|
* @param operation the operation to be performed on the path.
|
||||||
|
* @return
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String getSASToken(String accountName, String fileSystem, String path,
|
||||||
|
String operation) throws IOException {
|
||||||
|
String token = super.getSASToken(accountName, fileSystem, path, operation);
|
||||||
|
return "?" + token;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue