1 /*
2 * Copyright (C) 2010 Google 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 *
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14 * its contributors may be used to endorse or promote products derived
15 * from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29 #include "config.h"
30 #include "modules/indexeddb/IDBFactory.h"
31
32 #include "bindings/core/v8/ExceptionState.h"
33 #include "bindings/modules/v8/IDBBindingUtilities.h"
34 #include "core/dom/Document.h"
35 #include "core/dom/ExceptionCode.h"
36 #include "modules/indexeddb/IDBDatabase.h"
37 #include "modules/indexeddb/IDBDatabaseCallbacks.h"
38 #include "modules/indexeddb/IDBHistograms.h"
39 #include "modules/indexeddb/IDBKey.h"
40 #include "modules/indexeddb/IDBTracing.h"
41 #include "modules/indexeddb/IndexedDBClient.h"
42 #include "modules/indexeddb/WebIDBCallbacksImpl.h"
43 #include "modules/indexeddb/WebIDBDatabaseCallbacksImpl.h"
44 #include "platform/weborigin/DatabaseIdentifier.h"
45 #include "platform/weborigin/SecurityOrigin.h"
46 #include "public/platform/Platform.h"
47 #include "public/platform/WebIDBFactory.h"
48
49 namespace blink {
50
51 static const char permissionDeniedErrorMessage[] = "The user denied permission to access the database.";
52
IDBFactory(IndexedDBClient * permissionClient)53 IDBFactory::IDBFactory(IndexedDBClient* permissionClient)
54 : m_permissionClient(permissionClient)
55 {
56 }
57
trace(Visitor * visitor)58 void IDBFactory::trace(Visitor* visitor)
59 {
60 visitor->trace(m_permissionClient);
61 }
62
isContextValid(ExecutionContext * context)63 static bool isContextValid(ExecutionContext* context)
64 {
65 ASSERT(context->isDocument() || context->isWorkerGlobalScope());
66 if (context->isDocument()) {
67 Document* document = toDocument(context);
68 return document->frame() && document->page();
69 }
70 return true;
71 }
72
getDatabaseNames(ScriptState * scriptState,ExceptionState & exceptionState)73 IDBRequest* IDBFactory::getDatabaseNames(ScriptState* scriptState, ExceptionState& exceptionState)
74 {
75 IDB_TRACE("IDBFactory::getDatabaseNames");
76 if (!isContextValid(scriptState->executionContext()))
77 return nullptr;
78 if (!scriptState->executionContext()->securityOrigin()->canAccessDatabase()) {
79 exceptionState.throwSecurityError("access to the Indexed Database API is denied in this context.");
80 return 0;
81 }
82
83 IDBRequest* request = IDBRequest::create(scriptState, IDBAny::createNull(), 0);
84
85 if (!m_permissionClient->allowIndexedDB(scriptState->executionContext(), "Database Listing")) {
86 request->onError(DOMError::create(UnknownError, permissionDeniedErrorMessage));
87 return request;
88 }
89
90 Platform::current()->idbFactory()->getDatabaseNames(WebIDBCallbacksImpl::create(request).leakPtr(), createDatabaseIdentifierFromSecurityOrigin(scriptState->executionContext()->securityOrigin()));
91 return request;
92 }
93
open(ScriptState * scriptState,const String & name,unsigned long long version,ExceptionState & exceptionState)94 IDBOpenDBRequest* IDBFactory::open(ScriptState* scriptState, const String& name, unsigned long long version, ExceptionState& exceptionState)
95 {
96 IDB_TRACE("IDBFactory::open");
97 if (!version) {
98 exceptionState.throwTypeError("The version provided must not be 0.");
99 return 0;
100 }
101 return openInternal(scriptState, name, version, exceptionState);
102 }
103
openInternal(ScriptState * scriptState,const String & name,int64_t version,ExceptionState & exceptionState)104 IDBOpenDBRequest* IDBFactory::openInternal(ScriptState* scriptState, const String& name, int64_t version, ExceptionState& exceptionState)
105 {
106 Platform::current()->histogramEnumeration("WebCore.IndexedDB.FrontEndAPICalls", IDBOpenCall, IDBMethodsMax);
107 ASSERT(version >= 1 || version == IDBDatabaseMetadata::NoIntVersion);
108 if (name.isNull()) {
109 exceptionState.throwTypeError("The name provided must not be empty.");
110 return 0;
111 }
112 if (!isContextValid(scriptState->executionContext()))
113 return nullptr;
114 if (!scriptState->executionContext()->securityOrigin()->canAccessDatabase()) {
115 exceptionState.throwSecurityError("access to the Indexed Database API is denied in this context.");
116 return 0;
117 }
118
119 IDBDatabaseCallbacks* databaseCallbacks = IDBDatabaseCallbacks::create();
120 int64_t transactionId = IDBDatabase::nextTransactionId();
121 IDBOpenDBRequest* request = IDBOpenDBRequest::create(scriptState, databaseCallbacks, transactionId, version);
122
123 if (!m_permissionClient->allowIndexedDB(scriptState->executionContext(), name)) {
124 request->onError(DOMError::create(UnknownError, permissionDeniedErrorMessage));
125 return request;
126 }
127
128 Platform::current()->idbFactory()->open(name, version, transactionId, WebIDBCallbacksImpl::create(request).leakPtr(), WebIDBDatabaseCallbacksImpl::create(databaseCallbacks).leakPtr(), createDatabaseIdentifierFromSecurityOrigin(scriptState->executionContext()->securityOrigin()));
129 return request;
130 }
131
open(ScriptState * scriptState,const String & name,ExceptionState & exceptionState)132 IDBOpenDBRequest* IDBFactory::open(ScriptState* scriptState, const String& name, ExceptionState& exceptionState)
133 {
134 IDB_TRACE("IDBFactory::open");
135 return openInternal(scriptState, name, IDBDatabaseMetadata::NoIntVersion, exceptionState);
136 }
137
deleteDatabase(ScriptState * scriptState,const String & name,ExceptionState & exceptionState)138 IDBOpenDBRequest* IDBFactory::deleteDatabase(ScriptState* scriptState, const String& name, ExceptionState& exceptionState)
139 {
140 IDB_TRACE("IDBFactory::deleteDatabase");
141 Platform::current()->histogramEnumeration("WebCore.IndexedDB.FrontEndAPICalls", IDBDeleteDatabaseCall, IDBMethodsMax);
142 if (name.isNull()) {
143 exceptionState.throwTypeError("The name provided must not be empty.");
144 return 0;
145 }
146 if (!isContextValid(scriptState->executionContext()))
147 return nullptr;
148 if (!scriptState->executionContext()->securityOrigin()->canAccessDatabase()) {
149 exceptionState.throwSecurityError("access to the Indexed Database API is denied in this context.");
150 return 0;
151 }
152
153 IDBOpenDBRequest* request = IDBOpenDBRequest::create(scriptState, nullptr, 0, IDBDatabaseMetadata::DefaultIntVersion);
154
155 if (!m_permissionClient->allowIndexedDB(scriptState->executionContext(), name)) {
156 request->onError(DOMError::create(UnknownError, permissionDeniedErrorMessage));
157 return request;
158 }
159
160 Platform::current()->idbFactory()->deleteDatabase(name, WebIDBCallbacksImpl::create(request).leakPtr(), createDatabaseIdentifierFromSecurityOrigin(scriptState->executionContext()->securityOrigin()));
161 return request;
162 }
163
cmp(ScriptState * scriptState,const ScriptValue & firstValue,const ScriptValue & secondValue,ExceptionState & exceptionState)164 short IDBFactory::cmp(ScriptState* scriptState, const ScriptValue& firstValue, const ScriptValue& secondValue, ExceptionState& exceptionState)
165 {
166 IDBKey* first = scriptValueToIDBKey(scriptState->isolate(), firstValue);
167 IDBKey* second = scriptValueToIDBKey(scriptState->isolate(), secondValue);
168
169 ASSERT(first);
170 ASSERT(second);
171
172 if (!first->isValid() || !second->isValid()) {
173 exceptionState.throwDOMException(DataError, IDBDatabase::notValidKeyErrorMessage);
174 return 0;
175 }
176
177 return static_cast<short>(first->compare(second));
178 }
179
180 } // namespace blink
181