| // Copyright 2015 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "android_webview/browser/aw_permission_manager.h" |
| |
| #include <memory> |
| #include <string> |
| #include <unordered_map> |
| #include <utility> |
| |
| #include "android_webview/browser/aw_browser_permission_request_delegate.h" |
| #include "base/containers/contains.h" |
| #include "base/functional/bind.h" |
| #include "base/functional/callback.h" |
| #include "base/logging.h" |
| #include "components/permissions/permission_util.h" |
| #include "content/public/browser/child_process_security_policy.h" |
| #include "content/public/browser/permission_controller.h" |
| #include "content/public/browser/permission_result.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/render_process_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/content_features.h" |
| #include "third_party/blink/public/common/permissions/permission_utils.h" |
| |
| using blink::PermissionType; |
| using blink::mojom::PermissionStatus; |
| |
| using RequestPermissionsCallback = |
| base::OnceCallback<void(const std::vector<PermissionStatus>&)>; |
| |
| namespace android_webview { |
| |
| class LastRequestResultCache { |
| public: |
| LastRequestResultCache() = default; |
| |
| LastRequestResultCache(const LastRequestResultCache&) = delete; |
| LastRequestResultCache& operator=(const LastRequestResultCache&) = delete; |
| |
| void SetResult(PermissionType permission, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin, |
| PermissionStatus status) { |
| DCHECK(status == PermissionStatus::GRANTED || |
| status == PermissionStatus::DENIED); |
| |
| // TODO(ddorwin): We should be denying empty origins at a higher level. |
| if (requesting_origin.is_empty() || embedding_origin.is_empty()) { |
| DLOG(WARNING) << "Not caching result because of empty origin."; |
| return; |
| } |
| |
| if (!requesting_origin.is_valid()) { |
| NOTREACHED() << requesting_origin.possibly_invalid_spec(); |
| return; |
| } |
| if (!embedding_origin.is_valid()) { |
| NOTREACHED() << embedding_origin.possibly_invalid_spec(); |
| return; |
| } |
| |
| if (permission != PermissionType::PROTECTED_MEDIA_IDENTIFIER) { |
| // Other permissions are not cached. |
| return; |
| } |
| |
| std::string key = GetCacheKey(requesting_origin, embedding_origin); |
| if (key.empty()) { |
| NOTREACHED(); |
| // Never store an empty key because it could inadvertently be used for |
| // another combination. |
| return; |
| } |
| pmi_result_cache_[key] = status; |
| } |
| |
| PermissionStatus GetResult(PermissionType permission, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin) const { |
| // TODO(ddorwin): We should be denying empty origins at a higher level. |
| if (requesting_origin.is_empty() || embedding_origin.is_empty()) { |
| return PermissionStatus::ASK; |
| } |
| |
| DCHECK(requesting_origin.is_valid()) |
| << requesting_origin.possibly_invalid_spec(); |
| DCHECK(embedding_origin.is_valid()) |
| << embedding_origin.possibly_invalid_spec(); |
| |
| if (permission != PermissionType::PROTECTED_MEDIA_IDENTIFIER) { |
| NOTREACHED() << "Results are only cached for PROTECTED_MEDIA_IDENTIFIER"; |
| return PermissionStatus::ASK; |
| } |
| |
| std::string key = GetCacheKey(requesting_origin, embedding_origin); |
| StatusMap::const_iterator it = pmi_result_cache_.find(key); |
| if (it == pmi_result_cache_.end()) { |
| DLOG(WARNING) << "GetResult() called for uncached origins: " << key; |
| return PermissionStatus::ASK; |
| } |
| |
| DCHECK(!key.empty()); |
| return it->second; |
| } |
| |
| void ClearResult(PermissionType permission, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin) { |
| // TODO(ddorwin): We should be denying empty origins at a higher level. |
| if (requesting_origin.is_empty() || embedding_origin.is_empty()) { |
| return; |
| } |
| |
| DCHECK(requesting_origin.is_valid()) |
| << requesting_origin.possibly_invalid_spec(); |
| DCHECK(embedding_origin.is_valid()) |
| << embedding_origin.possibly_invalid_spec(); |
| |
| |
| if (permission != PermissionType::PROTECTED_MEDIA_IDENTIFIER) { |
| // Other permissions are not cached, so nothing to clear. |
| return; |
| } |
| |
| std::string key = GetCacheKey(requesting_origin, embedding_origin); |
| pmi_result_cache_.erase(key); |
| } |
| |
| private: |
| // Returns a concatenation of the origins to be used as the index. |
| // Returns the empty string if either origin is invalid or empty. |
| static std::string GetCacheKey(const GURL& requesting_origin, |
| const GURL& embedding_origin) { |
| const std::string& requesting = requesting_origin.spec(); |
| const std::string& embedding = embedding_origin.spec(); |
| if (requesting.empty() || embedding.empty()) |
| return std::string(); |
| return requesting + "," + embedding; |
| } |
| |
| using StatusMap = std::unordered_map<std::string, PermissionStatus>; |
| StatusMap pmi_result_cache_; |
| }; |
| |
| class AwPermissionManager::PendingRequest { |
| public: |
| PendingRequest(const std::vector<PermissionType> permissions, |
| GURL requesting_origin, |
| GURL embedding_origin, |
| int render_process_id, |
| int render_frame_id, |
| RequestPermissionsCallback callback) |
| : permissions(permissions), |
| requesting_origin(requesting_origin), |
| embedding_origin(embedding_origin), |
| render_process_id(render_process_id), |
| render_frame_id(render_frame_id), |
| callback(std::move(callback)), |
| results(permissions.size(), PermissionStatus::DENIED), |
| cancelled_(false) { |
| for (size_t i = 0; i < permissions.size(); ++i) |
| permission_index_map_.insert(std::make_pair(permissions[i], i)); |
| } |
| |
| ~PendingRequest() = default; |
| |
| void SetPermissionStatus(PermissionType type, PermissionStatus status) { |
| auto result = permission_index_map_.find(type); |
| if (result == permission_index_map_.end()) { |
| NOTREACHED(); |
| return; |
| } |
| DCHECK(!IsCompleted()); |
| results[result->second] = status; |
| if (base::FeatureList::IsEnabled(features::kBlockMidiByDefault)) { |
| if (type == PermissionType::MIDI && status == PermissionStatus::GRANTED) { |
| content::ChildProcessSecurityPolicy::GetInstance() |
| ->GrantSendMidiMessage(render_process_id); |
| } |
| } |
| if (type == PermissionType::MIDI_SYSEX && |
| status == PermissionStatus::GRANTED) { |
| content::ChildProcessSecurityPolicy::GetInstance() |
| ->GrantSendMidiSysExMessage(render_process_id); |
| } |
| resolved_permissions_.insert(type); |
| } |
| |
| PermissionStatus GetPermissionStatus(PermissionType type) { |
| auto result = permission_index_map_.find(type); |
| if (result == permission_index_map_.end()) { |
| NOTREACHED(); |
| return PermissionStatus::DENIED; |
| } |
| return results[result->second]; |
| } |
| |
| bool HasPermissionType(PermissionType type) { |
| return base::Contains(permission_index_map_, type); |
| } |
| |
| bool IsCompleted() const { |
| return results.size() == resolved_permissions_.size(); |
| } |
| |
| bool IsCompleted(PermissionType type) const { |
| return resolved_permissions_.count(type) != 0; |
| } |
| |
| void Cancel() { cancelled_ = true; } |
| |
| bool IsCancelled() const { return cancelled_; } |
| |
| std::vector<PermissionType> permissions; |
| GURL requesting_origin; |
| GURL embedding_origin; |
| int render_process_id; |
| int render_frame_id; |
| RequestPermissionsCallback callback; |
| std::vector<PermissionStatus> results; |
| |
| private: |
| std::map<PermissionType, size_t> permission_index_map_; |
| std::set<PermissionType> resolved_permissions_; |
| bool cancelled_; |
| }; |
| |
| AwPermissionManager::AwPermissionManager() |
| : result_cache_(new LastRequestResultCache) {} |
| |
| AwPermissionManager::~AwPermissionManager() { |
| CancelPermissionRequests(); |
| } |
| |
| void AwPermissionManager::RequestPermissions( |
| content::RenderFrameHost* render_frame_host, |
| const content::PermissionRequestDescription& request_description, |
| base::OnceCallback<void(const std::vector<PermissionStatus>&)> callback) { |
| auto const& permissions = request_description.permissions; |
| if (permissions.empty()) { |
| std::move(callback).Run(std::vector<PermissionStatus>()); |
| return; |
| } |
| |
| const GURL& embedding_origin = LastCommittedOrigin(render_frame_host); |
| const GURL& requesting_origin = request_description.requesting_origin; |
| |
| auto pending_request = std::make_unique<PendingRequest>( |
| permissions, requesting_origin, embedding_origin, |
| GetRenderProcessID(render_frame_host), |
| GetRenderFrameID(render_frame_host), std::move(callback)); |
| std::vector<bool> should_delegate_requests = |
| std::vector<bool>(permissions.size(), true); |
| for (size_t i = 0; i < permissions.size(); ++i) { |
| for (PendingRequestsMap::Iterator<PendingRequest> it(&pending_requests_); |
| !it.IsAtEnd(); it.Advance()) { |
| if (it.GetCurrentValue()->HasPermissionType(permissions[i]) && |
| it.GetCurrentValue()->requesting_origin == requesting_origin) { |
| if (it.GetCurrentValue()->IsCompleted(permissions[i])) { |
| pending_request->SetPermissionStatus( |
| permissions[i], |
| it.GetCurrentValue()->GetPermissionStatus(permissions[i])); |
| } |
| should_delegate_requests[i] = false; |
| break; |
| } |
| } |
| } |
| |
| // Keep copy of pointer for performing further operations after ownership is |
| // transferred to pending_requests_ |
| PendingRequest* pending_request_raw = pending_request.get(); |
| const int request_id = pending_requests_.Add(std::move(pending_request)); |
| |
| AwBrowserPermissionRequestDelegate* delegate = |
| GetDelegate(pending_request_raw->render_process_id, |
| pending_request_raw->render_frame_id); |
| |
| for (size_t i = 0; i < permissions.size(); ++i) { |
| if (!should_delegate_requests[i]) |
| continue; |
| |
| if (!delegate) { |
| DVLOG(0) << "Dropping permissions request for " |
| << static_cast<int>(permissions[i]); |
| pending_request_raw->SetPermissionStatus(permissions[i], |
| PermissionStatus::DENIED); |
| continue; |
| } |
| |
| switch (permissions[i]) { |
| case PermissionType::GEOLOCATION: |
| delegate->RequestGeolocationPermission( |
| pending_request_raw->requesting_origin, |
| base::BindOnce(&OnRequestResponse, weak_ptr_factory_.GetWeakPtr(), |
| request_id, permissions[i])); |
| break; |
| case PermissionType::PROTECTED_MEDIA_IDENTIFIER: |
| delegate->RequestProtectedMediaIdentifierPermission( |
| pending_request_raw->requesting_origin, |
| base::BindOnce(&OnRequestResponse, weak_ptr_factory_.GetWeakPtr(), |
| request_id, permissions[i])); |
| break; |
| case PermissionType::MIDI_SYSEX: |
| delegate->RequestMIDISysexPermission( |
| pending_request_raw->requesting_origin, |
| base::BindOnce(&OnRequestResponse, weak_ptr_factory_.GetWeakPtr(), |
| request_id, permissions[i])); |
| break; |
| case PermissionType::CLIPBOARD_SANITIZED_WRITE: |
| // This is the permission for writing vetted data (such as plain text or |
| // sanitized images) using the async clipboard API. Chrome automatically |
| // grants access with a user gesture, and alternatively queries for |
| // gesture-less access with a popup bubble. For now, just grant based on |
| // user gesture. |
| // Reading from the clipboard or writing custom data is represented with |
| // the CLIPBOARD_READ_WRITE permission, and that requires an explicit |
| // user approval, which is not implemented yet. See crbug.com/1271620 |
| pending_request_raw->SetPermissionStatus( |
| permissions[i], request_description.user_gesture |
| ? PermissionStatus::GRANTED |
| : PermissionStatus::DENIED); |
| break; |
| case PermissionType::AUDIO_CAPTURE: |
| case PermissionType::VIDEO_CAPTURE: |
| case PermissionType::NOTIFICATIONS: |
| case PermissionType::DURABLE_STORAGE: |
| case PermissionType::BACKGROUND_SYNC: |
| case PermissionType::ACCESSIBILITY_EVENTS: |
| case PermissionType::CLIPBOARD_READ_WRITE: |
| case PermissionType::PAYMENT_HANDLER: |
| case PermissionType::BACKGROUND_FETCH: |
| case PermissionType::IDLE_DETECTION: |
| case PermissionType::PERIODIC_BACKGROUND_SYNC: |
| case PermissionType::NFC: |
| case PermissionType::VR: |
| case PermissionType::AR: |
| case PermissionType::STORAGE_ACCESS_GRANT: |
| case PermissionType::TOP_LEVEL_STORAGE_ACCESS: |
| case PermissionType::CAMERA_PAN_TILT_ZOOM: |
| case PermissionType::WINDOW_MANAGEMENT: |
| case PermissionType::LOCAL_FONTS: |
| case PermissionType::DISPLAY_CAPTURE: |
| case PermissionType::CAPTURED_SURFACE_CONTROL: |
| case PermissionType::SMART_CARD: |
| case PermissionType::WEB_PRINTING: |
| NOTIMPLEMENTED() << "RequestPermissions is not implemented for " |
| << static_cast<int>(permissions[i]); |
| pending_request_raw->SetPermissionStatus(permissions[i], |
| PermissionStatus::DENIED); |
| break; |
| case PermissionType::MIDI: |
| case PermissionType::SENSORS: |
| case PermissionType::WAKE_LOCK_SCREEN: |
| // PermissionType::SENSORS requests are always granted so that access |
| // to device motion and device orientation data (and underlying |
| // sensors) works in the WebView. SensorProviderImpl::GetSensor() |
| // filters requests for other types of sensors. |
| pending_request_raw->SetPermissionStatus(permissions[i], |
| PermissionStatus::GRANTED); |
| break; |
| case PermissionType::WAKE_LOCK_SYSTEM: |
| pending_request_raw->SetPermissionStatus(permissions[i], |
| PermissionStatus::DENIED); |
| break; |
| case PermissionType::NUM: |
| NOTREACHED() << "PermissionType::NUM was not expected here."; |
| pending_request_raw->SetPermissionStatus(permissions[i], |
| PermissionStatus::DENIED); |
| break; |
| } |
| } |
| |
| // If delegate resolve the permission synchronously, all requests could be |
| // already resolved here. |
| if (!pending_requests_.Lookup(request_id)) |
| return; |
| |
| // If requests are resolved without calling delegate functions, e.g. |
| // PermissionType::MIDI is permitted within the previous for-loop, all |
| // requests could be already resolved, but still in the |pending_requests_| |
| // without invoking the callback. |
| if (pending_request_raw->IsCompleted()) { |
| std::vector<PermissionStatus> results = pending_request_raw->results; |
| RequestPermissionsCallback completed_callback = |
| std::move(pending_request_raw->callback); |
| pending_requests_.Remove(request_id); |
| std::move(completed_callback).Run(results); |
| } |
| } |
| |
| // static |
| void AwPermissionManager::OnRequestResponse( |
| const base::WeakPtr<AwPermissionManager>& manager, |
| int request_id, |
| PermissionType permission, |
| bool allowed) { |
| // All delegate functions should be cancelled when the manager runs |
| // destructor. Therefore |manager| should be always valid here. |
| DCHECK(manager); |
| |
| PermissionStatus status = |
| allowed ? PermissionStatus::GRANTED : PermissionStatus::DENIED; |
| PendingRequest* pending_request = |
| manager->pending_requests_.Lookup(request_id); |
| |
| manager->result_cache_->SetResult(permission, |
| pending_request->requesting_origin, |
| pending_request->embedding_origin, status); |
| |
| std::vector<int> complete_request_ids; |
| std::vector< |
| std::pair<RequestPermissionsCallback, std::vector<PermissionStatus>>> |
| complete_request_pairs; |
| for (PendingRequestsMap::Iterator<PendingRequest> it( |
| &manager->pending_requests_); |
| !it.IsAtEnd(); it.Advance()) { |
| if (!it.GetCurrentValue()->HasPermissionType(permission) || |
| it.GetCurrentValue()->requesting_origin != |
| pending_request->requesting_origin) { |
| continue; |
| } |
| it.GetCurrentValue()->SetPermissionStatus(permission, status); |
| if (it.GetCurrentValue()->IsCompleted()) { |
| complete_request_ids.push_back(it.GetCurrentKey()); |
| if (!it.GetCurrentValue()->IsCancelled()) { |
| complete_request_pairs.emplace_back( |
| std::move(it.GetCurrentValue()->callback), |
| std::move(it.GetCurrentValue()->results)); |
| } |
| } |
| } |
| for (auto id : complete_request_ids) |
| manager->pending_requests_.Remove(id); |
| for (auto& pair : complete_request_pairs) |
| std::move(pair.first).Run(pair.second); |
| } |
| |
| void AwPermissionManager::ResetPermission(PermissionType permission, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin) { |
| result_cache_->ClearResult(permission, requesting_origin, embedding_origin); |
| } |
| |
| void AwPermissionManager::RequestPermissionsFromCurrentDocument( |
| content::RenderFrameHost* render_frame_host, |
| const content::PermissionRequestDescription& request_description, |
| base::OnceCallback<void(const std::vector<blink::mojom::PermissionStatus>&)> |
| callback) { |
| RequestPermissions(render_frame_host, request_description, |
| std::move(callback)); |
| } |
| |
| PermissionStatus AwPermissionManager::GetPermissionStatus( |
| PermissionType permission, |
| const GURL& requesting_origin, |
| const GURL& embedding_origin) { |
| // Method is called outside the Permissions API only for this permission. |
| if (permission == PermissionType::PROTECTED_MEDIA_IDENTIFIER) { |
| return result_cache_->GetResult(permission, requesting_origin, |
| embedding_origin); |
| } else if (permission == PermissionType::MIDI || |
| permission == PermissionType::SENSORS) { |
| return PermissionStatus::GRANTED; |
| } |
| |
| return PermissionStatus::DENIED; |
| } |
| |
| content::PermissionResult |
| AwPermissionManager::GetPermissionResultForOriginWithoutContext( |
| blink::PermissionType permission, |
| const url::Origin& requesting_origin, |
| const url::Origin& embedding_origin) { |
| blink::mojom::PermissionStatus status = GetPermissionStatus( |
| permission, requesting_origin.GetURL(), embedding_origin.GetURL()); |
| |
| return content::PermissionResult( |
| status, content::PermissionStatusSource::UNSPECIFIED); |
| } |
| |
| PermissionStatus AwPermissionManager::GetPermissionStatusForCurrentDocument( |
| PermissionType permission, |
| content::RenderFrameHost* render_frame_host) { |
| return GetPermissionStatus( |
| permission, |
| permissions::PermissionUtil::GetLastCommittedOriginAsURL( |
| render_frame_host), |
| permissions::PermissionUtil::GetLastCommittedOriginAsURL( |
| render_frame_host->GetMainFrame())); |
| } |
| |
| PermissionStatus AwPermissionManager::GetPermissionStatusForWorker( |
| PermissionType permission, |
| content::RenderProcessHost* render_process_host, |
| const GURL& worker_origin) { |
| return GetPermissionStatus(permission, worker_origin, worker_origin); |
| } |
| |
| PermissionStatus AwPermissionManager::GetPermissionStatusForEmbeddedRequester( |
| blink::PermissionType permission, |
| content::RenderFrameHost* render_frame_host, |
| const url::Origin& requesting_origin) { |
| return GetPermissionStatus( |
| permission, requesting_origin.GetURL(), |
| permissions::PermissionUtil::GetLastCommittedOriginAsURL( |
| render_frame_host->GetMainFrame())); |
| } |
| |
| AwPermissionManager::SubscriptionId |
| AwPermissionManager::SubscribeToPermissionStatusChange( |
| PermissionType permission, |
| content::RenderProcessHost* render_process_host, |
| content::RenderFrameHost* render_frame_host, |
| const GURL& requesting_origin, |
| base::RepeatingCallback<void(PermissionStatus)> callback) { |
| return SubscriptionId(); |
| } |
| |
| void AwPermissionManager::UnsubscribeFromPermissionStatusChange( |
| SubscriptionId subscription_id) {} |
| |
| void AwPermissionManager::CancelPermissionRequest(int request_id) { |
| PendingRequest* pending_request = pending_requests_.Lookup(request_id); |
| if (!pending_request || pending_request->IsCancelled()) |
| return; |
| pending_request->Cancel(); |
| |
| const GURL& embedding_origin = pending_request->embedding_origin; |
| const GURL& requesting_origin = pending_request->requesting_origin; |
| for (auto permission : pending_request->permissions) |
| result_cache_->ClearResult(permission, requesting_origin, embedding_origin); |
| |
| AwBrowserPermissionRequestDelegate* delegate = GetDelegate( |
| pending_request->render_process_id, pending_request->render_frame_id); |
| |
| for (auto permission : pending_request->permissions) { |
| // If the permission was already resolved, we do not need to cancel it. |
| if (pending_request->IsCompleted(permission)) |
| continue; |
| |
| // If another pending_request waits for the same permission being resolved, |
| // we should not cancel the delegate's request. |
| bool should_not_cancel_ = false; |
| for (PendingRequestsMap::Iterator<PendingRequest> it(&pending_requests_); |
| !it.IsAtEnd(); it.Advance()) { |
| if (it.GetCurrentValue() != pending_request && |
| it.GetCurrentValue()->HasPermissionType(permission) && |
| it.GetCurrentValue()->requesting_origin == requesting_origin && |
| !it.GetCurrentValue()->IsCompleted(permission)) { |
| should_not_cancel_ = true; |
| break; |
| } |
| } |
| if (should_not_cancel_) |
| continue; |
| |
| switch (permission) { |
| case PermissionType::GEOLOCATION: |
| if (delegate) |
| delegate->CancelGeolocationPermissionRequests(requesting_origin); |
| break; |
| case PermissionType::PROTECTED_MEDIA_IDENTIFIER: |
| if (delegate) |
| delegate->CancelProtectedMediaIdentifierPermissionRequests( |
| requesting_origin); |
| break; |
| case PermissionType::MIDI_SYSEX: |
| if (delegate) |
| delegate->CancelMIDISysexPermissionRequests(requesting_origin); |
| break; |
| case PermissionType::NOTIFICATIONS: |
| case PermissionType::DURABLE_STORAGE: |
| case PermissionType::AUDIO_CAPTURE: |
| case PermissionType::VIDEO_CAPTURE: |
| case PermissionType::BACKGROUND_SYNC: |
| case PermissionType::ACCESSIBILITY_EVENTS: |
| case PermissionType::CLIPBOARD_READ_WRITE: |
| case PermissionType::CLIPBOARD_SANITIZED_WRITE: |
| case PermissionType::PAYMENT_HANDLER: |
| case PermissionType::BACKGROUND_FETCH: |
| case PermissionType::IDLE_DETECTION: |
| case PermissionType::PERIODIC_BACKGROUND_SYNC: |
| case PermissionType::NFC: |
| case PermissionType::VR: |
| case PermissionType::AR: |
| case PermissionType::STORAGE_ACCESS_GRANT: |
| case PermissionType::TOP_LEVEL_STORAGE_ACCESS: |
| case PermissionType::CAMERA_PAN_TILT_ZOOM: |
| case PermissionType::WINDOW_MANAGEMENT: |
| case PermissionType::LOCAL_FONTS: |
| case PermissionType::DISPLAY_CAPTURE: |
| case PermissionType::CAPTURED_SURFACE_CONTROL: |
| case PermissionType::SMART_CARD: |
| case PermissionType::WEB_PRINTING: |
| NOTIMPLEMENTED() << "CancelPermission not implemented for " |
| << static_cast<int>(permission); |
| break; |
| case PermissionType::MIDI: |
| case PermissionType::SENSORS: |
| case PermissionType::WAKE_LOCK_SCREEN: |
| case PermissionType::WAKE_LOCK_SYSTEM: |
| // There is nothing to cancel so this is simply ignored. |
| break; |
| case PermissionType::NUM: |
| NOTREACHED() << "PermissionType::NUM was not expected here."; |
| break; |
| } |
| pending_request->SetPermissionStatus(permission, PermissionStatus::DENIED); |
| } |
| |
| // If there are still active requests, we should not remove request_id here, |
| // but just do not invoke a relevant callback when the request is resolved in |
| // OnRequestResponse(). |
| if (pending_request->IsCompleted()) |
| pending_requests_.Remove(request_id); |
| } |
| |
| void AwPermissionManager::CancelPermissionRequests() { |
| std::vector<int> request_ids; |
| for (PendingRequestsMap::Iterator<PendingRequest> it(&pending_requests_); |
| !it.IsAtEnd(); it.Advance()) { |
| request_ids.push_back(it.GetCurrentKey()); |
| } |
| for (auto request_id : request_ids) |
| CancelPermissionRequest(request_id); |
| DCHECK(pending_requests_.IsEmpty()); |
| } |
| |
| void AwPermissionManager::SetOriginCanReadEnumerateDevicesAudioLabels( |
| const url::Origin& origin, |
| bool audio) { |
| auto it = enumerate_devices_labels_cache_.find(origin); |
| if (it == enumerate_devices_labels_cache_.end()) { |
| enumerate_devices_labels_cache_[origin] = std::make_pair(audio, false); |
| } else { |
| it->second.first = audio; |
| } |
| } |
| |
| void AwPermissionManager::SetOriginCanReadEnumerateDevicesVideoLabels( |
| const url::Origin& origin, |
| bool video) { |
| auto it = enumerate_devices_labels_cache_.find(origin); |
| if (it == enumerate_devices_labels_cache_.end()) |
| enumerate_devices_labels_cache_[origin] = std::make_pair(false, video); |
| else |
| it->second.second = video; |
| } |
| |
| bool AwPermissionManager::ShouldShowEnumerateDevicesAudioLabels( |
| const url::Origin& origin) { |
| auto it = enumerate_devices_labels_cache_.find(origin); |
| if (it == enumerate_devices_labels_cache_.end()) |
| return false; |
| return it->second.first; |
| } |
| |
| bool AwPermissionManager::ShouldShowEnumerateDevicesVideoLabels( |
| const url::Origin& origin) { |
| auto it = enumerate_devices_labels_cache_.find(origin); |
| if (it == enumerate_devices_labels_cache_.end()) |
| return false; |
| return it->second.second; |
| } |
| |
| void AwPermissionManager::ClearEnumerateDevicesCachedPermission( |
| const url::Origin& origin, |
| bool remove_audio, |
| bool remove_video) { |
| auto it = enumerate_devices_labels_cache_.find(origin); |
| if (it == enumerate_devices_labels_cache_.end()) |
| return; |
| else if (remove_audio && remove_video) { |
| enumerate_devices_labels_cache_.erase(origin); |
| } else if (remove_audio) { |
| it->second.first = false; |
| } else if (remove_video) { |
| it->second.second = false; |
| } |
| } |
| |
| int AwPermissionManager::GetRenderProcessID( |
| content::RenderFrameHost* render_frame_host) { |
| return render_frame_host->GetProcess()->GetID(); |
| } |
| |
| int AwPermissionManager::GetRenderFrameID( |
| content::RenderFrameHost* render_frame_host) { |
| return render_frame_host->GetRoutingID(); |
| } |
| |
| GURL AwPermissionManager::LastCommittedOrigin( |
| content::RenderFrameHost* render_frame_host) { |
| return permissions::PermissionUtil::GetLastCommittedOriginAsURL( |
| render_frame_host); |
| } |
| |
| AwBrowserPermissionRequestDelegate* AwPermissionManager::GetDelegate( |
| int render_process_id, |
| int render_frame_id) { |
| return AwBrowserPermissionRequestDelegate::FromID(render_process_id, |
| render_frame_id); |
| } |
| |
| } // namespace android_webview |