Chromium Code Reviews| Index: chrome/browser/browsing_data/clear_site_data_throttle.cc |
| diff --git a/chrome/browser/browsing_data/clear_site_data_throttle.cc b/chrome/browser/browsing_data/clear_site_data_throttle.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..aa2d9fb16c0c01c4e1f05d9ba9755e5b732b05e0 |
| --- /dev/null |
| +++ b/chrome/browser/browsing_data/clear_site_data_throttle.cc |
| @@ -0,0 +1,154 @@ |
| +// Copyright 2016 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 "chrome/browser/browsing_data/clear_site_data_throttle.h" |
| + |
| +#include "base/bind.h" |
| +#include "base/callback.h" |
| +#include "base/values.h" |
| +#include "chrome/browser/browsing_data/browsing_data_helper.h" |
| +#include "chrome/browser/browsing_data/browsing_data_remover.h" |
| +#include "chrome/browser/browsing_data/browsing_data_remover_factory.h" |
| +#include "chrome/browser/browsing_data/registrable_domain_filter_builder.h" |
| +#include "chrome/browser/profiles/profile.h" |
| +#include "components/safe_json/safe_json_parser.h" |
| +#include "content/public/browser/browser_thread.h" |
| +#include "content/public/browser/web_contents.h" |
| +#include "net/base/registry_controlled_domains/registry_controlled_domain.h" |
| +#include "net/url_request/url_request.h" |
| +#include "url/gurl.h" |
| + |
| +namespace { |
| + |
| +const char* kNameForLogging = "ClearSiteDataThrottle"; |
| + |
| +const char* kClearSiteDataHeader = "Clear-Site-Data"; |
| + |
| +const char* kTypesKey = "types"; |
| + |
| +} // namespace |
| + |
| +ClearSiteDataThrottle::ClearSiteDataThrottle(const net::URLRequest* request) |
| + : request_(request) {} |
| + |
| +ClearSiteDataThrottle::~ClearSiteDataThrottle() {} |
| + |
| +void ClearSiteDataThrottle::WillProcessResponse(bool* defer) { |
| + std::string header; |
| + request_->GetResponseHeaderByName(kClearSiteDataHeader, &header); |
| + if (header.empty()) |
| + return; |
| + |
| + std::string domain; |
| + if (request_->url().HostIsIPAddress()) { |
| + domain = request_->url().host(); |
| + } else { |
| + domain = GetDomainAndRegistry( |
|
Mike West
2016/05/31 05:35:51
This is going to strip `www.example.com` down to `
msramek
2016/06/01 14:20:40
Done. Sorry, forgot to add a comment about this. I
|
| + request_->url(), |
| + net::registry_controlled_domains::INCLUDE_PRIVATE_REGISTRIES); |
| + } |
| + |
| + if (domain.empty()) |
|
Mike West
2016/05/31 05:35:51
Another implication is that folks on an intranet w
msramek
2016/06/01 14:20:40
Yes, or even a simpler example: localhost. I think
|
| + return; |
| + |
| + const content::ResourceRequestInfo* info = |
| + content::ResourceRequestInfo::ForRequest(request_); |
| + content::BrowserThread::PostTask( |
| + content::BrowserThread::UI, FROM_HERE, |
| + base::Bind( |
| + &ClearSiteDataThrottle::ProcessResponseOnUIThread, |
| + info->GetWebContentsGetterForRequest(), |
| + domain, |
| + header)); |
| +} |
| + |
| +const char* ClearSiteDataThrottle::GetNameForLogging() const { |
| + return kNameForLogging; |
| +} |
| + |
| +// static |
| +void ClearSiteDataThrottle::ProcessResponseOnUIThread( |
| + const content::ResourceRequestInfo::WebContentsGetter& getter, |
| + std::string domain, |
| + std::string header) { |
| + DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| + |
| + content::WebContents* web_contents = getter.Run(); |
| + if (!web_contents) |
| + return; |
| + |
| + Profile* profile = |
| + Profile::FromBrowserContext( |
| + web_contents->GetBrowserContext())->GetOriginalProfile(); |
| + |
| + safe_json::SafeJsonParser::Parse( |
|
Mike West
2016/05/31 05:35:51
Why do we need to parse JSON on the UI thread? Can
msramek
2016/06/01 14:20:41
N/A, since after moving to content/ were using the
|
| + header, |
| + base::Bind( |
| + &ClearSiteDataThrottle::ClearData, |
| + base::Unretained(profile), |
| + domain), |
| + base::Bind(&ClearSiteDataThrottle::OnParseError)); |
| +} |
| + |
| +// static |
| +void ClearSiteDataThrottle::ClearData( |
| + Profile* profile, |
| + std::string domain, |
| + std::unique_ptr<base::Value> header) { |
| + if (!header || !header->GetAsDictionary(nullptr)) { |
| + DVLOG(1) << "Not a valid dictionary value."; |
|
Mike West
2016/05/31 05:35:52
// TODO(you): Write a console error.
msramek
2016/06/01 14:20:40
Done. (actually implemented the console error)
|
| + return; |
| + } |
| + |
| + const base::ListValue* types; |
| + if (!static_cast<base::DictionaryValue*>(header.get()) |
| + ->GetListWithoutPathExpansion(kTypesKey, &types)) { |
| + DVLOG(1) << "No 'types' field present."; |
|
Mike West
2016/05/31 05:35:51
// TODO(you): Write a console error.
msramek
2016/06/01 14:20:40
Done. (actually implemented the console error)
|
| + return; |
| + } |
| + |
| + int remove_mask = 0; |
| + for (const base::Value* value : *types) { |
|
Mike West
2016/05/31 05:35:51
Nit: These are `unique_ptr` aren't they?
msramek
2016/06/01 14:20:40
types->begin() is std::vector<base::Value*>::const
msramek
2016/06/01 14:25:30
Ah. Because this is a recent change and I have an
|
| + std::string type; |
| + if (!value->GetAsString(&type)) { |
| + DVLOG(1) << "Invalid type: " << *value; |
|
Mike West
2016/05/31 05:35:51
// TODO(you): Write a console error.
msramek
2016/06/01 14:20:40
Done. (actually implemented the console error)
|
| + continue; |
| + } |
| + |
| + if (type == "cache") { |
| + remove_mask |= BrowsingDataRemover::REMOVE_CACHE; |
| + DVLOG(2) << "Clearing cache."; |
| + } else if (type == "cookies") { |
| + remove_mask |= BrowsingDataRemover::REMOVE_COOKIES; |
| + DVLOG(2) << "Clearing cookies."; |
| + } else if (type == "storage") { |
| + remove_mask |= BrowsingDataRemover::REMOVE_SITE_DATA; |
|
Mike West
2016/05/31 05:35:51
This includes cookies; it shouldn't. I think you'l
msramek
2016/06/01 14:20:40
Done. Thanks for paying attention to this :)
|
| + DVLOG(2) << "Clearing storage."; |
| + } |
| + } |
| + |
| + if (!remove_mask) { |
| + DVLOG(2) << "No valid types specified for clearing."; |
|
Mike West
2016/05/31 05:35:51
// TODO(you): Write a console error.
msramek
2016/06/01 14:20:41
Done. (actually implemented the console error)
|
| + return; |
| + } |
| + |
| + BrowsingDataRemover* browsing_data_remover = |
| + BrowsingDataRemoverFactory::GetForBrowserContext(profile); |
| + |
| + RegistrableDomainFilterBuilder filter_builder( |
| + BrowsingDataFilterBuilder::WHITELIST); |
| + filter_builder.AddRegisterableDomain(domain); |
|
Mike West
2016/05/31 05:35:51
Again, we should talk about the registerable domai
msramek
2016/06/01 14:20:40
Please see my current implementation. I'm using or
|
| + |
| + browsing_data_remover->RemoveWithFilter( |
|
Mike West
2016/05/31 05:35:51
This eventually hops back to the IO thread, right?
msramek
2016/06/01 14:20:40
N/A, since we're not doing any more thread hopping
|
| + BrowsingDataRemover::Period(BrowsingDataRemover::TimePeriod::EVERYTHING), |
| + remove_mask, |
| + BrowsingDataHelper::UNPROTECTED_WEB, |
| + filter_builder); |
| +} |
| + |
| +// static |
| +void ClearSiteDataThrottle::OnParseError(const std::string& error) { |
| + DVLOG(1) << "The header value is not a valid JSON."; |
| + DVLOG(1) << error; |
|
Mike West
2016/05/31 05:35:52
// TODO(you): write a console error.
msramek
2016/06/01 14:20:41
Done. (actually implemented the console error)
|
| +} |