Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[7.1.0] Add support for arbitrary headers to rctx.download[_and_extract] #20979

Merged
merged 1 commit into from
Jan 22, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ public void setDelegate(@Nullable Downloader delegate) {
@Override
public void download(
List<URL> urls,
Map<String, List<String>> headers,
Credentials credentials,
Optional<Checksum> checksum,
String canonicalId,
Expand All @@ -60,6 +61,14 @@ public void download(
downloader = delegate;
}
downloader.download(
urls, credentials, checksum, canonicalId, destination, eventHandler, clientEnv, type);
urls,
headers,
credentials,
checksum,
canonicalId,
destination,
eventHandler,
clientEnv,
type);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,7 @@ public void setCredentialFactory(CredentialFactory credentialFactory) {

public Future<Path> startDownload(
List<URL> originalUrls,
Map<String, List<String>> headers,
Map<URI, Map<String, List<String>>> authHeaders,
Optional<Checksum> checksum,
String canonicalId,
Expand All @@ -129,6 +130,7 @@ public Future<Path> startDownload(
try (SilentCloseable c = Profiler.instance().profile("fetching: " + context)) {
return downloadInExecutor(
originalUrls,
headers,
authHeaders,
checksum,
canonicalId,
Expand All @@ -154,6 +156,7 @@ public Path finalizeDownload(Future<Path> download) throws IOException, Interrup

public Path download(
List<URL> originalUrls,
Map<String, List<String>> headers,
Map<URI, Map<String, List<String>>> authHeaders,
Optional<Checksum> checksum,
String canonicalId,
Expand All @@ -166,6 +169,7 @@ public Path download(
Future<Path> future =
startDownload(
originalUrls,
headers,
authHeaders,
checksum,
canonicalId,
Expand Down Expand Up @@ -197,6 +201,7 @@ public Path download(
*/
private Path downloadInExecutor(
List<URL> originalUrls,
Map<String, List<String>> headers,
Map<URI, Map<String, List<String>>> authHeaders,
Optional<Checksum> checksum,
String canonicalId,
Expand Down Expand Up @@ -339,6 +344,7 @@ private Path downloadInExecutor(
try {
downloader.download(
rewrittenUrls,
headers,
credentialFactory.create(rewrittenAuthHeaders),
checksum,
canonicalId,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ public interface Downloader {
*/
void download(
List<URL> urls,
Map<String, List<String>> headers,
Credentials credentials,
Optional<Checksum> checksum,
String canonicalId,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ final class HttpConnectorMultiplexer {
}

public HttpStream connect(URL url, Optional<Checksum> checksum) throws IOException {
return connect(url, checksum, StaticCredentials.EMPTY, Optional.empty());
return connect(url, checksum, ImmutableMap.of(), StaticCredentials.EMPTY, Optional.empty());
}

/**
Expand All @@ -96,14 +96,23 @@ public HttpStream connect(URL url, Optional<Checksum> checksum) throws IOExcepti
* @throws IllegalArgumentException if {@code urls} is empty or has an unsupported protocol
*/
public HttpStream connect(
URL url, Optional<Checksum> checksum, Credentials credentials, Optional<String> type)
URL url,
Optional<Checksum> checksum,
Map<String, List<String>> headers,
Credentials credentials,
Optional<String> type)
throws IOException {
Preconditions.checkArgument(HttpUtils.isUrlSupportedByDownloader(url));
if (Thread.interrupted()) {
throw new InterruptedIOException();
}
ImmutableMap.Builder<String, List<String>> baseHeaders = new ImmutableMap.Builder<>();
baseHeaders.putAll(headers);
// REQUEST_HEADERS should not be overridable by user provided headers
baseHeaders.putAll(REQUEST_HEADERS);

Function<URL, ImmutableMap<String, List<String>>> headerFunction =
getHeaderFunction(REQUEST_HEADERS, credentials);
getHeaderFunction(baseHeaders.buildKeepingLast(), credentials);
URLConnection connection = connector.connect(url, headerFunction);
return httpStreamFactory.create(
connection,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@

import com.google.auth.Credentials;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Iterables;
import com.google.common.io.ByteStreams;
import com.google.devtools.build.lib.buildeventstream.FetchEvent;
Expand Down Expand Up @@ -75,6 +76,7 @@ public void setMaxRetryTimeout(Duration maxRetryTimeout) {
@Override
public void download(
List<URL> urls,
Map<String, List<String>> headers,
Credentials credentials,
Optional<Checksum> checksum,
String canonicalId,
Expand All @@ -94,7 +96,7 @@ public void download(
for (URL url : urls) {
SEMAPHORE.acquire();

try (HttpStream payload = multiplexer.connect(url, checksum, credentials, type);
try (HttpStream payload = multiplexer.connect(url, checksum, headers, credentials, type);
OutputStream out = destination.getOutputStream()) {
try {
ByteStreams.copy(payload, out);
Expand Down Expand Up @@ -153,7 +155,8 @@ public byte[] downloadAndReadOneUrl(
ByteArrayOutputStream out = new ByteArrayOutputStream();
SEMAPHORE.acquire();
try (HttpStream payload =
multiplexer.connect(url, Optional.empty(), credentials, Optional.empty())) {
multiplexer.connect(
url, Optional.empty(), ImmutableMap.of(), credentials, Optional.empty())) {
ByteStreams.copy(payload, out);
} catch (SocketTimeoutException e) {
// SocketTimeoutExceptions are InterruptedIOExceptions; however they do not signify
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,32 @@ private static ImmutableMap<URI, Map<String, List<String>>> getAuthHeaders(
return res;
}

private static ImmutableMap<String, List<String>> getHeaderContents(Dict<?, ?> x, String what)
throws EvalException {
Dict<String, Object> headersUnchecked =
(Dict<String, Object>) Dict.cast(x, String.class, Object.class, what);
ImmutableMap.Builder<String, List<String>> headers = new ImmutableMap.Builder<>();

for (Map.Entry<String, Object> entry : headersUnchecked.entrySet()) {
ImmutableList<String> headerValue;
Object valueUnchecked = entry.getValue();
if (valueUnchecked instanceof Sequence) {
headerValue =
Sequence.cast(valueUnchecked, String.class, "header values").getImmutableList();
} else if (valueUnchecked instanceof String) {
headerValue = ImmutableList.of(valueUnchecked.toString());
} else {
throw new EvalException(
String.format(
"%s argument must be a dict whose keys are string and whose values are either"
+ " string or sequence of string",
what));
}
headers.put(entry.getKey(), headerValue);
}
return headers.buildOrThrow();
}

private static ImmutableList<String> checkAllUrls(Iterable<?> urlList) throws EvalException {
ImmutableList.Builder<String> result = ImmutableList.builder();

Expand Down Expand Up @@ -587,6 +613,11 @@ private StructImpl completeDownload(PendingDownload pendingDownload)
defaultValue = "{}",
named = true,
doc = "An optional dict specifying authentication information for some of the URLs."),
@Param(
name = "headers",
defaultValue = "{}",
named = true,
doc = "An optional dict specifying http headers for all URLs."),
@Param(
name = "integrity",
defaultValue = "''",
Expand Down Expand Up @@ -617,6 +648,7 @@ public Object download(
Boolean allowFail,
String canonicalId,
Dict<?, ?> authUnchecked, // <String, Dict> expected
Dict<?, ?> headersUnchecked, // <String, List<String> | String> expected
String integrity,
Boolean block,
StarlarkThread thread)
Expand All @@ -625,6 +657,8 @@ public Object download(
ImmutableMap<URI, Map<String, List<String>>> authHeaders =
getAuthHeaders(getAuthContents(authUnchecked, "auth"));

ImmutableMap<String, List<String>> headers = getHeaderContents(headersUnchecked, "headers");

ImmutableList<URL> urls =
getUrls(
url,
Expand Down Expand Up @@ -670,6 +704,7 @@ public Object download(
Future<Path> downloadFuture =
downloadManager.startDownload(
urls,
headers,
authHeaders,
checksum,
canonicalId,
Expand Down Expand Up @@ -778,6 +813,11 @@ public Object download(
defaultValue = "{}",
named = true,
doc = "An optional dict specifying authentication information for some of the URLs."),
@Param(
name = "headers",
defaultValue = "{}",
named = true,
doc = "An optional dict specifying http headers for all URLs."),
@Param(
name = "integrity",
defaultValue = "''",
Expand Down Expand Up @@ -809,13 +849,16 @@ public StructImpl downloadAndExtract(
String stripPrefix,
Boolean allowFail,
String canonicalId,
Dict<?, ?> auth, // <String, Dict> expected
Dict<?, ?> authUnchecked, // <String, Dict> expected
Dict<?, ?> headersUnchecked, // <String, List<String> | String> expected
String integrity,
Dict<?, ?> renameFiles, // <String, String> expected
StarlarkThread thread)
throws RepositoryFunctionException, InterruptedException, EvalException {
ImmutableMap<URI, Map<String, List<String>>> authHeaders =
getAuthHeaders(getAuthContents(auth, "auth"));
getAuthHeaders(getAuthContents(authUnchecked, "auth"));

ImmutableMap<String, List<String>> headers = getHeaderContents(headersUnchecked, "headers");

ImmutableList<URL> urls =
getUrls(
Expand Down Expand Up @@ -862,6 +905,7 @@ public StructImpl downloadAndExtract(
Future<Path> pendingDownload =
downloadManager.startDownload(
urls,
headers,
authHeaders,
checksum,
canonicalId,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@ public void close() {
@Override
public void download(
List<URL> urls,
Map<String, List<String>> headers,
Credentials credentials,
Optional<Checksum> checksum,
String canonicalId,
Expand Down Expand Up @@ -154,7 +155,15 @@ public void download(
eventHandler.handle(
Event.warn("Remote Cache: " + Utils.grpcAwareErrorMessage(e, verboseFailures)));
fallbackDownloader.download(
urls, credentials, checksum, canonicalId, destination, eventHandler, clientEnv, type);
urls,
headers,
credentials,
checksum,
canonicalId,
destination,
eventHandler,
clientEnv,
type);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,7 @@ public void downloadFrom1UrlOk() throws IOException, InterruptedException {
Collections.singletonList(
new URL(String.format("http://localhost:%d/foo", server.getLocalPort()))),
Collections.emptyMap(),
Collections.emptyMap(),
Optional.empty(),
"testCanonicalId",
Optional.empty(),
Expand Down Expand Up @@ -181,6 +182,7 @@ public void downloadFrom2UrlsFirstOk() throws IOException, InterruptedException
downloadManager.download(
urls,
Collections.emptyMap(),
Collections.emptyMap(),
Optional.empty(),
"testCanonicalId",
Optional.empty(),
Expand Down Expand Up @@ -248,6 +250,7 @@ public void downloadFrom2UrlsFirstSocketTimeoutOnBodyReadSecondOk()
downloadManager.download(
urls,
Collections.emptyMap(),
Collections.emptyMap(),
Optional.empty(),
"testCanonicalId",
Optional.empty(),
Expand Down Expand Up @@ -317,6 +320,7 @@ public void downloadFrom2UrlsBothSocketTimeoutDuringBodyRead()
downloadManager.download(
urls,
Collections.emptyMap(),
Collections.emptyMap(),
Optional.empty(),
"testCanonicalId",
Optional.empty(),
Expand Down Expand Up @@ -371,6 +375,7 @@ public void downloadOneUrl_ok() throws IOException, InterruptedException {
httpDownloader.download(
Collections.singletonList(
new URL(String.format("http://localhost:%d/foo", server.getLocalPort()))),
Collections.emptyMap(),
StaticCredentials.EMPTY,
Optional.empty(),
"testCanonicalId",
Expand Down Expand Up @@ -410,6 +415,7 @@ public void downloadOneUrl_notFound() throws IOException, InterruptedException {
httpDownloader.download(
Collections.singletonList(
new URL(String.format("http://localhost:%d/foo", server.getLocalPort()))),
Collections.emptyMap(),
StaticCredentials.EMPTY,
Optional.empty(),
"testCanonicalId",
Expand Down Expand Up @@ -470,6 +476,7 @@ public void downloadTwoUrls_firstNotFoundAndSecondOk() throws IOException, Inter
Path destination = fs.getPath(workingDir.newFile().getAbsolutePath());
httpDownloader.download(
urls,
Collections.emptyMap(),
StaticCredentials.EMPTY,
Optional.empty(),
"testCanonicalId",
Expand Down Expand Up @@ -564,13 +571,14 @@ public void download_contentLengthMismatch_propagateErrorIfNotRetry() throws Exc
throw new ContentLengthMismatchException(0, data.length);
})
.when(downloader)
.download(any(), any(), any(), any(), any(), any(), any(), any());
.download(any(), any(), any(), any(), any(), any(), any(), any(), any());

assertThrows(
ContentLengthMismatchException.class,
() ->
downloadManager.download(
ImmutableList.of(new URL("http://localhost")),
Collections.emptyMap(),
ImmutableMap.of(),
Optional.empty(),
"testCanonicalId",
Expand All @@ -597,20 +605,21 @@ public void download_contentLengthMismatch_retries() throws Exception {
if (times.getAndIncrement() < 3) {
throw new ContentLengthMismatchException(0, data.length);
}
Path output = invocationOnMock.getArgument(4, Path.class);
Path output = invocationOnMock.getArgument(5, Path.class);
try (OutputStream outputStream = output.getOutputStream()) {
ByteStreams.copy(new ByteArrayInputStream(data), outputStream);
}

return null;
})
.when(downloader)
.download(any(), any(), any(), any(), any(), any(), any(), any());
.download(any(), any(), any(), any(), any(), any(), any(), any(), any());

Path result =
downloadManager.download(
ImmutableList.of(new URL("http://localhost")),
ImmutableMap.of(),
ImmutableMap.of(),
Optional.empty(),
"testCanonicalId",
Optional.empty(),
Expand Down
Loading
Loading