blob: 4890135b4d96b4b042ae9fe526dd0c9631c3da2e [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/security_state/core/security_state.h"
#include <stdint.h>
#include <string>
#include "base/command_line.h"
#include "base/metrics/field_trial.h"
#include "base/metrics/field_trial_params.h"
#include "base/metrics/histogram_macros.h"
#include "components/security_state/core/features.h"
#include "net/ssl/ssl_cipher_suite_names.h"
#include "net/ssl/ssl_connection_status_flags.h"
namespace security_state {
namespace {
// Returns true if |url| is a blob: URL and its path parses as a GURL with a
// nonsecure origin, and false otherwise. See
// https://url.spec.whatwg.org/#origin.
bool IsNonsecureBlobUrl(
const GURL& url,
const IsOriginSecureCallback& is_origin_secure_callback) {
if (!url.SchemeIs(url::kBlobScheme))
return false;
GURL inner_url(url.path());
return !is_origin_secure_callback.Run(inner_url);
}
// For nonsecure pages, returns a SecurityLevel based on the
// provided information and the kMarkHttpAsFeature field trial.
SecurityLevel GetSecurityLevelForNonSecureFieldTrial(
bool is_error_page,
const InsecureInputEventData& input_events) {
if (base::FeatureList::IsEnabled(features::kMarkHttpAsFeature)) {
std::string parameter = base::GetFieldTrialParamValueByFeature(
features::kMarkHttpAsFeature,
features::kMarkHttpAsFeatureParameterName);
if (parameter == features::kMarkHttpAsParameterDangerous) {
return DANGEROUS;
}
}
// Default to dangerous on editing form fields and otherwise
// warning.
return input_events.insecure_field_edited ? DANGEROUS : HTTP_SHOW_WARNING;
}
std::string GetHistogramSuffixForSecurityLevel(
security_state::SecurityLevel level) {
switch (level) {
case EV_SECURE:
return "EV_SECURE";
case SECURE:
return "SECURE";
case NONE:
return "NONE";
case HTTP_SHOW_WARNING:
return "HTTP_SHOW_WARNING";
case SECURE_WITH_POLICY_INSTALLED_CERT:
return "SECURE_WITH_POLICY_INSTALLED_CERT";
case DANGEROUS:
return "DANGEROUS";
default:
return "OTHER";
}
}
} // namespace
SecurityLevel GetSecurityLevel(
const VisibleSecurityState& visible_security_state,
bool used_policy_installed_certificate,
IsOriginSecureCallback is_origin_secure_callback) {
// Override the connection security information if the website failed the
// browser's malware checks.
if (visible_security_state.malicious_content_status !=
MALICIOUS_CONTENT_STATUS_NONE) {
return DANGEROUS;
}
if (!visible_security_state.connection_info_initialized) {
return NONE;
}
const GURL url = visible_security_state.url;
const bool is_cryptographic_with_certificate =
(url.SchemeIsCryptographic() && visible_security_state.certificate);
const bool is_major_cert_error =
net::IsCertStatusError(visible_security_state.cert_status) &&
!net::IsCertStatusMinorError(visible_security_state.cert_status);
// Set the security level to DANGEROUS for major certificate errors.
if (is_cryptographic_with_certificate && is_major_cert_error) {
return DANGEROUS;
}
// data: URLs don't define a secure context, and are a vector for spoofing.
// Likewise, ftp: URLs are always non-secure, and are uncommon enough that
// we can treat them as such without significant user impact.
//
// Display a "Not secure" badge for all these URLs.
if (url.SchemeIs(url::kDataScheme) || url.SchemeIs(url::kFtpScheme)) {
return HTTP_SHOW_WARNING;
}
// Choose the appropriate security level for requests to HTTP and remaining
// pseudo URLs (blob:, filesystem:). filesystem: is a standard scheme so does
// not need to be explicitly listed here.
// TODO(meacer): Remove special case for blob (crbug.com/684751).
if (!is_cryptographic_with_certificate) {
if (!visible_security_state.is_error_page &&
!is_origin_secure_callback.Run(url) &&
(url.IsStandard() ||
IsNonsecureBlobUrl(url, is_origin_secure_callback))) {
return GetSecurityLevelForNonSecureFieldTrial(
visible_security_state.is_error_page,
visible_security_state.insecure_input_events);
}
return NONE;
}
// Downgrade the security level for active insecure subresources.
if (visible_security_state.ran_mixed_content ||
visible_security_state.ran_content_with_cert_errors) {
return kRanInsecureContentLevel;
}
// In most cases, SHA1 use is treated as a certificate error, in which case
// DANGEROUS will have been returned above. If SHA1 was permitted by policy,
// downgrade the security level to Neutral.
if (IsSHA1InChain(visible_security_state)) {
return NONE;
}
// Active mixed content is handled above.
DCHECK(!visible_security_state.ran_mixed_content);
DCHECK(!visible_security_state.ran_content_with_cert_errors);
if (visible_security_state.contained_mixed_form ||
visible_security_state.displayed_mixed_content ||
visible_security_state.displayed_content_with_cert_errors) {
return kDisplayedInsecureContentLevel;
}
if (net::IsCertStatusError(visible_security_state.cert_status)) {
// Major cert errors are handled above.
DCHECK(net::IsCertStatusMinorError(visible_security_state.cert_status));
return NONE;
}
if (visible_security_state.is_view_source) {
return NONE;
}
// Any prior observation of a policy-installed cert is a strong indicator
// of a MITM being present (the enterprise), so a "secure-but-inspected"
// security level is returned.
if (used_policy_installed_certificate) {
return SECURE_WITH_POLICY_INSTALLED_CERT;
}
if ((visible_security_state.cert_status & net::CERT_STATUS_IS_EV) &&
visible_security_state.certificate) {
return EV_SECURE;
}
return SECURE;
}
VisibleSecurityState::VisibleSecurityState()
: malicious_content_status(MALICIOUS_CONTENT_STATUS_NONE),
connection_info_initialized(false),
cert_status(0),
connection_status(0),
key_exchange_group(0),
peer_signature_algorithm(0),
displayed_mixed_content(false),
contained_mixed_form(false),
ran_mixed_content(false),
displayed_content_with_cert_errors(false),
ran_content_with_cert_errors(false),
pkp_bypassed(false),
is_error_page(false),
is_view_source(false) {}
VisibleSecurityState::~VisibleSecurityState() {}
bool IsSchemeCryptographic(const GURL& url) {
return url.is_valid() && url.SchemeIsCryptographic();
}
bool IsOriginLocalhostOrFile(const GURL& url) {
return url.is_valid() && (net::IsLocalhost(url) || url.SchemeIsFile());
}
bool IsSslCertificateValid(SecurityLevel security_level) {
return security_level == SECURE || security_level == EV_SECURE ||
security_level == SECURE_WITH_POLICY_INSTALLED_CERT;
}
std::string GetSecurityLevelHistogramName(
const std::string& prefix,
security_state::SecurityLevel level) {
return prefix + "." + GetHistogramSuffixForSecurityLevel(level);
}
bool IsSHA1InChain(const VisibleSecurityState& visible_security_state) {
return visible_security_state.certificate &&
(visible_security_state.cert_status &
net::CERT_STATUS_SHA1_SIGNATURE_PRESENT);
}
} // namespace security_state