Restrict access to notifications for unique origins and file URLs with no local file...
[WebKit-https.git] / Source / WebKit2 / WebProcess / Notifications / WebNotificationManager.cpp
1 /*
2  * Copyright (C) 2011, 2012 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "WebNotificationManager.h"
28
29 #include "WebPage.h"
30 #include "WebProcess.h"
31
32 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
33 #include "WebNotification.h"
34 #include "WebNotificationManagerProxyMessages.h"
35 #include "WebPageProxyMessages.h"
36 #include <WebCore/Document.h>
37 #include <WebCore/Notification.h>
38 #include <WebCore/Page.h>
39 #include <WebCore/ScriptExecutionContext.h>
40 #include <WebCore/SecurityOrigin.h>
41 #include <WebCore/Settings.h>
42 #endif
43
44 using namespace WebCore;
45
46 namespace WebKit {
47
48 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
49 static uint64_t generateNotificationID()
50 {
51     static uint64_t uniqueNotificationID = 1;
52     return uniqueNotificationID++;
53 }
54 #endif
55
56 WebNotificationManager::WebNotificationManager(WebProcess* process)
57     : m_process(process)
58 {
59 }
60
61 WebNotificationManager::~WebNotificationManager()
62 {
63 }
64
65 void WebNotificationManager::didReceiveMessage(CoreIPC::Connection* connection, CoreIPC::MessageID messageID, CoreIPC::ArgumentDecoder* arguments)
66 {
67     didReceiveWebNotificationManagerMessage(connection, messageID, arguments);
68 }
69
70 void WebNotificationManager::initialize(const HashMap<String, bool>& permissions)
71 {
72 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
73     m_permissionsMap = permissions;
74 #endif
75 }
76
77 void WebNotificationManager::didUpdateNotificationDecision(const String& originString, bool allowed)
78 {
79 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
80     m_permissionsMap.set(originString, allowed);
81 #endif
82 }
83
84 void WebNotificationManager::didRemoveNotificationDecisions(const Vector<String>& originStrings)
85 {
86 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
87     size_t count = originStrings.size();
88     for (size_t i = 0; i < count; ++i)
89         m_permissionsMap.remove(originStrings[i]);
90 #endif
91 }
92
93 NotificationClient::Permission WebNotificationManager::policyForOrigin(WebCore::SecurityOrigin *origin) const
94 {
95 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
96     if (!origin)
97         return NotificationClient::PermissionNotAllowed;
98
99     ASSERT(!origin->isUnique());
100     HashMap<String, bool>::const_iterator it = m_permissionsMap.find(origin->toRawString());
101     if (it != m_permissionsMap.end())
102         return it->second ? NotificationClient::PermissionAllowed : NotificationClient::PermissionDenied;
103 #endif
104     
105     return NotificationClient::PermissionNotAllowed;
106 }
107
108 bool WebNotificationManager::show(Notification* notification, WebPage* page)
109 {
110 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
111     if (!notification || !page->corePage()->settings()->notificationsEnabled())
112         return true;
113     
114     uint64_t notificationID = generateNotificationID();
115     m_notificationMap.set(notification, notificationID);
116     m_notificationIDMap.set(notificationID, notification);
117     
118     NotificationContextMap::iterator it = m_notificationContextMap.add(notification->scriptExecutionContext(), Vector<uint64_t>()).first;
119     it->second.append(notificationID);
120     
121     m_process->connection()->send(Messages::WebPageProxy::ShowNotification(notification->title(), notification->body(), notification->iconURL().string(), notification->replaceId(), notification->scriptExecutionContext()->securityOrigin()->toString(), notificationID), page->pageID());
122     return true;
123 #else
124     return false;
125 #endif
126 }
127
128 void WebNotificationManager::cancel(Notification* notification, WebPage* page)
129 {
130 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
131     if (!notification || !page->corePage()->settings()->notificationsEnabled())
132         return;
133     
134     uint64_t notificationID = m_notificationMap.get(notification);
135     if (!notificationID)
136         return;
137     
138     m_process->connection()->send(Messages::WebNotificationManagerProxy::Cancel(notificationID), page->pageID());
139 #endif
140 }
141
142 void WebNotificationManager::clearNotifications(WebCore::ScriptExecutionContext* context, WebPage* page)
143 {
144 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
145     NotificationContextMap::iterator it = m_notificationContextMap.find(context);
146     if (it == m_notificationContextMap.end())
147         return;
148     
149     Vector<uint64_t>& notificationIDs = it->second;
150     m_process->connection()->send(Messages::WebNotificationManagerProxy::ClearNotifications(notificationIDs), page->pageID());
151     size_t count = notificationIDs.size();
152     for (size_t i = 0; i < count; ++i) {
153         RefPtr<Notification> notification = m_notificationIDMap.take(notificationIDs[i]);
154         if (!notification)
155             continue;
156         m_notificationMap.remove(notification);
157     }
158     
159     m_notificationContextMap.remove(it);
160 #endif
161 }
162
163 void WebNotificationManager::didDestroyNotification(Notification* notification, WebPage* page)
164 {
165 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
166     uint64_t notificationID = m_notificationMap.take(notification);
167     if (!notificationID)
168         return;
169
170     m_notificationIDMap.remove(notificationID);
171     removeNotificationFromContextMap(notificationID, notification);
172     m_process->connection()->send(Messages::WebNotificationManagerProxy::DidDestroyNotification(notificationID), page->pageID());
173 #endif
174 }
175
176 void WebNotificationManager::didShowNotification(uint64_t notificationID)
177 {
178 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
179     if (!isNotificationIDValid(notificationID))
180         return;
181     
182     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
183     if (!notification)
184         return;
185
186     notification->dispatchShowEvent();
187 #endif
188 }
189
190 void WebNotificationManager::didClickNotification(uint64_t notificationID)
191 {
192 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
193     if (!isNotificationIDValid(notificationID))
194         return;
195
196     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
197     if (!notification)
198         return;
199
200     notification->dispatchClickEvent();
201 #endif
202 }
203
204 void WebNotificationManager::didCloseNotifications(const Vector<uint64_t>& notificationIDs)
205 {
206 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
207     size_t count = notificationIDs.size();
208     for (size_t i = 0; i < count; ++i) {
209         uint64_t notificationID = notificationIDs[i];
210         if (!isNotificationIDValid(notificationID))
211             continue;
212
213         RefPtr<Notification> notification = m_notificationIDMap.take(notificationID);
214         if (!notification)
215             continue;
216
217         m_notificationMap.remove(notification);
218         removeNotificationFromContextMap(notificationID, notification.get());
219
220         notification->dispatchCloseEvent();
221     }
222 #endif
223 }
224
225 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
226 void WebNotificationManager::removeNotificationFromContextMap(uint64_t notificationID, Notification* notification)
227 {
228     // This is a helper function for managing the hash maps.
229     NotificationContextMap::iterator it = m_notificationContextMap.find(notification->scriptExecutionContext());
230     ASSERT(it != m_notificationContextMap.end());
231     size_t index = it->second.find(notificationID);
232     ASSERT(index != notFound);
233     it->second.remove(index);
234     if (it->second.isEmpty())
235         m_notificationContextMap.remove(it);
236 }
237 #endif
238
239 } // namespace WebKit