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

Unified Diff: third_party/WebKit/Source/modules/permissions/Permissions.cpp

Issue 2255933002: Add PermissionDescriptor to the permissions Mojo interface. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@no_notification_dispatcher
Patch Set: Demonstrate PermissionDescriptor extensibility with MIDI. Created 4 years, 4 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: third_party/WebKit/Source/modules/permissions/Permissions.cpp
diff --git a/third_party/WebKit/Source/modules/permissions/Permissions.cpp b/third_party/WebKit/Source/modules/permissions/Permissions.cpp
index 08781a8b9649b49d8dfccc72a4298592df79b773..9851dbfd1fc7c7e1e8be8a2f504861eb9352bcb7 100644
--- a/third_party/WebKit/Source/modules/permissions/Permissions.cpp
+++ b/third_party/WebKit/Source/modules/permissions/Permissions.cpp
@@ -28,70 +28,64 @@
namespace blink {
+using mojom::blink::PermissionDescriptorPtr;
using mojom::blink::PermissionName;
using mojom::blink::PermissionService;
namespace {
-// Websites will be able to run code when `name()` is called, changing the
-// current context. The caller should make sure that no assumption is made
-// after this has been called.
-PermissionName getPermissionName(ScriptState* scriptState, const Dictionary& rawPermission, const PermissionDescriptor& permission, ExceptionState& exceptionState)
-{
- const String& name = permission.name();
- if (name == "geolocation")
- return PermissionName::GEOLOCATION;
- if (name == "notifications")
- return PermissionName::NOTIFICATIONS;
- if (name == "push")
- return PermissionName::PUSH_NOTIFICATIONS;
- if (name == "midi") {
- MidiPermissionDescriptor midiPermission = NativeValueTraits<MidiPermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
- return midiPermission.sysex() ? PermissionName::MIDI_SYSEX : PermissionName::MIDI;
- }
- if (name == "background-sync")
- return PermissionName::BACKGROUND_SYNC;
-
- ASSERT_NOT_REACHED();
- return PermissionName::GEOLOCATION;
-}
-
-// Parses the raw permission dictionary and returns the PermissionType if
-// parsing was successful. If an exception occurs, it will be stored in
-// |exceptionState| and null will be returned. Therefore, the |exceptionState|
-// should be checked before attempting to use the returned permission as the
-// non-null assert will be fired otherwise.
-Nullable<PermissionName> parsePermission(ScriptState* scriptState, const Dictionary rawPermission, ExceptionState& exceptionState)
-{
- PermissionDescriptor permission = NativeValueTraits<PermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
-
- if (exceptionState.hadException()) {
- exceptionState.throwTypeError(exceptionState.message());
- return Nullable<PermissionName>();
- }
+ // Parses the raw permission dictionary and returns the Mojo
+ // PermissionDescriptor if parsing was successful. If an exception occurs, it
+ // will be stored in |exceptionState| and null will be returned. Therefore, the
+ // |exceptionState| should be checked before attempting to use the returned
+ // permission as the non-null assert will be fired otherwise.
+ //
+ // Websites will be able to run code when `name()` is called, changing the
+ // current context. The caller should make sure that no assumption is made
+ // after this has been called.
+ PermissionDescriptorPtr parsePermission(ScriptState* scriptState, const Dictionary rawPermission, ExceptionState& exceptionState)
+ {
+ PermissionDescriptor permission = NativeValueTraits<PermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
- PermissionName name = getPermissionName(scriptState, rawPermission, permission, exceptionState);
- if (exceptionState.hadException()) {
- exceptionState.throwTypeError(exceptionState.message());
- return Nullable<PermissionName>();
- }
-
- // Here we reject any permissions which are not yet supported by Blink.
- if (name == PermissionName::PUSH_NOTIFICATIONS) {
- PushPermissionDescriptor pushPermission = NativeValueTraits<PushPermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
if (exceptionState.hadException()) {
exceptionState.throwTypeError(exceptionState.message());
- return Nullable<PermissionName>();
+ return nullptr;
}
- // Only "userVisibleOnly" push is supported for now.
- if (!pushPermission.userVisibleOnly()) {
- exceptionState.throwDOMException(NotSupportedError, "Push Permission without userVisibleOnly:true isn't supported yet.");
- return Nullable<PermissionName>();
+ auto permissionDescriptor = mojom::blink::PermissionDescriptor::New();
+ const String& name = permission.name();
+ if (name == "geolocation") {
+ permissionDescriptor->name = PermissionName::GEOLOCATION;
+ } else if (name == "notifications") {
+ permissionDescriptor->name = PermissionName::NOTIFICATIONS;
+ } else if (name == "push") {
+ PushPermissionDescriptor pushPermission = NativeValueTraits<PushPermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
+ if (exceptionState.hadException()) {
+ exceptionState.throwTypeError(exceptionState.message());
+ return nullptr;
+ }
+
+ // Only "userVisibleOnly" push is supported for now.
+ if (!pushPermission.userVisibleOnly()) {
+ exceptionState.throwDOMException(NotSupportedError, "Push Permission without userVisibleOnly:true isn't supported yet.");
+ return nullptr;
+ }
+
+ permissionDescriptor->name = PermissionName::PUSH_NOTIFICATIONS;
+ } else if (name == "midi") {
+ MidiPermissionDescriptor midiPermission = NativeValueTraits<MidiPermissionDescriptor>::nativeValue(scriptState->isolate(), rawPermission.v8Value(), exceptionState);
+ permissionDescriptor->name = PermissionName::MIDI;
+ auto midiPermissionDescriptor = mojom::blink::MidiPermissionDescriptor::New();
+ midiPermissionDescriptor->sysex = midiPermission.sysex();
+ permissionDescriptor->extension = mojom::blink::PermissionDescriptorExtension::New();
+ permissionDescriptor->extension->set_midi(std::move(midiPermissionDescriptor));
+ } else if (name == "background-sync") {
+ permissionDescriptor->name = PermissionName::BACKGROUND_SYNC;
+ } else {
+ return nullptr;
}
- }
- return Nullable<PermissionName>(name);
+ return permissionDescriptor;
}
} // anonymous namespace
@@ -116,7 +110,7 @@ bool Permissions::connectToService(ExecutionContext* executionContext, mojom::bl
ScriptPromise Permissions::query(ScriptState* scriptState, const Dictionary& rawPermission)
{
ExceptionState exceptionState(ExceptionState::GetterContext, "query", "Permissions", scriptState->context()->Global(), scriptState->isolate());
- Nullable<PermissionName> name = parsePermission(scriptState, rawPermission, exceptionState);
+ auto descriptor = parsePermission(scriptState, rawPermission, exceptionState);
if (exceptionState.hadException())
return exceptionState.reject(scriptState);
@@ -133,14 +127,15 @@ ScriptPromise Permissions::query(ScriptState* scriptState, const Dictionary& raw
// meaningful value because most APIs are broken on file scheme and no
// permission prompt will be shown even if the returned permission will most
// likely be "prompt".
- service->HasPermission(name.get(), scriptState->getExecutionContext()->getSecurityOrigin(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), name.get())));
+ auto descriptorCopy = descriptor->Clone();
+ service->HasPermission(std::move(descriptor), scriptState->getExecutionContext()->getSecurityOrigin(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), passed(std::move(descriptorCopy)))));
return promise;
}
ScriptPromise Permissions::request(ScriptState* scriptState, const Dictionary& rawPermission)
{
ExceptionState exceptionState(ExceptionState::GetterContext, "request", "Permissions", scriptState->context()->Global(), scriptState->isolate());
- Nullable<PermissionName> name = parsePermission(scriptState, rawPermission, exceptionState);
+ auto descriptor = parsePermission(scriptState, rawPermission, exceptionState);
if (exceptionState.hadException())
return exceptionState.reject(scriptState);
@@ -153,14 +148,15 @@ ScriptPromise Permissions::request(ScriptState* scriptState, const Dictionary& r
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
- service->RequestPermission(name.get(), scriptState->getExecutionContext()->getSecurityOrigin(), UserGestureIndicator::processingUserGesture(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), name.get())));
+ auto descriptorCopy = descriptor->Clone();
+ service->RequestPermission(std::move(descriptor), scriptState->getExecutionContext()->getSecurityOrigin(), UserGestureIndicator::processingUserGesture(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), passed(std::move(descriptorCopy)))));
return promise;
}
ScriptPromise Permissions::revoke(ScriptState* scriptState, const Dictionary& rawPermission)
{
ExceptionState exceptionState(ExceptionState::GetterContext, "revoke", "Permissions", scriptState->context()->Global(), scriptState->isolate());
- Nullable<PermissionName> name = parsePermission(scriptState, rawPermission, exceptionState);
+ auto descriptor = parsePermission(scriptState, rawPermission, exceptionState);
if (exceptionState.hadException())
return exceptionState.reject(scriptState);
@@ -173,32 +169,35 @@ ScriptPromise Permissions::revoke(ScriptState* scriptState, const Dictionary& ra
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
- service->RevokePermission(name.get(), scriptState->getExecutionContext()->getSecurityOrigin(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), name.get())));
+ auto descriptorCopy = descriptor->Clone();
+ service->RevokePermission(std::move(descriptor), scriptState->getExecutionContext()->getSecurityOrigin(), convertToBaseCallback(WTF::bind(&Permissions::taskComplete, wrapPersistent(this), wrapPersistent(resolver), passed(std::move(descriptorCopy)))));
return promise;
}
ScriptPromise Permissions::requestAll(ScriptState* scriptState, const Vector<Dictionary>& rawPermissions)
{
ExceptionState exceptionState(ExceptionState::GetterContext, "request", "Permissions", scriptState->context()->Global(), scriptState->isolate());
- Vector<PermissionName> internalPermissions;
+ Vector<PermissionDescriptorPtr> internalPermissions;
Vector<int> callerIndexToInternalIndex;
callerIndexToInternalIndex.resize(rawPermissions.size());
for (size_t i = 0; i < rawPermissions.size(); ++i) {
const Dictionary& rawPermission = rawPermissions[i];
- Nullable<PermissionName> name = parsePermission(scriptState, rawPermission, exceptionState);
+ auto descriptor = parsePermission(scriptState, rawPermission, exceptionState);
if (exceptionState.hadException())
return exceptionState.reject(scriptState);
- // Only append permissions to the vector that is passed to the client if it is not already
- // in the vector (i.e. do not duplicate permisison types).
- int internalIndex;
- auto it = internalPermissions.find(name.get());
- if (it == kNotFound) {
+ // Only append permissions types that are not already present in the vector.
+ int internalIndex = -1;
+ for (size_t j = 0; j < internalPermissions.size(); ++j) {
+ if (internalPermissions[j]->name == descriptor->name) {
+ internalIndex = j;
+ break;
+ }
+ }
+ if (internalIndex == -1) {
internalIndex = internalPermissions.size();
- internalPermissions.append(name.get());
- } else {
- internalIndex = it;
+ internalPermissions.append(std::move(descriptor));
}
callerIndexToInternalIndex[i] = internalIndex;
}
@@ -212,8 +211,13 @@ ScriptPromise Permissions::requestAll(ScriptState* scriptState, const Vector<Dic
ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState);
ScriptPromise promise = resolver->promise();
- service->RequestPermissions(internalPermissions, scriptState->getExecutionContext()->getSecurityOrigin(), UserGestureIndicator::processingUserGesture(),
- convertToBaseCallback(WTF::bind(&Permissions::batchTaskComplete, wrapPersistent(this), wrapPersistent(resolver), internalPermissions, callerIndexToInternalIndex)));
+ Vector<PermissionDescriptorPtr> internalPermissionsCopy;
+ internalPermissionsCopy.reserveCapacity(internalPermissions.size());
+ for (const auto& descriptor : internalPermissions)
+ internalPermissionsCopy.append(descriptor->Clone());
+
+ service->RequestPermissions(std::move(internalPermissions), scriptState->getExecutionContext()->getSecurityOrigin(), UserGestureIndicator::processingUserGesture(),
+ convertToBaseCallback(WTF::bind(&Permissions::batchTaskComplete, wrapPersistent(this), wrapPersistent(resolver), passed(std::move(internalPermissionsCopy)), passed(std::move(callerIndexToInternalIndex)))));
return promise;
}
@@ -229,14 +233,14 @@ void Permissions::serviceConnectionError()
m_service.reset();
}
-void Permissions::taskComplete(ScriptPromiseResolver* resolver, PermissionName name, mojom::blink::PermissionStatus result)
+void Permissions::taskComplete(ScriptPromiseResolver* resolver, mojom::blink::PermissionDescriptorPtr descriptor, mojom::blink::PermissionStatus result)
{
if (!resolver->getExecutionContext() || resolver->getExecutionContext()->activeDOMObjectsAreStopped())
return;
- resolver->resolve(PermissionStatus::take(resolver, result, name));
+ resolver->resolve(PermissionStatus::take(resolver, result, std::move(descriptor)));
}
-void Permissions::batchTaskComplete(ScriptPromiseResolver* resolver, Vector<PermissionName> names, Vector<int> callerIndexToInternalIndex, const Vector<mojom::blink::PermissionStatus>& results)
+void Permissions::batchTaskComplete(ScriptPromiseResolver* resolver, Vector<mojom::blink::PermissionDescriptorPtr> descriptors, Vector<int> callerIndexToInternalIndex, const Vector<mojom::blink::PermissionStatus>& results)
{
if (!resolver->getExecutionContext() || resolver->getExecutionContext()->activeDOMObjectsAreStopped())
return;
@@ -247,7 +251,7 @@ void Permissions::batchTaskComplete(ScriptPromiseResolver* resolver, Vector<Perm
HeapVector<Member<PermissionStatus>> result;
result.reserveInitialCapacity(callerIndexToInternalIndex.size());
for (int internalIndex : callerIndexToInternalIndex)
- result.append(PermissionStatus::createAndListen(resolver->getExecutionContext(), results[internalIndex], names[internalIndex]));
+ result.append(PermissionStatus::createAndListen(resolver->getExecutionContext(), results[internalIndex], descriptors[internalIndex]->Clone()));
resolver->resolve(result);
}

Powered by Google App Engine
This is Rietveld 408576698