blob: dbc0dfe79347a2042c04d6ad06b008df0c1a9380 [file] [log] [blame]
// Copyright 2025 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/optimization_guide/content/browser/page_context_eligibility.h"
#include "base/compiler_specific.h"
#include "base/memory/ptr_util.h"
#include "base/no_destructor.h"
#include "components/optimization_guide/content/browser/page_context_eligibility_api.h"
#include "components/optimization_guide/core/optimization_guide_library_holder.h"
namespace optimization_guide {
namespace {
base::Lock& GetIsInitializedLock() {
static base::NoDestructor<base::Lock> singleton_initialized_lock;
return *singleton_initialized_lock.get();
}
// Tracks whether the `page_context_eligibility` singleton has been initialized
// yet. This is tracked separately so that it's possible to get this bool
// without triggering the initialization (which may block).
//
// This variable is protected by the lock returned by `GetIsInitializedLock()`.
// But since global initializers are banned, that lock must also be a
// function-scoped (lazily-initialized) static variable, so we cannot use thread
// annotations to refer to it here.
bool g_singleton_initialized = false;
} // namespace
PageContextEligibility::PageContextEligibility(
const PageContextEligibilityAPI* api)
: api_(api) {}
PageContextEligibility::~PageContextEligibility() = default;
// static
DISABLE_CFI_DLSYM
PageContextEligibility* PageContextEligibility::Get() {
static base::NoDestructor<std::unique_ptr<PageContextEligibility>>
page_context_eligibility{Create()};
return page_context_eligibility->get();
}
// static
bool PageContextEligibility::IsInitialized() {
base::AutoLock auto_lock(GetIsInitializedLock());
return g_singleton_initialized;
}
// static
DISABLE_CFI_DLSYM
std::unique_ptr<PageContextEligibility> PageContextEligibility::Create() {
// TODO(crbug.com/414828945): Move this creation out of this file if multiple
// use cases for it in browser.
static base::NoDestructor<std::unique_ptr<OptimizationGuideLibraryHolder>>
holder{OptimizationGuideLibraryHolder::Create()};
base::AutoLock auto_lock(GetIsInitializedLock());
g_singleton_initialized = true;
// Pointer will be null if the library was not created.
OptimizationGuideLibraryHolder* holder_ptr = holder->get();
if (!holder_ptr) {
return {};
}
auto get_api = reinterpret_cast<PageContextEligibilityAPIGetter>(
holder_ptr->GetFunctionPointer("GetPageContextEligibilityAPI"));
if (!get_api) {
return {};
}
const PageContextEligibilityAPI* api = get_api();
if (!api) {
return {};
}
return base::WrapUnique(new PageContextEligibility(api));
}
std::vector<FrameMetadata> GetFrameMetadataFromPageContent(
const AIPageContentResult& result) {
std::vector<FrameMetadata> frame_metadata_structs;
const auto& page_metadata = result.metadata;
frame_metadata_structs.reserve(page_metadata->frame_metadata.size());
for (auto& frame_metadata_mojom : page_metadata->frame_metadata) {
std::vector<MetaTag> meta_tags;
meta_tags.reserve(frame_metadata_mojom->meta_tags.size());
for (auto& tag : frame_metadata_mojom->meta_tags) {
MetaTag meta_tag(tag->name, tag->content);
meta_tags.push_back(std::move(meta_tag));
}
FrameMetadata metadata(frame_metadata_mojom->url.host(),
frame_metadata_mojom->url.path(),
std::move(meta_tags));
frame_metadata_structs.push_back(std::move(metadata));
}
return frame_metadata_structs;
}
DISABLE_CFI_DLSYM
bool IsPageContextEligible(
const std::string& host,
const std::string& path,
const std::vector<optimization_guide::FrameMetadata>& frame_metadata,
const PageContextEligibility* api_holder) {
// TODO(crbug.com/421932889): `api_holder` should not be provided by caller
// and instead be retrieved as part of this function call.
if (!api_holder) {
return true;
}
return api_holder->api().IsPageContextEligible(host, path,
std::move(frame_metadata));
}
} // namespace optimization_guide