/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "URLWorker.h" #include "mozilla/dom/Blob.h" #include "nsHostObjectProtocolHandler.h" #include "WorkerPrivate.h" #include "WorkerRunnable.h" #include "WorkerScope.h" #include "nsStandardURL.h" #include "nsURLHelper.h" namespace mozilla { using net::nsStandardURL; namespace dom { using namespace workers; // Proxy class to forward all the requests to a URLMainThread object. class URLWorker::URLProxy final { public: NS_INLINE_DECL_THREADSAFE_REFCOUNTING(URLProxy) explicit URLProxy(already_AddRefed aURL) : mURL(aURL) { MOZ_ASSERT(NS_IsMainThread()); } URLMainThread* URL() { MOZ_ASSERT(NS_IsMainThread()); return mURL; } nsIURI* URI() { MOZ_ASSERT(NS_IsMainThread()); return mURL->GetURI(); } void ReleaseURI() { MOZ_ASSERT(NS_IsMainThread()); mURL = nullptr; } private: // Private destructor, to discourage deletion outside of Release(): ~URLProxy() { MOZ_ASSERT(!mURL); } RefPtr mURL; }; // This class creates an URL from a DOM Blob on the main thread. class CreateURLRunnable : public WorkerMainThreadRunnable { private: BlobImpl* mBlobImpl; nsAString& mURL; public: CreateURLRunnable(WorkerPrivate* aWorkerPrivate, BlobImpl* aBlobImpl, nsAString& aURL) : WorkerMainThreadRunnable(aWorkerPrivate, NS_LITERAL_CSTRING("URL :: CreateURL")) , mBlobImpl(aBlobImpl) , mURL(aURL) { MOZ_ASSERT(aBlobImpl); DebugOnly isMutable; MOZ_ASSERT(NS_SUCCEEDED(aBlobImpl->GetMutable(&isMutable))); MOZ_ASSERT(!isMutable); } bool MainThreadRun() override { using namespace mozilla::ipc; AssertIsOnMainThread(); DebugOnly isMutable; MOZ_ASSERT(NS_SUCCEEDED(mBlobImpl->GetMutable(&isMutable))); MOZ_ASSERT(!isMutable); nsCOMPtr principal = mWorkerPrivate->GetPrincipal(); nsAutoCString url; nsresult rv = nsHostObjectProtocolHandler::AddDataEntry(mBlobImpl, principal, url); if (NS_FAILED(rv)) { NS_WARNING("Failed to add data entry for the blob!"); SetDOMStringToNull(mURL); return false; } if (!mWorkerPrivate->IsSharedWorker() && !mWorkerPrivate->IsServiceWorker()) { // Walk up to top worker object. WorkerPrivate* wp = mWorkerPrivate; while (WorkerPrivate* parent = wp->GetParent()) { wp = parent; } nsCOMPtr sc = wp->GetScriptContext(); // We could not have a ScriptContext in JSM code. In this case, we leak. if (sc) { nsCOMPtr global = sc->GetGlobalObject(); MOZ_ASSERT(global); global->RegisterHostObjectURI(url); } } mURL = NS_ConvertUTF8toUTF16(url); return true; } }; // This class revokes an URL on the main thread. class RevokeURLRunnable : public WorkerMainThreadRunnable { private: const nsString mURL; public: RevokeURLRunnable(WorkerPrivate* aWorkerPrivate, const nsAString& aURL) : WorkerMainThreadRunnable(aWorkerPrivate, NS_LITERAL_CSTRING("URL :: RevokeURL")) , mURL(aURL) {} bool MainThreadRun() override { AssertIsOnMainThread(); NS_ConvertUTF16toUTF8 url(mURL); nsIPrincipal* urlPrincipal = nsHostObjectProtocolHandler::GetDataEntryPrincipal(url); nsCOMPtr principal = mWorkerPrivate->GetPrincipal(); bool subsumes; if (urlPrincipal && NS_SUCCEEDED(principal->Subsumes(urlPrincipal, &subsumes)) && subsumes) { nsHostObjectProtocolHandler::RemoveDataEntry(url); } if (!mWorkerPrivate->IsSharedWorker() && !mWorkerPrivate->IsServiceWorker()) { // Walk up to top worker object. WorkerPrivate* wp = mWorkerPrivate; while (WorkerPrivate* parent = wp->GetParent()) { wp = parent; } nsCOMPtr sc = wp->GetScriptContext(); // We could not have a ScriptContext in JSM code. In this case, we leak. if (sc) { nsCOMPtr global = sc->GetGlobalObject(); MOZ_ASSERT(global); global->UnregisterHostObjectURI(url); } } return true; } }; // This class checks if an URL is valid on the main thread. class IsValidURLRunnable : public WorkerMainThreadRunnable { private: const nsString mURL; bool mValid; public: IsValidURLRunnable(WorkerPrivate* aWorkerPrivate, const nsAString& aURL) : WorkerMainThreadRunnable(aWorkerPrivate, NS_LITERAL_CSTRING("URL :: IsValidURL")) , mURL(aURL) , mValid(false) {} bool MainThreadRun() override { AssertIsOnMainThread(); NS_ConvertUTF16toUTF8 url(mURL); mValid = nsHostObjectProtocolHandler::HasDataEntry(url); return true; } bool IsValidURL() const { return mValid; } }; // This class creates a URL object on the main thread. class ConstructorRunnable : public WorkerMainThreadRunnable { private: const nsString mURL; nsString mBase; // IsVoid() if we have no base URI string. RefPtr mRetval; public: ConstructorRunnable(WorkerPrivate* aWorkerPrivate, const nsAString& aURL, const Optional& aBase) : WorkerMainThreadRunnable(aWorkerPrivate, NS_LITERAL_CSTRING("URL :: Constructor")) , mURL(aURL) { if (aBase.WasPassed()) { mBase = aBase.Value(); } else { mBase.SetIsVoid(true); } mWorkerPrivate->AssertIsOnWorkerThread(); } bool MainThreadRun() override { AssertIsOnMainThread(); ErrorResult rv; RefPtr url; if (!mBase.IsVoid()) { url = URLMainThread::Constructor(nullptr, mURL, mBase, rv); } else { url = URLMainThread::Constructor(nullptr, mURL, nullptr, rv); } if (rv.Failed()) { rv.SuppressException(); return true; } mRetval = new URLWorker::URLProxy(url.forget()); return true; } URLWorker::URLProxy* GetURLProxy(ErrorResult& aRv) const { MOZ_ASSERT(mWorkerPrivate); mWorkerPrivate->AssertIsOnWorkerThread(); if (!mRetval) { aRv.ThrowTypeError(mURL); } return mRetval; } }; class TeardownURLRunnable : public Runnable { public: explicit TeardownURLRunnable(URLWorker::URLProxy* aURLProxy) : Runnable("dom::TeardownURLRunnable") , mURLProxy(aURLProxy) { } NS_IMETHOD Run() override { AssertIsOnMainThread(); mURLProxy->ReleaseURI(); mURLProxy = nullptr; return NS_OK; } private: RefPtr mURLProxy; }; // This class is the generic getter for any URL property. class GetterRunnable : public WorkerMainThreadRunnable { public: enum GetterType { GetterHref, GetterOrigin, GetterProtocol, GetterUsername, GetterPassword, GetterHost, GetterHostname, GetterPort, GetterPathname, GetterSearch, GetterHash, }; GetterRunnable(WorkerPrivate* aWorkerPrivate, GetterType aType, nsAString& aValue, URLWorker::URLProxy* aURLProxy) : WorkerMainThreadRunnable(aWorkerPrivate, // We can have telemetry keys for each getter when // needed. NS_LITERAL_CSTRING("URL :: getter")) , mValue(aValue) , mType(aType) , mURLProxy(aURLProxy) { mWorkerPrivate->AssertIsOnWorkerThread(); } bool MainThreadRun() override { AssertIsOnMainThread(); ErrorResult rv; switch (mType) { case GetterHref: mURLProxy->URL()->GetHref(mValue, rv); break; case GetterOrigin: mURLProxy->URL()->GetOrigin(mValue, rv); break; case GetterProtocol: mURLProxy->URL()->GetProtocol(mValue, rv); break; case GetterUsername: mURLProxy->URL()->GetUsername(mValue, rv); break; case GetterPassword: mURLProxy->URL()->GetPassword(mValue, rv); break; case GetterHost: mURLProxy->URL()->GetHost(mValue, rv); break; case GetterHostname: mURLProxy->URL()->GetHostname(mValue, rv); break; case GetterPort: mURLProxy->URL()->GetPort(mValue, rv); break; case GetterPathname: mURLProxy->URL()->GetPathname(mValue, rv); break; case GetterSearch: mURLProxy->URL()->GetSearch(mValue, rv); break; case GetterHash: mURLProxy->URL()->GetHash(mValue, rv); break; } MOZ_ASSERT(!rv.Failed(), "Main-thread getters do not fail."); return true; } void Dispatch(ErrorResult& aRv) { WorkerMainThreadRunnable::Dispatch(Terminating, aRv); } private: nsAString& mValue; GetterType mType; RefPtr mURLProxy; }; // This class is the generic setter for any URL property. class SetterRunnable : public WorkerMainThreadRunnable { public: enum SetterType { SetterHref, SetterProtocol, SetterUsername, SetterPassword, SetterHost, SetterHostname, SetterPort, SetterPathname, SetterSearch, SetterHash, }; SetterRunnable(WorkerPrivate* aWorkerPrivate, SetterType aType, const nsAString& aValue, URLWorker::URLProxy* aURLProxy) : WorkerMainThreadRunnable(aWorkerPrivate, // We can have telemetry keys for each setter when // needed. NS_LITERAL_CSTRING("URL :: setter")) , mValue(aValue) , mType(aType) , mURLProxy(aURLProxy) , mFailed(false) { mWorkerPrivate->AssertIsOnWorkerThread(); } bool MainThreadRun() override { AssertIsOnMainThread(); ErrorResult rv; switch (mType) { case SetterHref: { mURLProxy->URL()->SetHref(mValue, rv); break; } case SetterProtocol: mURLProxy->URL()->SetProtocol(mValue, rv); break; case SetterUsername: mURLProxy->URL()->SetUsername(mValue, rv); break; case SetterPassword: mURLProxy->URL()->SetPassword(mValue, rv); break; case SetterHost: mURLProxy->URL()->SetHost(mValue, rv); break; case SetterHostname: mURLProxy->URL()->SetHostname(mValue, rv); break; case SetterPort: mURLProxy->URL()->SetPort(mValue, rv); break; case SetterPathname: mURLProxy->URL()->SetPathname(mValue, rv); break; case SetterSearch: mURLProxy->URL()->SetSearch(mValue, rv); break; case SetterHash: mURLProxy->URL()->SetHash(mValue, rv); break; } if (NS_WARN_IF(rv.Failed())) { rv.SuppressException(); mFailed = true; } return true; } bool Failed() const { return mFailed; } void Dispatch(ErrorResult& aRv) { WorkerMainThreadRunnable::Dispatch(Terminating, aRv); } private: const nsString mValue; SetterType mType; RefPtr mURLProxy; bool mFailed; }; /* static */ already_AddRefed URLWorker::Constructor(const GlobalObject& aGlobal, const nsAString& aURL, const Optional& aBase, ErrorResult& aRv) { JSContext* cx = aGlobal.Context(); WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(cx); RefPtr url = new URLWorker(workerPrivate); url->Init(aURL, aBase, aRv); return aRv.Failed() ? nullptr : url.forget(); } /* static */ already_AddRefed URLWorker::Constructor(const GlobalObject& aGlobal, const nsAString& aURL, const nsAString& aBase, ErrorResult& aRv) { Optional base; base = &aBase; return Constructor(aGlobal, aURL, base, aRv); } /* static */ void URLWorker::CreateObjectURL(const GlobalObject& aGlobal, Blob& aBlob, nsAString& aResult, mozilla::ErrorResult& aRv) { JSContext* cx = aGlobal.Context(); WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(cx); RefPtr blobImpl = aBlob.Impl(); MOZ_ASSERT(blobImpl); aRv = blobImpl->SetMutable(false); if (NS_WARN_IF(aRv.Failed())) { return; } RefPtr runnable = new CreateURLRunnable(workerPrivate, blobImpl, aResult); runnable->Dispatch(Terminating, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } if (workerPrivate->IsSharedWorker() || workerPrivate->IsServiceWorker()) { WorkerGlobalScope* scope = workerPrivate->GlobalScope(); MOZ_ASSERT(scope); scope->RegisterHostObjectURI(NS_ConvertUTF16toUTF8(aResult)); } } /* static */ void URLWorker::RevokeObjectURL(const GlobalObject& aGlobal, const nsAString& aUrl, ErrorResult& aRv) { JSContext* cx = aGlobal.Context(); WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(cx); RefPtr runnable = new RevokeURLRunnable(workerPrivate, aUrl); runnable->Dispatch(Terminating, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } if (workerPrivate->IsSharedWorker() || workerPrivate->IsServiceWorker()) { WorkerGlobalScope* scope = workerPrivate->GlobalScope(); MOZ_ASSERT(scope); scope->UnregisterHostObjectURI(NS_ConvertUTF16toUTF8(aUrl)); } } /* static */ bool URLWorker::IsValidURL(const GlobalObject& aGlobal, const nsAString& aUrl, ErrorResult& aRv) { JSContext* cx = aGlobal.Context(); WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(cx); RefPtr runnable = new IsValidURLRunnable(workerPrivate, aUrl); runnable->Dispatch(Terminating, aRv); if (NS_WARN_IF(aRv.Failed())) { return false; } return runnable->IsValidURL(); } URLWorker::URLWorker(WorkerPrivate* aWorkerPrivate) : URL(nullptr) , mWorkerPrivate(aWorkerPrivate) {} void URLWorker::Init(const nsAString& aURL, const Optional& aBase, ErrorResult& aRv) { nsAutoCString scheme; nsresult rv = net_ExtractURLScheme(NS_ConvertUTF16toUTF8(aURL), scheme); if (NS_FAILED(rv)) { // this may be a relative URL, check baseURL if (!aBase.WasPassed()) { aRv.ThrowTypeError(aURL); return; } rv = net_ExtractURLScheme(NS_ConvertUTF16toUTF8(aBase.Value()), scheme); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.ThrowTypeError(aURL); return; } } if (scheme.EqualsLiteral("http") || scheme.EqualsLiteral("https")) { nsCOMPtr baseURL; if (aBase.WasPassed()) { // XXXcatalinb: SetSpec only writes a warning to the console on urls // without a valid scheme. I can't fix that because we've come to rely // on that behaviour in a bunch of different places. nsresult rv = NS_MutateURI(new nsStandardURL::Mutator()) .SetSpec(NS_ConvertUTF16toUTF8(aBase.Value())) .Finalize(baseURL); nsAutoCString baseScheme; if (baseURL) { baseURL->GetScheme(baseScheme); } if (NS_WARN_IF(NS_FAILED(rv)) || baseScheme.IsEmpty()) { aRv.ThrowTypeError(aBase.Value()); return; } } nsCOMPtr uri; rv = NS_MutateURI(new nsStandardURL::Mutator()) .Apply(&nsIStandardURLMutator::Init, nsIStandardURL::URLTYPE_STANDARD, -1, NS_ConvertUTF16toUTF8(aURL), nullptr, baseURL, nullptr) .Finalize(uri); aRv = rv; if (NS_SUCCEEDED(rv)) { mStdURL = static_cast(uri.get()); } return; } // create url proxy RefPtr runnable = new ConstructorRunnable(mWorkerPrivate, aURL, aBase); runnable->Dispatch(Terminating, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } mURLProxy = runnable->GetURLProxy(aRv); } URLWorker::~URLWorker() { if (mURLProxy) { mWorkerPrivate->AssertIsOnWorkerThread(); RefPtr runnable = new TeardownURLRunnable(mURLProxy); mURLProxy = nullptr; if (NS_FAILED(NS_DispatchToMainThread(runnable))) { NS_ERROR("Failed to dispatch teardown runnable!"); } } } void URLWorker::GetHref(nsAString& aHref, ErrorResult& aRv) const { aHref.Truncate(); if (mStdURL) { nsAutoCString href; aRv = mStdURL->GetSpec(href); if (!aRv.Failed()) { CopyUTF8toUTF16(href, aHref); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterHref, aHref, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetHref(const nsAString& aHref, ErrorResult& aRv) { nsAutoCString scheme; nsresult rv = net_ExtractURLScheme(NS_ConvertUTF16toUTF8(aHref), scheme); if (NS_FAILED(rv)) { aRv.ThrowTypeError(aHref); return; } if (scheme.EqualsLiteral("http") || scheme.EqualsLiteral("https")) { nsCOMPtr uri; aRv = NS_MutateURI(new nsStandardURL::Mutator()) .SetSpec(NS_ConvertUTF16toUTF8(aHref)) .Finalize(uri); mStdURL = static_cast(uri.get()); if (mURLProxy) { mWorkerPrivate->AssertIsOnWorkerThread(); RefPtr runnable = new TeardownURLRunnable(mURLProxy); mURLProxy = nullptr; if (NS_WARN_IF(NS_FAILED(NS_DispatchToMainThread(runnable)))) { return; } } UpdateURLSearchParams(); return; } mStdURL = nullptr; // fallback to using a main thread url proxy if (mURLProxy) { RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterHref, aHref, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } UpdateURLSearchParams(); return; } // create the proxy now RefPtr runnable = new ConstructorRunnable(mWorkerPrivate, aHref, Optional()); runnable->Dispatch(Terminating, aRv); if (NS_WARN_IF(aRv.Failed())) { return; } mURLProxy = runnable->GetURLProxy(aRv); UpdateURLSearchParams(); } void URLWorker::GetOrigin(nsAString& aOrigin, ErrorResult& aRv) const { if (mStdURL) { nsContentUtils::GetUTFOrigin(mStdURL, aOrigin); return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterOrigin, aOrigin, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::GetProtocol(nsAString& aProtocol, ErrorResult& aRv) const { aProtocol.Truncate(); nsAutoCString protocol; if (mStdURL) { if (NS_SUCCEEDED(mStdURL->GetScheme(protocol))) { CopyASCIItoUTF16(protocol, aProtocol); aProtocol.Append(char16_t(':')); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterProtocol, aProtocol, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetProtocol(const nsAString& aProtocol, ErrorResult& aRv) { if (mStdURL) { nsAString::const_iterator start, end; aProtocol.BeginReading(start); aProtocol.EndReading(end); nsAString::const_iterator iter(start); FindCharInReadable(':', iter, end); nsresult rv = mStdURL->SetScheme(NS_ConvertUTF16toUTF8(Substring(start, iter))); if (NS_WARN_IF(NS_FAILED(rv))) { return; } nsAutoCString href; rv = mStdURL->GetSpec(href); if (NS_WARN_IF(NS_FAILED(rv))) { return; } SetHref(NS_ConvertUTF8toUTF16(href), aRv); return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterProtocol, aProtocol, mURLProxy); runnable->Dispatch(aRv); MOZ_ASSERT(!runnable->Failed()); } #define STDURL_GETTER(value, method) \ if (mStdURL) { \ value.Truncate(); \ nsAutoCString tmp; \ nsresult rv = mStdURL->method(tmp); \ if (NS_SUCCEEDED(rv)) { \ CopyUTF8toUTF16(tmp, value); \ } \ return; \ } #define STDURL_SETTER(value, method) \ if (mStdURL) { \ aRv = mStdURL->method(NS_ConvertUTF16toUTF8(value)); \ return; \ } void URLWorker::GetUsername(nsAString& aUsername, ErrorResult& aRv) const { STDURL_GETTER(aUsername, GetUsername); MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterUsername, aUsername, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetUsername(const nsAString& aUsername, ErrorResult& aRv) { STDURL_SETTER(aUsername, SetUsername); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterUsername, aUsername, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetPassword(nsAString& aPassword, ErrorResult& aRv) const { STDURL_GETTER(aPassword, GetPassword); MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterPassword, aPassword, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetPassword(const nsAString& aPassword, ErrorResult& aRv) { STDURL_SETTER(aPassword, SetPassword); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterPassword, aPassword, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetHost(nsAString& aHost, ErrorResult& aRv) const { STDURL_GETTER(aHost, GetHostPort); MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterHost, aHost, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetHost(const nsAString& aHost, ErrorResult& aRv) { STDURL_SETTER(aHost, SetHostPort); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterHost, aHost, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetHostname(nsAString& aHostname, ErrorResult& aRv) const { aHostname.Truncate(); if (mStdURL) { aRv = nsContentUtils::GetHostOrIPv6WithBrackets(mStdURL, aHostname); return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterHostname, aHostname, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetHostname(const nsAString& aHostname, ErrorResult& aRv) { STDURL_SETTER(aHostname, SetHost); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterHostname, aHostname, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetPort(nsAString& aPort, ErrorResult& aRv) const { aPort.Truncate(); if (mStdURL) { int32_t port; nsresult rv = mStdURL->GetPort(&port); if (NS_SUCCEEDED(rv) && port != -1) { nsAutoString portStr; portStr.AppendInt(port, 10); aPort.Assign(portStr); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterPort, aPort, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetPort(const nsAString& aPort, ErrorResult& aRv) { if (mStdURL) { nsresult rv; nsAutoString portStr(aPort); int32_t port = -1; // nsIURI uses -1 as default value. if (!portStr.IsEmpty()) { port = portStr.ToInteger(&rv); if (NS_FAILED(rv)) { return; } } mStdURL->SetPort(port); return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterPort, aPort, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetPathname(nsAString& aPathname, ErrorResult& aRv) const { aPathname.Truncate(); if (mStdURL) { nsAutoCString file; nsresult rv = mStdURL->GetFilePath(file); if (NS_SUCCEEDED(rv)) { CopyUTF8toUTF16(file, aPathname); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterPathname, aPathname, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetPathname(const nsAString& aPathname, ErrorResult& aRv) { STDURL_SETTER(aPathname, SetFilePath); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterPathname, aPathname, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::GetSearch(nsAString& aSearch, ErrorResult& aRv) const { aSearch.Truncate(); if (mStdURL) { nsAutoCString search; nsresult rv; rv = mStdURL->GetQuery(search); if (NS_SUCCEEDED(rv) && !search.IsEmpty()) { CopyUTF8toUTF16(NS_LITERAL_CSTRING("?") + search, aSearch); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterSearch, aSearch, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::GetHash(nsAString& aHash, ErrorResult& aRv) const { aHash.Truncate(); if (mStdURL) { nsAutoCString ref; nsresult rv = mStdURL->GetRef(ref); if (NS_SUCCEEDED(rv) && !ref.IsEmpty()) { aHash.Assign(char16_t('#')); AppendUTF8toUTF16(ref, aHash); } return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new GetterRunnable(mWorkerPrivate, GetterRunnable::GetterHash, aHash, mURLProxy); runnable->Dispatch(aRv); } void URLWorker::SetHash(const nsAString& aHash, ErrorResult& aRv) { STDURL_SETTER(aHash, SetRef); MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterHash, aHash, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::SetSearchInternal(const nsAString& aSearch, ErrorResult& aRv) { if (mStdURL) { // URLMainThread ignores failures here. mStdURL->SetQuery(NS_ConvertUTF16toUTF8(aSearch)); return; } MOZ_ASSERT(mURLProxy); RefPtr runnable = new SetterRunnable(mWorkerPrivate, SetterRunnable::SetterSearch, aSearch, mURLProxy); runnable->Dispatch(aRv); if (NS_WARN_IF(aRv.Failed())) { return; } MOZ_ASSERT(!runnable->Failed()); } void URLWorker::UpdateURLSearchParams() { if (mSearchParams) { nsAutoString search; ErrorResult rv; GetSearch(search, rv); if (NS_WARN_IF(rv.Failed())) { rv.SuppressException(); } mSearchParams->ParseInput(NS_ConvertUTF16toUTF8(Substring(search, 1))); } } } // namespace dom } // namespace mozilla