Upstream version 10.39.225.0
[platform/framework/web/crosswalk.git] / src / third_party / WebKit / Source / modules / webdatabase / DatabaseManager.cpp
1 /*
2  * Copyright (C) 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. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "modules/webdatabase/DatabaseManager.h"
28
29 #include "bindings/core/v8/ExceptionMessages.h"
30 #include "bindings/core/v8/ExceptionState.h"
31 #include "core/dom/ExceptionCode.h"
32 #include "core/dom/ExecutionContext.h"
33 #include "core/dom/ExecutionContextTask.h"
34 #include "core/inspector/ConsoleMessage.h"
35 #include "modules/webdatabase/Database.h"
36 #include "modules/webdatabase/DatabaseCallback.h"
37 #include "modules/webdatabase/DatabaseClient.h"
38 #include "modules/webdatabase/DatabaseContext.h"
39 #include "modules/webdatabase/DatabaseTask.h"
40 #include "modules/webdatabase/DatabaseTracker.h"
41 #include "platform/Logging.h"
42 #include "platform/weborigin/SecurityOrigin.h"
43 #include "wtf/MainThread.h"
44
45 namespace blink {
46
47 DatabaseManager& DatabaseManager::manager()
48 {
49     ASSERT(isMainThread());
50     DEFINE_STATIC_LOCAL(DatabaseManager, dbManager, ());
51     return dbManager;
52 }
53
54 DatabaseManager::DatabaseManager()
55 #if ENABLE(ASSERT)
56     : m_databaseContextRegisteredCount(0)
57     , m_databaseContextInstanceCount(0)
58 #endif
59 {
60 }
61
62 DatabaseManager::~DatabaseManager()
63 {
64 }
65
66 class DatabaseCreationCallbackTask FINAL : public ExecutionContextTask {
67 public:
68     static PassOwnPtr<DatabaseCreationCallbackTask> create(PassRefPtrWillBeRawPtr<Database> database, DatabaseCallback* creationCallback)
69     {
70         return adoptPtr(new DatabaseCreationCallbackTask(database, creationCallback));
71     }
72
73     virtual void performTask(ExecutionContext*) OVERRIDE
74     {
75         m_creationCallback->handleEvent(m_database.get());
76     }
77
78 private:
79     DatabaseCreationCallbackTask(PassRefPtrWillBeRawPtr<Database> database, DatabaseCallback* callback)
80         : m_database(database)
81         , m_creationCallback(callback)
82     {
83     }
84
85     RefPtrWillBePersistent<Database> m_database;
86     Persistent<DatabaseCallback> m_creationCallback;
87 };
88
89 DatabaseContext* DatabaseManager::existingDatabaseContextFor(ExecutionContext* context)
90 {
91     ASSERT(m_databaseContextRegisteredCount >= 0);
92     ASSERT(m_databaseContextInstanceCount >= 0);
93     ASSERT(m_databaseContextRegisteredCount <= m_databaseContextInstanceCount);
94     return m_contextMap.get(context);
95 }
96
97 DatabaseContext* DatabaseManager::databaseContextFor(ExecutionContext* context)
98 {
99     if (DatabaseContext* databaseContext = existingDatabaseContextFor(context))
100         return databaseContext;
101     // We don't need to hold a reference returned by DatabaseContext::create
102     // because DatabaseContext::create calls registerDatabaseContext, and the
103     // DatabaseManager holds a reference.
104     return DatabaseContext::create(context).get();
105 }
106
107 void DatabaseManager::registerDatabaseContext(DatabaseContext* databaseContext)
108 {
109     ExecutionContext* context = databaseContext->executionContext();
110     m_contextMap.set(context, databaseContext);
111 #if ENABLE(ASSERT)
112     m_databaseContextRegisteredCount++;
113 #endif
114 }
115
116 void DatabaseManager::unregisterDatabaseContext(DatabaseContext* databaseContext)
117 {
118     ExecutionContext* context = databaseContext->executionContext();
119     ASSERT(m_contextMap.get(context));
120 #if ENABLE(ASSERT)
121     m_databaseContextRegisteredCount--;
122 #endif
123     m_contextMap.remove(context);
124 }
125
126 #if ENABLE(ASSERT)
127 void DatabaseManager::didConstructDatabaseContext()
128 {
129     m_databaseContextInstanceCount++;
130 }
131
132 void DatabaseManager::didDestructDatabaseContext()
133 {
134     m_databaseContextInstanceCount--;
135     ASSERT(m_databaseContextRegisteredCount <= m_databaseContextInstanceCount);
136 }
137 #endif
138
139 void DatabaseManager::throwExceptionForDatabaseError(DatabaseError error, const String& errorMessage, ExceptionState& exceptionState)
140 {
141     switch (error) {
142     case DatabaseError::None:
143         return;
144     case DatabaseError::GenericSecurityError:
145         exceptionState.throwSecurityError(errorMessage);
146         return;
147     case DatabaseError::InvalidDatabaseState:
148         exceptionState.throwDOMException(InvalidStateError, errorMessage);
149         return;
150     default:
151         ASSERT_NOT_REACHED();
152     }
153 }
154
155 static void logOpenDatabaseError(ExecutionContext* context, const String& name)
156 {
157     WTF_LOG(StorageAPI, "Database %s for origin %s not allowed to be established", name.ascii().data(),
158         context->securityOrigin()->toString().ascii().data());
159 }
160
161 PassRefPtrWillBeRawPtr<Database> DatabaseManager::openDatabaseInternal(ExecutionContext* context,
162     const String& name, const String& expectedVersion, const String& displayName,
163     unsigned long estimatedSize, bool setVersionInNewDatabase, DatabaseError& error, String& errorMessage)
164 {
165     ASSERT(error == DatabaseError::None);
166
167     DatabaseContext* backendContext = databaseContextFor(context)->backend();
168     if (DatabaseTracker::tracker().canEstablishDatabase(backendContext, name, displayName, estimatedSize, error)) {
169         RefPtrWillBeRawPtr<Database> backend = adoptRefWillBeNoop(new Database(backendContext, name, expectedVersion, displayName, estimatedSize));
170         if (backend->openAndVerifyVersion(setVersionInNewDatabase, error, errorMessage))
171             return backend.release();
172     }
173
174     ASSERT(error != DatabaseError::None);
175     switch (error) {
176     case DatabaseError::GenericSecurityError:
177         logOpenDatabaseError(context, name);
178         return nullptr;
179
180     case DatabaseError::InvalidDatabaseState:
181         logErrorMessage(context, errorMessage);
182         return nullptr;
183
184     default:
185         ASSERT_NOT_REACHED();
186     }
187     return nullptr;
188 }
189
190 PassRefPtrWillBeRawPtr<Database> DatabaseManager::openDatabase(ExecutionContext* context,
191     const String& name, const String& expectedVersion, const String& displayName,
192     unsigned long estimatedSize, DatabaseCallback* creationCallback,
193     DatabaseError& error, String& errorMessage)
194 {
195     ASSERT(error == DatabaseError::None);
196
197     bool setVersionInNewDatabase = !creationCallback;
198     RefPtrWillBeRawPtr<Database> database = openDatabaseInternal(context, name,
199         expectedVersion, displayName, estimatedSize, setVersionInNewDatabase, error, errorMessage);
200     if (!database)
201         return nullptr;
202
203     databaseContextFor(context)->setHasOpenDatabases();
204     DatabaseClient::from(context)->didOpenDatabase(database, context->securityOrigin()->host(), name, expectedVersion);
205
206     if (database->isNew() && creationCallback) {
207         WTF_LOG(StorageAPI, "Scheduling DatabaseCreationCallbackTask for database %p\n", database.get());
208         database->executionContext()->postTask(DatabaseCreationCallbackTask::create(database, creationCallback));
209     }
210
211     ASSERT(database);
212     return database.release();
213 }
214
215 String DatabaseManager::fullPathForDatabase(SecurityOrigin* origin, const String& name, bool createIfDoesNotExist)
216 {
217     return DatabaseTracker::tracker().fullPathForDatabase(origin, name, createIfDoesNotExist);
218 }
219
220 void DatabaseManager::logErrorMessage(ExecutionContext* context, const String& message)
221 {
222     context->addConsoleMessage(ConsoleMessage::create(StorageMessageSource, ErrorMessageLevel, message));
223 }
224
225 } // namespace blink