blob: 3df0b5270f40a860cb28f91f42227492b4908a24 [file] [log] [blame]
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ash/ambient/ambient_photo_cache.h"
#include <fstream>
#include <iostream>
#include "ash/ambient/ambient_access_token_controller.h"
#include "ash/ambient/ambient_constants.h"
#include "ash/ambient/ambient_photo_cache_settings.h"
#include "ash/constants/ash_features.h"
#include "ash/public/cpp/ambient/ambient_client.h"
#include "ash/public/cpp/ambient/proto/photo_cache_entry.pb.h"
#include "base/files/file_util.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ref.h"
#include "base/memory/weak_ptr.h"
#include "base/no_destructor.h"
#include "base/notreached.h"
#include "base/path_service.h"
#include "base/strings/string_number_conversions.h"
#include "base/system/sys_info.h"
#include "base/task/sequenced_task_runner.h"
#include "services/network/public/cpp/resource_request.h"
#include "services/network/public/cpp/shared_url_loader_factory.h"
#include "services/network/public/cpp/simple_url_loader.h"
#include "services/network/public/mojom/url_response_head.mojom.h"
namespace ash {
namespace {
constexpr net::NetworkTrafficAnnotationTag kAmbientPhotoCacheNetworkTag =
net::DefineNetworkTrafficAnnotation("ambient_photo_cache", R"(
semantics {
sender: "Ambient photo"
description:
"Get ambient photo from url to store limited number of photos in "
"the device cache. This is used to show the screensaver when the "
"user is idle. The url can be Backdrop service to provide pictures"
" from internal gallery, weather/time photos served by Google, or "
"user selected album from Google photos."
trigger:
"Triggered by a photo refresh timer, after the device has been "
"idle and the battery is charging."
data: "None."
destination: GOOGLE_OWNED_SERVICE
}
policy {
cookies_allowed: NO
setting:
"This feature is off by default and can be overridden by users."
policy_exception_justification:
"This feature is set by user settings.ambient_mode.enabled pref. "
"The user setting is per device and cannot be overriden by admin."
})");
// Helper function to extract response code from |SimpleURLLoader|.
int GetResponseCode(network::SimpleURLLoader* simple_loader) {
if (simple_loader->ResponseInfo() && simple_loader->ResponseInfo()->headers)
return simple_loader->ResponseInfo()->headers->response_code();
else
return -1;
}
std::unique_ptr<network::SimpleURLLoader> CreateSimpleURLLoader(
const std::string& url,
const std::string& token) {
auto resource_request = std::make_unique<network::ResourceRequest>();
resource_request->url = GURL(url);
resource_request->method = "GET";
resource_request->credentials_mode = network::mojom::CredentialsMode::kOmit;
if (token.empty())
DVLOG(2) << "Failed to fetch access token";
else
resource_request->headers.SetHeader("Authorization", "Bearer " + token);
return network::SimpleURLLoader::Create(std::move(resource_request),
kAmbientPhotoCacheNetworkTag);
}
bool CreateDirIfNotExists(const base::FilePath& path) {
return base::DirectoryExists(path) || base::CreateDirectory(path);
}
bool WriteOrDeleteFile(const base::FilePath& path,
const ambient::PhotoCacheEntry& cache_entry) {
// If the primary photo is empty, the same as the related photo.
if (!cache_entry.has_primary_photo() ||
cache_entry.primary_photo().image().empty()) {
base::DeleteFile(path);
return false;
}
if (!CreateDirIfNotExists(path.DirName())) {
LOG(ERROR) << "Cannot create ambient mode directory.";
return false;
}
if (base::SysInfo::AmountOfFreeDiskSpace(path.DirName()) <
kMaxReservedAvailableDiskSpaceByte) {
LOG(ERROR) << "Not enough disk space left.";
return false;
}
// Create a temp file.
base::FilePath temp_file;
if (!base::CreateTemporaryFileInDir(path.DirName(), &temp_file)) {
LOG(ERROR) << "Cannot create a temporary file.";
return false;
}
// Write to the tmp file.
const char* path_str = temp_file.value().c_str();
std::fstream output(path_str,
std::ios::out | std::ios::trunc | std::ios::binary);
if (!cache_entry.SerializeToOstream(&output)) {
LOG(ERROR) << "Cannot write the temporary file.";
base::DeleteFile(temp_file);
return false;
}
// Replace the current file with the temp file.
if (!base::ReplaceFile(temp_file, path, /*error=*/nullptr)) {
LOG(ERROR) << "Cannot replace the temporary file.";
base::DeleteFile(temp_file);
base::DeleteFile(path);
return false;
}
return true;
}
const base::FilePath& GetCacheRootDir(ambient_photo_cache::Store store) {
switch (store) {
case ambient_photo_cache::Store::kPrimary:
return GetAmbientPhotoCacheRootDir();
case ambient_photo_cache::Store::kBackup:
return GetAmbientBackupPhotoCacheRootDir();
}
NOTREACHED_NORETURN() << "Unknown cache store: " << static_cast<int>(store);
}
base::FilePath GetCachePath(int cache_index, const base::FilePath& root_path) {
return root_path.Append(base::NumberToString(cache_index) + kPhotoCacheExt);
}
scoped_refptr<base::SequencedTaskRunner>& GetFileTaskRunner() {
static base::NoDestructor<scoped_refptr<base::SequencedTaskRunner>>
kFileTaskRunner;
return *kFileTaskRunner;
}
void OnUrlDownloaded(
base::OnceCallback<void(std::string&&)> callback,
std::unique_ptr<network::SimpleURLLoader> simple_loader,
scoped_refptr<network::SharedURLLoaderFactory> loader_factory,
std::unique_ptr<std::string> response_body) {
if (simple_loader->NetError() == net::OK && response_body) {
std::move(callback).Run(std::move(*response_body));
return;
}
LOG(ERROR) << "Downloading to string failed with error code: "
<< GetResponseCode(simple_loader.get()) << " with network error "
<< simple_loader->NetError();
std::move(callback).Run(std::string());
}
void OnUrlDownloadedToTempFile(
base::OnceCallback<void(base::FilePath)> callback,
std::unique_ptr<network::SimpleURLLoader> simple_loader,
scoped_refptr<network::SharedURLLoaderFactory> loader_factory,
base::FilePath temp_path) {
CHECK(callback);
if (simple_loader->NetError() != net::OK || temp_path.empty()) {
LOG(ERROR) << "Downloading to file failed with error code: "
<< GetResponseCode(simple_loader.get()) << " with network error "
<< simple_loader->NetError();
if (!temp_path.empty()) {
// Clean up temporary file.
GetFileTaskRunner()->PostTask(
FROM_HERE,
base::BindOnce(
[](const base::FilePath& path) { base::DeleteFile(path); },
temp_path));
}
std::move(callback).Run(base::FilePath());
return;
}
std::move(callback).Run(std::move(temp_path));
}
void DownloadPhotoInternal(
const std::string& url,
scoped_refptr<network::SharedURLLoaderFactory> loader_factory,
base::OnceCallback<void(std::string&&)> callback,
const std::string& gaia_id,
const std::string& access_token) {
std::unique_ptr<network::SimpleURLLoader> simple_loader =
CreateSimpleURLLoader(url, access_token);
auto* loader_ptr = simple_loader.get();
auto* loader_factory_ptr = loader_factory.get();
loader_ptr->DownloadToString(
loader_factory_ptr,
base::BindOnce(&OnUrlDownloaded, std::move(callback),
std::move(simple_loader), std::move(loader_factory)),
kMaxImageSizeInBytes);
}
void DownloadPhotoToTempFileInternal(
const std::string& url,
scoped_refptr<network::SharedURLLoaderFactory> loader_factory,
base::OnceCallback<void(base::FilePath)> callback,
const std::string& gaia_id,
const std::string& access_token) {
std::unique_ptr<network::SimpleURLLoader> simple_loader =
CreateSimpleURLLoader(url, access_token);
auto* loader_ptr = simple_loader.get();
auto* loader_factory_ptr = loader_factory.get();
loader_ptr->DownloadToTempFile(
loader_factory_ptr,
base::BindOnce(&OnUrlDownloadedToTempFile, std::move(callback),
std::move(simple_loader), std::move(loader_factory)));
}
} // namespace
namespace ambient_photo_cache {
void SetFileTaskRunner(scoped_refptr<base::SequencedTaskRunner> task_runner) {
GetFileTaskRunner() = std::move(task_runner);
}
void DownloadPhoto(const std::string& url,
AmbientAccessTokenController& access_token_controller,
base::OnceCallback<void(std::string&&)> callback) {
access_token_controller.RequestAccessToken(base::BindOnce(
&DownloadPhotoInternal, url, AmbientClient::Get()->GetURLLoaderFactory(),
std::move(callback)));
}
void DownloadPhotoToTempFile(
const std::string& url,
AmbientAccessTokenController& access_token_controller,
base::OnceCallback<void(base::FilePath)> callback) {
access_token_controller.RequestAccessToken(base::BindOnce(
&DownloadPhotoToTempFileInternal, url,
AmbientClient::Get()->GetURLLoaderFactory(), std::move(callback)));
}
void WritePhotoCache(Store store,
int cache_index,
const ambient::PhotoCacheEntry& cache_entry,
base::OnceClosure callback) {
DCHECK_LT(cache_index, kMaxNumberOfCachedImages);
GetFileTaskRunner()->PostTaskAndReply(
FROM_HERE,
base::BindOnce(
[](int cache_index, const base::FilePath& root_path,
const ambient::PhotoCacheEntry& cache_entry) {
auto cache_path = GetCachePath(cache_index, root_path);
WriteOrDeleteFile(cache_path, cache_entry);
},
cache_index, GetCacheRootDir(store), cache_entry),
std::move(callback));
}
void ReadPhotoCache(
Store store,
int cache_index,
base::OnceCallback<void(::ambient::PhotoCacheEntry)> callback) {
GetFileTaskRunner()->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(
[](int cache_index, const base::FilePath& root_path) {
auto cache_path = GetCachePath(cache_index, root_path);
// Read the existing cache.
const char* path_str = cache_path.value().c_str();
std::fstream input(path_str, std::ios::in | std::ios::binary);
ambient::PhotoCacheEntry cache_entry;
if (!input || !cache_entry.ParseFromIstream(&input)) {
LOG(ERROR) << "Unable to read photo cache";
cache_entry = ::ambient::PhotoCacheEntry();
base::DeleteFile(cache_path);
}
return cache_entry;
},
cache_index, GetCacheRootDir(store)),
std::move(callback));
}
void Clear(Store store) {
GetFileTaskRunner()->PostTask(FROM_HERE,
base::BindOnce(
[](const base::FilePath& file_path) {
base::DeletePathRecursively(file_path);
},
GetCacheRootDir(store)));
}
} // namespace ambient_photo_cache
} // namespace ash