blob: a89077ecbe1a120b37c30c292c6488f6f20c16c8 [file] [log] [blame]
// Copyright 2016 The Bazel Authors. All rights reserved.
//
// Licensed 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 com.google.devtools.build.lib.bazel.repository.downloader;
import com.google.common.base.MoreObjects;
import com.google.common.base.Optional;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.bazel.repository.cache.RepositoryCache;
import com.google.devtools.build.lib.bazel.repository.cache.RepositoryCache.KeyType;
import com.google.devtools.build.lib.bazel.repository.cache.RepositoryCacheHitEvent;
import com.google.devtools.build.lib.events.Event;
import com.google.devtools.build.lib.events.ExtendedEventHandler;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
import java.io.IOException;
import java.io.InterruptedIOException;
import java.net.URI;
import java.net.URL;
import java.util.List;
import java.util.Map;
/**
* Bazel file downloader.
*
* <p>This class uses a {@link Downloader} to download files from external mirrors and writes them
* to disk.
*/
public class DownloadManager {
private final RepositoryCache repositoryCache;
private List<Path> distdir = ImmutableList.of();
private final Downloader downloader;
public DownloadManager(RepositoryCache repositoryCache, Downloader downloader) {
this.repositoryCache = repositoryCache;
this.downloader = downloader;
}
public void setDistdir(List<Path> distdir) {
this.distdir = ImmutableList.copyOf(distdir);
}
/**
* Downloads file to disk and returns path.
*
* <p>If the checksum and path to the repository cache is specified, attempt to load the file from
* the {@link RepositoryCache}. If it doesn't exist, proceed to download the file and load it into
* the cache prior to returning the value.
*
* @param urls list of mirror URLs with identical content
* @param checksum valid checksum which is checked, or absent to disable
* @param type extension, e.g. "tar.gz" to force on downloaded filename, or empty to not do this
* @param output destination filename if {@code type} is <i>absent</i>, otherwise output directory
* @param eventHandler CLI progress reporter
* @param clientEnv environment variables in shell issuing this command
* @param repo the name of the external repository for which the file was fetched; used only for
* reporting
* @throws IllegalArgumentException on parameter badness, which should be checked beforehand
* @throws IOException if download was attempted and ended up failing
* @throws InterruptedException if this thread is being cast into oblivion
*/
public Path download(
List<URL> urls,
Map<URI, Map<String, String>> authHeaders,
Optional<Checksum> checksum,
String canonicalId,
Optional<String> type,
Path output,
ExtendedEventHandler eventHandler,
Map<String, String> clientEnv,
String repo)
throws IOException, InterruptedException {
if (Thread.interrupted()) {
throw new InterruptedException();
}
URL mainUrl; // The "main" URL for this request
// Used for reporting only and determining the file name only.
if (urls.isEmpty()) {
if (type.isPresent() && !Strings.isNullOrEmpty(type.get())) {
mainUrl = new URL("http://nonexistent.example.org/cacheprobe." + type.get());
} else {
mainUrl = new URL("http://nonexistent.example.org/cacheprobe");
}
} else {
mainUrl = urls.get(0);
}
Path destination = getDownloadDestination(mainUrl, type, output);
ImmutableSet<String> candidateFileNames = getCandidateFileNames(mainUrl, destination);
// Is set to true if the value should be cached by the checksum value provided
boolean isCachingByProvidedChecksum = false;
if (checksum.isPresent()) {
String cacheKey = checksum.get().toString();
KeyType cacheKeyType = checksum.get().getKeyType();
try {
eventHandler.post(
new CacheProgress(mainUrl.toString(), "Checking in " + cacheKeyType + " cache"));
String currentChecksum = RepositoryCache.getChecksum(cacheKeyType, destination);
if (currentChecksum.equals(cacheKey)) {
// No need to download.
return destination;
}
} catch (IOException e) {
// Ignore error trying to hash. We'll attempt to retrieve from cache or just download again.
} finally {
eventHandler.post(new CacheProgress(mainUrl.toString()));
}
if (repositoryCache.isEnabled()) {
isCachingByProvidedChecksum = true;
try {
Path cachedDestination =
repositoryCache.get(cacheKey, destination, cacheKeyType, canonicalId);
if (cachedDestination != null) {
// Cache hit!
eventHandler.post(new RepositoryCacheHitEvent(repo, cacheKey, mainUrl));
return cachedDestination;
}
} catch (IOException e) {
// Ignore error trying to get. We'll just download again.
}
}
if (urls.isEmpty()) {
throw new IOException("Cache miss and no url specified");
}
for (Path dir : distdir) {
if (!dir.exists()) {
// This is not a warning (and probably we even should drop the message); it is
// perfectly fine to have a common rc-file pointing to a volume that is sometimes,
// but not always mounted.
eventHandler.handle(Event.info("non-existent distdir " + dir));
} else if (!dir.isDirectory()) {
eventHandler.handle(Event.warn("distdir " + dir + " is not a directory"));
} else {
for (String name : candidateFileNames) {
boolean match = false;
Path candidate = dir.getRelative(name);
try {
eventHandler.post(
new CacheProgress(
mainUrl.toString(), "Checking " + cacheKeyType + " of " + candidate));
match = RepositoryCache.getChecksum(cacheKeyType, candidate).equals(cacheKey);
} catch (IOException e) {
// Not finding anything in a distdir is a normal case, so handle it absolutely
// quietly. In fact, it is common to specify a whole list of dist dirs,
// with the assumption that only one will contain an entry.
} finally {
eventHandler.post(new CacheProgress(mainUrl.toString()));
}
if (match) {
if (isCachingByProvidedChecksum) {
try {
repositoryCache.put(cacheKey, candidate, cacheKeyType, canonicalId);
} catch (IOException e) {
eventHandler.handle(
Event.warn("Failed to copy " + candidate + " to repository cache: " + e));
}
}
FileSystemUtils.createDirectoryAndParents(destination.getParentDirectory());
FileSystemUtils.copyFile(candidate, destination);
return destination;
}
}
}
}
}
try {
downloader.download(
urls, authHeaders, checksum, canonicalId, destination, eventHandler, clientEnv);
} catch (InterruptedIOException e) {
throw new InterruptedException(e.getMessage());
}
if (isCachingByProvidedChecksum) {
repositoryCache.put(
checksum.get().toString(), destination, checksum.get().getKeyType(), canonicalId);
} else if (repositoryCache.isEnabled()) {
String newSha256 = repositoryCache.put(destination, KeyType.SHA256, canonicalId);
eventHandler.handle(Event.info("SHA256 (" + urls.get(0) + ") = " + newSha256));
}
return destination;
}
private Path getDownloadDestination(URL url, Optional<String> type, Path output) {
if (!type.isPresent()) {
return output;
}
String basename =
MoreObjects.firstNonNull(
Strings.emptyToNull(PathFragment.create(url.getPath()).getBaseName()), "temp");
if (!type.get().isEmpty()) {
String suffix = "." + type.get();
if (!basename.endsWith(suffix)) {
basename += suffix;
}
}
return output.getRelative(basename);
}
/**
* Deterimine the list of filenames to look for in the distdirs. Note that an output name may be
* specified that is unrelated to the primary URL. This happens, e.g., when the paramter output is
* specified in ctx.download.
*/
private static ImmutableSet<String> getCandidateFileNames(URL url, Path destination) {
String urlBaseName = PathFragment.create(url.getPath()).getBaseName();
if (!Strings.isNullOrEmpty(urlBaseName) && !urlBaseName.equals(destination.getBaseName())) {
return ImmutableSet.of(urlBaseName, destination.getBaseName());
} else {
return ImmutableSet.of(destination.getBaseName());
}
}
private static class CacheProgress implements ExtendedEventHandler.FetchProgress {
private final String originalUrl;
private final String progress;
private final boolean isFinished;
CacheProgress(String originalUrl, String progress) {
this.originalUrl = originalUrl;
this.progress = progress;
this.isFinished = false;
}
CacheProgress(String originalUrl) {
this.originalUrl = originalUrl;
this.progress = "";
this.isFinished = true;
}
@Override
public String getResourceIdentifier() {
return originalUrl;
}
@Override
public String getProgress() {
return progress;
}
@Override
public boolean isFinished() {
return isFinished;
}
}
}