| /* |
| * Copyright (C) 2012 Motorola Mobility Inc. |
| * Copyright (C) 2013 Google Inc. All Rights Reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "third_party/blink/renderer/core/fileapi/public_url_manager.h" |
| |
| #include "base/metrics/histogram_macros.h" |
| #include "mojo/public/cpp/bindings/pending_remote.h" |
| #include "third_party/blink/public/common/blob/blob_utils.h" |
| #include "third_party/blink/public/mojom/blob/blob_registry.mojom-blink.h" |
| #include "third_party/blink/renderer/core/execution_context/execution_context.h" |
| #include "third_party/blink/renderer/core/fileapi/url_registry.h" |
| #include "third_party/blink/renderer/platform/blob/blob_data.h" |
| #include "third_party/blink/renderer/platform/blob/blob_url.h" |
| #include "third_party/blink/renderer/platform/blob/blob_url_null_origin_map.h" |
| #include "third_party/blink/renderer/platform/runtime_enabled_features.h" |
| #include "third_party/blink/renderer/platform/scheduler/main_thread/task_type_names.h" |
| #include "third_party/blink/renderer/platform/weborigin/kurl.h" |
| #include "third_party/blink/renderer/platform/weborigin/security_origin.h" |
| #include "third_party/blink/renderer/platform/wtf/vector.h" |
| |
| namespace blink { |
| |
| namespace { |
| |
| static void RemoveFromNullOriginMapIfNecessary(const KURL& blob_url) { |
| DCHECK(blob_url.ProtocolIs("blob")); |
| if (BlobURL::GetOrigin(blob_url) == "null") |
| BlobURLNullOriginMap::GetInstance()->Remove(blob_url); |
| } |
| |
| } // namespace |
| |
| PublicURLManager::PublicURLManager(ExecutionContext* context) |
| : ExecutionContextLifecycleObserver(context), |
| is_stopped_(false), |
| url_store_(context) { |
| BlobDataHandle::GetBlobRegistry()->URLStoreForOrigin( |
| context->GetSecurityOrigin(), |
| url_store_.BindNewEndpointAndPassReceiver( |
| context->GetTaskRunner(TaskType::kFileReading))); |
| } |
| |
| String PublicURLManager::RegisterURL(URLRegistrable* registrable) { |
| if (is_stopped_) |
| return String(); |
| |
| SecurityOrigin* origin = GetExecutionContext()->GetMutableSecurityOrigin(); |
| const KURL& url = BlobURL::CreatePublicURL(origin); |
| DCHECK(!url.IsEmpty()); |
| const String& url_string = url.GetString(); |
| |
| if (registrable->IsMojoBlob()) { |
| // Measure how much jank the following synchronous IPC introduces. |
| SCOPED_UMA_HISTOGRAM_TIMER("Storage.Blob.RegisterPublicURLTime"); |
| mojo::PendingRemote<mojom::blink::Blob> blob_remote; |
| mojo::PendingReceiver<mojom::blink::Blob> blob_receiver = |
| blob_remote.InitWithNewPipeAndPassReceiver(); |
| url_store_->Register(std::move(blob_remote), url); |
| mojo_urls_.insert(url_string); |
| registrable->CloneMojoBlob(std::move(blob_receiver)); |
| } else { |
| URLRegistry* registry = ®istrable->Registry(); |
| registry->RegisterURL(origin, url, registrable); |
| url_to_registry_.insert(url_string, registry); |
| } |
| |
| if (origin->SerializesAsNull()) |
| BlobURLNullOriginMap::GetInstance()->Add(url, origin); |
| |
| return url_string; |
| } |
| |
| void PublicURLManager::Revoke(const KURL& url) { |
| if (is_stopped_) |
| return; |
| // Don't bother trying to revoke URLs that can't have been registered anyway. |
| if (!url.ProtocolIs("blob") || url.HasFragmentIdentifier()) |
| return; |
| // Don't support revoking cross-origin blob URLs. |
| if (!SecurityOrigin::Create(url)->IsSameOriginWith( |
| GetExecutionContext()->GetSecurityOrigin())) |
| return; |
| |
| url_store_->Revoke(url); |
| mojo_urls_.erase(url.GetString()); |
| |
| RemoveFromNullOriginMapIfNecessary(url); |
| auto it = url_to_registry_.find(url.GetString()); |
| if (it == url_to_registry_.end()) |
| return; |
| it->value->UnregisterURL(url); |
| url_to_registry_.erase(it); |
| } |
| |
| void PublicURLManager::Resolve( |
| const KURL& url, |
| mojo::PendingReceiver<network::mojom::blink::URLLoaderFactory> |
| factory_receiver) { |
| if (is_stopped_) |
| return; |
| |
| DCHECK(url.ProtocolIs("blob")); |
| url_store_->ResolveAsURLLoaderFactory(url, std::move(factory_receiver)); |
| } |
| |
| void PublicURLManager::Resolve( |
| const KURL& url, |
| mojo::PendingReceiver<mojom::blink::BlobURLToken> token_receiver) { |
| if (is_stopped_) |
| return; |
| |
| DCHECK(url.ProtocolIs("blob")); |
| url_store_->ResolveForNavigation(url, std::move(token_receiver)); |
| } |
| |
| void PublicURLManager::ContextDestroyed() { |
| if (is_stopped_) |
| return; |
| |
| is_stopped_ = true; |
| for (auto& url_registry : url_to_registry_) { |
| url_registry.value->UnregisterURL(KURL(url_registry.key)); |
| RemoveFromNullOriginMapIfNecessary(KURL(url_registry.key)); |
| } |
| for (const auto& url : mojo_urls_) |
| RemoveFromNullOriginMapIfNecessary(KURL(url)); |
| |
| url_to_registry_.clear(); |
| mojo_urls_.clear(); |
| } |
| |
| void PublicURLManager::Trace(Visitor* visitor) const { |
| visitor->Trace(url_store_); |
| ExecutionContextLifecycleObserver::Trace(visitor); |
| } |
| |
| } // namespace blink |