Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(109)

Unified Diff: net/base/registry_controlled_domains/registry_controlled_domain.cc

Issue 2433583002: Reduce buggy usage of the registry controlled domain service. (Closed)
Patch Set: Fix Created 4 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: net/base/registry_controlled_domains/registry_controlled_domain.cc
diff --git a/net/base/registry_controlled_domains/registry_controlled_domain.cc b/net/base/registry_controlled_domains/registry_controlled_domain.cc
index 607d735f3080be6b37354b5963064df6e78bd7fa..ece6c7927ef61388af59f98fbedea549158bdb8c 100644
--- a/net/base/registry_controlled_domains/registry_controlled_domain.cc
+++ b/net/base/registry_controlled_domains/registry_controlled_domain.cc
@@ -66,10 +66,19 @@ namespace {
const unsigned char* g_graph = kDafsa;
size_t g_graph_length = sizeof(kDafsa);
+struct MappedHostComponent {
+ size_t original_begin;
+ size_t original_end;
+
+ size_t canonical_begin;
+ size_t canonical_end;
+};
+
size_t GetRegistryLengthImpl(base::StringPiece host,
UnknownRegistryFilter unknown_filter,
PrivateRegistryFilter private_filter) {
- DCHECK(!host.empty());
+ if (host.empty())
+ return std::string::npos;
// Skip leading dots.
const size_t host_check_begin = host.find_first_not_of('.');
@@ -189,6 +198,106 @@ base::StringPiece GetDomainAndRegistryAsStringPiece(
return GetDomainAndRegistryImpl(host, filter);
}
+// Backend for PermissiveGetHostRegistryLength that handles both UTF-8 and
+// UTF-16 input. The template type is the std::string type to use (it makes the
+// typedefs easier than using the character type).
+template <typename Str>
+size_t DoPermissiveGetHostRegistryLength(base::BasicStringPiece<Str> host,
+ UnknownRegistryFilter unknown_filter,
+ PrivateRegistryFilter private_filter) {
+ std::string canonical_host; // No not modify outside of canon_output.
Peter Kasting 2016/10/22 05:04:19 Nit: No -> Do ?
+ canonical_host.reserve(host.size());
Peter Kasting 2016/10/22 05:04:19 Nit: The rest of this file, and the majority of Ch
brettw 2016/10/24 21:45:24 I've always written size() but will use length() f
+ url::StdStringCanonOutput canon_output(&canonical_host);
+
+ std::vector<MappedHostComponent> components;
+
+ size_t current = 0;
+ while (current < host.size()) {
Peter Kasting 2016/10/22 05:04:19 Nit: Shorter, narrows scope of |current|: for (
brettw 2016/10/24 21:45:24 Done.
+ // Advance to next "." or end.
Peter Kasting 2016/10/22 05:04:19 Nit: Technically, this comment belongs on the whil
brettw 2016/10/24 21:45:24 Done.
+ size_t begin = current;
Peter Kasting 2016/10/22 05:04:19 Nit: Can be const
brettw 2016/10/24 21:45:24 I prefer not to mark random local variables const
Peter Kasting 2016/10/24 23:04:32 There's no Chromium-wide pattern, so that's fine.
+ while (current < host.size() && host[current] != '.')
+ current++;
Peter Kasting 2016/10/22 05:04:19 Nit: Shorter: current = std::min(host.find('.
brettw 2016/10/24 21:45:24 This seems less readable to me.
Peter Kasting 2016/10/24 23:04:32 What about: current = host.find('.', begin); if (
brettw 2016/10/25 20:28:17 I can live with this.
+
+ MappedHostComponent mapping;
+ mapping.original_begin = begin;
+ mapping.original_end = current;
+ mapping.canonical_begin = static_cast<size_t>(canon_output.length());
Peter Kasting 2016/10/22 05:04:19 Nit: If you save this in a temp here instead of cr
brettw 2016/10/24 21:45:24 This feels icky to me, I don't like implicitly dep
+
+ // Append the canonicalized version of this component.
Peter Kasting 2016/10/22 05:04:19 Nit: Append -> Try to append ?
brettw 2016/10/24 21:45:24 Done.
+ if (!url::CanonicalizeHostSubstring(
+ host.data(), url::Component(static_cast<int>(begin),
+ static_cast<int>(current - begin)),
+ &canon_output)) {
+ // Failure to canonicalize this component, append as-is.
Peter Kasting 2016/10/22 05:04:19 Nit: Failure -> Failed; comma -> semicolon
+ for (size_t i = begin; i < current; i++)
+ canon_output.push_back(host[i]);
Peter Kasting 2016/10/22 05:04:19 Nit: Shorter and more efficient: canon_outp
brettw 2016/10/24 21:45:24 Actually this old code was wrong in the UTF-16 cas
+ }
+
+ mapping.canonical_end = static_cast<size_t>(canon_output.length());
+ components.push_back(mapping);
+
+ if (current < host.size()) {
+ canon_output.push_back('.');
+ current++; // Skip over '.' in input.
+ }
+ }
+ canon_output.Complete();
+
+ size_t canonical_rcd_len =
+ GetRegistryLengthImpl(canonical_host, unknown_filter, private_filter);
+ if (canonical_rcd_len == 0 || canonical_rcd_len == std::string::npos)
+ return canonical_rcd_len; // Error or no registry controlled domain.
+
+ // Find which host component the result started in.
+ size_t canonical_rcd_begin = canonical_host.size() - canonical_rcd_len;
+ for (const auto& mapping : components) {
+ // In the common case, GetRegistryLengthImpl will identify the beginning
+ // of a component and we can just return where that component was in the
+ // original string.
+ if (canonical_rcd_begin == mapping.canonical_begin)
+ return host.size() - mapping.original_begin;
+
+ if (canonical_rcd_begin < mapping.canonical_end) {
Peter Kasting 2016/10/22 05:04:19 Nit: If you change this to this, at the top of the
brettw 2016/10/24 21:45:24 Done.
+ // The registry controlled domain begin was identified as being in the
+ // middle of this dot-separated domain component in the non-canonical
+ // input. This indicates some form of escaped dot, or a non-ASCII
+ // character that was canonicalized to a dot.
+ //
+ // Brute-force search from the end by repeatedly canonicalizing longer
+ // substrings until we get a match for the canonicalized version. This
+ // depends on the canonicalization process not changing the order of the
+ // characters. Punycode can change the order of characters, but it
+ // doesn't work across dots so this is safe.
+
+ // Expected canonical registry controlled domain.
+ base::StringPiece canonical_rcd(&canonical_host[canonical_rcd_begin],
+ canonical_rcd_len);
+
+ for (int current_try = static_cast<int>(mapping.original_end) - 1;
Peter Kasting 2016/10/22 05:04:19 Can't we binary-search this instead of linear-sear
brettw 2016/10/24 21:45:24 That doesn't work. Canonicalization can make the r
Peter Kasting 2016/10/24 23:04:32 Yeah, you're right. I thought about this some but
brettw 2016/10/25 20:28:17 Done.
+ current_try >= static_cast<int>(mapping.original_begin);
Peter Kasting 2016/10/22 05:04:19 Seems like we could get by with ">" instead of ">=
Peter Kasting 2016/10/25 01:33:32 Still wondering about this.
brettw 2016/10/25 20:28:17 You're correct, but in the context of a loop count
Peter Kasting 2016/10/25 20:37:53 Maybe write '>' with a comment saying we already c
+ current_try--) {
Peter Kasting 2016/10/22 05:04:19 Nit: Predecrement
brettw 2016/10/24 21:45:24 I don't understand this comment. Are you asking be
Peter Kasting 2016/10/24 23:04:32 There's no logic problem, I was suggesting using p
brettw 2016/10/25 20:28:17 I always write postfix for scalars because I think
Peter Kasting 2016/10/25 20:37:53 This surprised me so I tried to look myself: http
+ std::string try_string;
+ url::StdStringCanonOutput try_output(&try_string);
+
+ if (!url::CanonicalizeHostSubstring(
+ host.data(),
+ url::Component(
+ current_try,
+ static_cast<int>(mapping.original_end) - current_try),
+ &try_output))
+ continue; // Invalid substring, skip.
+
+ try_output.Complete();
+ if (try_string == canonical_rcd)
+ return host.size() - current_try;
+ }
+ }
+ }
+
+ NOTREACHED();
+ return canonical_rcd_len;
+}
+
} // namespace
std::string GetDomainAndRegistry(const GURL& gurl,
@@ -238,26 +347,59 @@ size_t GetRegistryLength(
const GURL& gurl,
UnknownRegistryFilter unknown_filter,
PrivateRegistryFilter private_filter) {
- base::StringPiece host = gurl.host_piece();
- if (host.empty())
- return std::string::npos;
- if (gurl.HostIsIPAddress())
Peter Kasting 2016/10/22 05:04:19 I assume deleting this is not a behavior change be
brettw 2016/10/24 21:45:24 I did this because it can never match, and decodin
- return 0;
- return GetRegistryLengthImpl(host, unknown_filter, private_filter);
+ return GetRegistryLengthImpl(gurl.host_piece(), unknown_filter,
+ private_filter);
}
-size_t GetRegistryLength(base::StringPiece host,
- UnknownRegistryFilter unknown_filter,
- PrivateRegistryFilter private_filter) {
+bool HostHasRegistryControlledDomain(base::StringPiece host,
+ UnknownRegistryFilter unknown_filter,
+ PrivateRegistryFilter private_filter) {
url::CanonHostInfo host_info;
const std::string canon_host(CanonicalizeHost(host, &host_info));
- if (canon_host.empty())
- return std::string::npos;
- if (host_info.IsIPAddress())
- return 0;
+ switch (host_info.family) {
+ case url::CanonHostInfo::IPV4:
+ case url::CanonHostInfo::IPV6:
+ // IP addresses don't have R.C.D.'s.
+ return false;
+ case url::CanonHostInfo::BROKEN:
+ // Host is not canonicalizable. Fall back to the slower "permissive"
+ // version.
+ return PermissiveGetHostRegistryLength(host, unknown_filter,
+ private_filter) > 0;
Peter Kasting 2016/10/22 05:04:19 "> 0" doesn't seem right; we want to return false
brettw 2016/10/24 21:45:23 Doth done, thanks for catching. I added some unit
+ case url::CanonHostInfo::NEUTRAL:
+ return GetRegistryLengthImpl(host, unknown_filter, private_filter) > 0;
Peter Kasting 2016/10/22 05:04:19 Same question.
+ default:
+ NOTREACHED();
+ return false
Peter Kasting 2016/10/22 05:04:19 Semicolon
+ }
+}
+
+size_t GetCanonicalHostRegistryLength(base::StringPiece canon_host,
+ UnknownRegistryFilter unknown_filter,
+ PrivateRegistryFilter private_filter) {
+#ifndef NDEBUG
+ // Ensure passed-in host name is canonical.
+ url::CanonHostInfo host_info;
+ DCHECK_EQ(net::CanonicalizeHost(canon_host, &host_info), canon_host);
+#endif
+
return GetRegistryLengthImpl(canon_host, unknown_filter, private_filter);
}
+size_t PermissiveGetHostRegistryLength(base::StringPiece host,
+ UnknownRegistryFilter unknown_filter,
+ PrivateRegistryFilter private_filter) {
+ return DoPermissiveGetHostRegistryLength<std::string>(host, unknown_filter,
+ private_filter);
+}
+
+size_t PermissiveGetHostRegistryLength(base::StringPiece16 host,
+ UnknownRegistryFilter unknown_filter,
+ PrivateRegistryFilter private_filter) {
+ return DoPermissiveGetHostRegistryLength<base::string16>(host, unknown_filter,
+ private_filter);
+}
+
void SetFindDomainGraph() {
g_graph = kDafsa;
g_graph_length = sizeof(kDafsa);

Powered by Google App Engine
This is Rietveld 408576698