| // Copyright 2014 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/supervised_user/supervised_user_navigation_throttle.h" |
| |
| #include "base/check_op.h" |
| #include "base/feature_list.h" |
| #include "base/functional/bind.h" |
| #include "base/location.h" |
| #include "base/memory/ptr_util.h" |
| #include "base/metrics/histogram_functions.h" |
| #include "base/notreached.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/signin/identity_manager_factory.h" |
| #include "chrome/browser/supervised_user/supervised_user_browser_utils.h" |
| #include "chrome/browser/supervised_user/supervised_user_navigation_observer.h" |
| #include "chrome/browser/supervised_user/supervised_user_service_factory.h" |
| #include "components/supervised_user/core/browser/supervised_user_interstitial.h" |
| #include "components/supervised_user/core/browser/supervised_user_preferences.h" |
| #include "components/supervised_user/core/browser/supervised_user_service.h" |
| #include "components/supervised_user/core/browser/supervised_user_url_filter.h" |
| #include "components/supervised_user/core/browser/supervised_user_utils.h" |
| #include "components/supervised_user/core/common/features.h" |
| #include "components/supervised_user/core/common/supervised_user_constants.h" |
| #include "content/public/browser/navigation_handle.h" |
| #include "ui/base/page_transition_types.h" |
| #include "url/gurl.h" |
| |
| // static |
| std::unique_ptr<SupervisedUserNavigationThrottle> |
| SupervisedUserNavigationThrottle::MaybeCreateThrottleFor( |
| content::NavigationHandle* navigation_handle) { |
| Profile* profile = Profile::FromBrowserContext( |
| navigation_handle->GetWebContents()->GetBrowserContext()); |
| CHECK(profile); |
| if (!profile->IsChild()) { |
| return nullptr; |
| } |
| |
| // Can't use std::make_unique because the constructor is private. |
| return base::WrapUnique( |
| new SupervisedUserNavigationThrottle(navigation_handle)); |
| } |
| |
| SupervisedUserNavigationThrottle::SupervisedUserNavigationThrottle( |
| content::NavigationHandle* navigation_handle) |
| : NavigationThrottle(navigation_handle), |
| url_filter_( |
| SupervisedUserServiceFactory::GetForProfile( |
| Profile::FromBrowserContext( |
| navigation_handle->GetWebContents()->GetBrowserContext())) |
| ->GetURLFilter()), |
| deferred_(false), |
| behavior_(supervised_user::FilteringBehavior::kInvalid) {} |
| |
| SupervisedUserNavigationThrottle::~SupervisedUserNavigationThrottle() {} |
| |
| void SupervisedUserNavigationThrottle::CheckURL() { |
| GURL url = navigation_handle()->GetURL(); |
| |
| bool skip_manual_parent_filter = |
| supervised_user::ShouldContentSkipParentAllowlistFiltering( |
| navigation_handle()->GetWebContents()->GetOutermostWebContents()); |
| |
| bool got_result = false; |
| |
| if (navigation_handle()->IsInPrimaryMainFrame()) { |
| got_result = url_filter_->GetFilteringBehaviorForURLWithAsyncChecks( |
| url, |
| base::BindOnce(&SupervisedUserNavigationThrottle::OnCheckDone, |
| weak_ptr_factory_.GetWeakPtr(), url), |
| skip_manual_parent_filter); |
| } else { |
| got_result = url_filter_->GetFilteringBehaviorForSubFrameURLWithAsyncChecks( |
| url, navigation_handle()->GetWebContents()->GetVisibleURL(), |
| base::BindOnce(&SupervisedUserNavigationThrottle::OnCheckDone, |
| weak_ptr_factory_.GetWeakPtr(), url)); |
| } |
| |
| DCHECK_EQ(got_result, |
| behavior_ != supervised_user::FilteringBehavior::kInvalid); |
| // If we got a "not blocked" result synchronously, don't defer. |
| deferred_ = |
| !got_result || (behavior_ == supervised_user::FilteringBehavior::kBlock); |
| if (got_result) { |
| behavior_ = supervised_user::FilteringBehavior::kInvalid; |
| } |
| } |
| |
| void SupervisedUserNavigationThrottle::ShowInterstitial( |
| const GURL& url, |
| supervised_user::FilteringBehaviorReason reason) { |
| // Don't show interstitial synchronously - it doesn't seem like a good idea to |
| // show an interstitial right in the middle of a call into a |
| // NavigationThrottle. This also lets OnInterstitialResult to be invoked |
| // synchronously, once a callback is passed into the |
| // SupervisedUserNavigationObserver. |
| base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, |
| base::BindOnce(&SupervisedUserNavigationThrottle::ShowInterstitialAsync, |
| weak_ptr_factory_.GetWeakPtr(), reason)); |
| } |
| |
| void SupervisedUserNavigationThrottle::ShowInterstitialAsync( |
| supervised_user::FilteringBehaviorReason reason) { |
| // May not yet have been set when ShowInterstitial was called, but should have |
| // been set by the time this is invoked. |
| DCHECK(deferred_); |
| SupervisedUserNavigationObserver::OnRequestBlocked( |
| navigation_handle()->GetWebContents(), navigation_handle()->GetURL(), |
| reason, navigation_handle()->GetNavigationId(), |
| navigation_handle()->GetFrameTreeNodeId(), |
| base::BindRepeating( |
| &SupervisedUserNavigationThrottle::OnInterstitialResult, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| content::NavigationThrottle::ThrottleCheckResult |
| SupervisedUserNavigationThrottle::ProcessRequest() { |
| deferred_ = false; |
| DCHECK_EQ(supervised_user::FilteringBehavior::kInvalid, behavior_); |
| |
| // We do not yet support prerendering for supervised users. |
| if (navigation_handle()->IsInPrerenderedMainFrame()) { |
| return NavigationThrottle::CANCEL; |
| } |
| |
| CheckURL(); |
| |
| if (deferred_) { |
| waiting_for_decision_.emplace(); |
| return NavigationThrottle::DEFER; |
| } |
| return NavigationThrottle::PROCEED; |
| } |
| |
| content::NavigationThrottle::ThrottleCheckResult |
| SupervisedUserNavigationThrottle::WillStartRequest() { |
| return ProcessRequest(); |
| } |
| |
| content::NavigationThrottle::ThrottleCheckResult |
| SupervisedUserNavigationThrottle::WillRedirectRequest() { |
| return ProcessRequest(); |
| } |
| |
| content::NavigationThrottle::ThrottleCheckResult |
| SupervisedUserNavigationThrottle::WillProcessResponse() { |
| if (base::FeatureList::GetInstance()->IsFeatureOverridden( |
| supervised_user::kClassifyUrlOnProcessResponseEvent.name)) { |
| // Safety measure: do not execute the code below for the Default experiment |
| // groups. 0 means that either the checks were never asynchronous, or that |
| // they took less than 0ms rounded combined, which is safe approximation. |
| base::UmaHistogramTimes( |
| supervised_user::kClassifiedLaterThanContentResponseHistogramName, |
| total_delay_); |
| VLOG(1) << "Time spent waiting for classifications: " << total_delay_; |
| } |
| return NavigationThrottle::PROCEED; |
| } |
| |
| const char* SupervisedUserNavigationThrottle::GetNameForLogging() { |
| return "SupervisedUserNavigationThrottle"; |
| } |
| |
| void SupervisedUserNavigationThrottle::OnCheckDone( |
| const GURL& url, |
| supervised_user::FilteringBehavior behavior, |
| supervised_user::FilteringBehaviorReason reason, |
| bool uncertain) { |
| DCHECK_EQ(supervised_user::FilteringBehavior::kInvalid, behavior_); |
| |
| // If we got a result synchronously, pass it back to ShowInterstitialIfNeeded. |
| if (!deferred_) { |
| behavior_ = behavior; |
| } |
| |
| reason_ = reason; |
| |
| ui::PageTransition transition = navigation_handle()->GetPageTransition(); |
| |
| supervised_user::SupervisedUserURLFilter::RecordFilterResultEvent( |
| behavior, reason, /*is_filtering_behavior_known=*/!uncertain, transition); |
| |
| if (behavior == supervised_user::FilteringBehavior::kBlock) { |
| ShowInterstitial(url, reason); |
| } else if (deferred_) { |
| if (base::FeatureList::GetInstance()->IsFeatureOverridden( |
| supervised_user::kClassifyUrlOnProcessResponseEvent.name)) { |
| // Safety measure: do not execute the code below for the Default |
| // experiment groups. |
| total_delay_ += waiting_for_decision_->Elapsed(); |
| waiting_for_decision_ = std::nullopt; |
| } |
| Resume(); |
| } |
| } |
| |
| #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_MAC) || BUILDFLAG(IS_WIN) |
| // Whether to show a re-auth interstitial instead of the parent approval |
| // interstitial. |
| bool SupervisedUserNavigationThrottle::ShouldShowReauthInterstitial( |
| const Profile* profile) { |
| if (!base::FeatureList::IsEnabled( |
| supervised_user:: |
| kForceSupervisedUserReauthenticationForBlockedSites)) { |
| return false; |
| } |
| |
| signin::IdentityManager* identity_manager = |
| IdentityManagerFactory::GetForProfileIfExists(profile); |
| CHECK(identity_manager); |
| |
| // Show the re-auth interstitial if the account is in a persistent error |
| // state. If the error state is transient, show the approval interstitial |
| // instead (an approval request will either succeed, or display a "try again |
| // later" message). |
| return identity_manager->HasAccountWithRefreshTokenInPersistentErrorState( |
| identity_manager->GetPrimaryAccountId(signin::ConsentLevel::kSignin)); |
| } |
| #endif |
| |
| void SupervisedUserNavigationThrottle::OnInterstitialResult( |
| CallbackActions action, |
| bool already_sent_request, |
| bool is_main_frame) { |
| switch (action) { |
| case kCancelNavigation: { |
| CancelDeferredNavigation(CANCEL); |
| break; |
| } |
| case kCancelWithInterstitial: { |
| CHECK(navigation_handle()); |
| Profile* profile = Profile::FromBrowserContext( |
| navigation_handle()->GetWebContents()->GetBrowserContext()); |
| |
| #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_MAC) || BUILDFLAG(IS_WIN) |
| if (ShouldShowReauthInterstitial(profile)) { |
| // Show the re-authentication interstitial if the user signed out of the |
| // content area, as parent's approval requires authentication. This |
| // interstitial is only available on Linux/Mac/Windows as ChromeOS and |
| // Android have different re-auth mechanisms. |
| CancelDeferredNavigation( |
| content::NavigationThrottle::ThrottleCheckResult( |
| CANCEL, net::ERR_BLOCKED_BY_CLIENT, |
| supervised_user::CreateReauthenticationInterstitial( |
| *navigation_handle(), |
| SupervisedUserVerificationPage::VerificationPurpose:: |
| BLOCKED_SITE))); |
| return; |
| } |
| #endif |
| |
| std::string interstitial_html = |
| supervised_user::SupervisedUserInterstitial::GetHTMLContents( |
| SupervisedUserServiceFactory::GetForProfile(profile), |
| profile->GetPrefs(), reason_, already_sent_request, is_main_frame, |
| g_browser_process->GetApplicationLocale()); |
| CancelDeferredNavigation(content::NavigationThrottle::ThrottleCheckResult( |
| CANCEL, net::ERR_BLOCKED_BY_CLIENT, interstitial_html)); |
| } |
| } |
| } |