blob: b3a88686ffb9a38239d21c2bea605904245440db [file] [log] [blame]
/*
* Copyright (C) 2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#import "config.h"
#import "DeprecatedGlobalValues.h"
#import "PlatformUtilities.h"
#import "Test.h"
#import "TestURLSchemeHandler.h"
#import <WebKit/WKPreferencesPrivate.h>
#import <WebKit/WKPreferencesRefPrivate.h>
#import <WebKit/WKProcessPoolPrivate.h>
#import <WebKit/WKUserContentControllerPrivate.h>
#import <WebKit/WKWebViewConfigurationPrivate.h>
#import <WebKit/WKWebViewPrivate.h>
#import <WebKit/WKWebsiteDataStorePrivate.h>
#import <WebKit/WebKit.h>
#import <WebKit/_WKProcessPoolConfiguration.h>
#import <WebKit/_WKUserStyleSheet.h>
#import <wtf/RetainPtr.h>
@interface IndexedDBMessageHandler : NSObject <WKScriptMessageHandler>
@end
@implementation IndexedDBMessageHandler
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
receivedScriptMessage = true;
scriptMessages.append(message);
}
@end
TEST(IndexedDB, IndexedDBPersistence)
{
RetainPtr<IndexedDBMessageHandler> handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
RetainPtr<WKWebViewConfiguration> configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
RetainPtr<WKWebView> webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
NSURLRequest *request = [NSURLRequest requestWithURL:[[NSBundle mainBundle] URLForResource:@"IndexedDBPersistence-1" withExtension:@"html" subdirectory:@"TestWebKitAPI.resources"]];
[webView loadRequest:request];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string1 = (NSString *)[getNextMessage() body];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string2 = (NSString *)[getNextMessage() body];
// Ditch this web view (ditching its web process)
webView = nil;
// Terminate the network process
[configuration.get().websiteDataStore _terminateNetworkProcess];
// Make a new web view to finish the test
webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
request = [NSURLRequest requestWithURL:[[NSBundle mainBundle] URLForResource:@"IndexedDBPersistence-2" withExtension:@"html" subdirectory:@"TestWebKitAPI.resources"]];
[webView loadRequest:request];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string3 = (NSString *)[getNextMessage() body];
EXPECT_WK_STREQ(@"UpgradeNeeded", string1.get());
EXPECT_WK_STREQ(@"Success", string2.get());
EXPECT_WK_STREQ(@"2 TestObjectStore", string3.get());
}
TEST(IndexedDB, IndexedDBPersistencePrivate)
{
RetainPtr<IndexedDBMessageHandler> handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
RetainPtr<WKWebViewConfiguration> configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto ephemeralStore = [WKWebsiteDataStore nonPersistentDataStore];
configuration.get().websiteDataStore = ephemeralStore;
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
NSURLRequest *request = [NSURLRequest requestWithURL:[[NSBundle mainBundle] URLForResource:@"IndexedDBPersistence-1" withExtension:@"html" subdirectory:@"TestWebKitAPI.resources"]];
[webView loadRequest:request];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string1 = (NSString *)[getNextMessage() body];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string2 = (NSString *)[getNextMessage() body];
auto webViewPid1 = [webView _webProcessIdentifier];
// Ditch this web view (ditching its web process)
webView = nil;
// Make a new web view to finish the test
webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
request = [NSURLRequest requestWithURL:[[NSBundle mainBundle] URLForResource:@"IndexedDBPersistence-2" withExtension:@"html" subdirectory:@"TestWebKitAPI.resources"]];
[webView loadRequest:request];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
RetainPtr<NSString> string3 = (NSString *)[getNextMessage() body];
auto webViewPid2 = [webView _webProcessIdentifier];
EXPECT_NE(webViewPid1, webViewPid2);
EXPECT_WK_STREQ(@"UpgradeNeeded", string1.get());
EXPECT_WK_STREQ(@"Success", string2.get());
EXPECT_WK_STREQ(@"2 TestObjectStore", string3.get());
}
TEST(IndexedDB, IndexedDBDataRemoval)
{
auto websiteDataTypes = adoptNS([[NSSet alloc] initWithArray:@[WKWebsiteDataTypeIndexedDBDatabases]]);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:websiteDataTypes.get() modifiedSince:[NSDate distantPast] completionHandler:^() {
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] fetchDataRecordsOfTypes:websiteDataTypes.get() completionHandler:^(NSArray<WKWebsiteDataRecord *> *dataRecords) {
readyToContinue = true;
ASSERT_EQ(0u, dataRecords.count);
}];
TestWebKitAPI::Util::run(&readyToContinue);
receivedScriptMessage = false;
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
NSURLRequest *request = [NSURLRequest requestWithURL:[[NSBundle mainBundle] URLForResource:@"IndexedDBPersistence-1" withExtension:@"html" subdirectory:@"TestWebKitAPI.resources"]];
[webView loadRequest:request];
TestWebKitAPI::Util::run(&receivedScriptMessage);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] fetchDataRecordsOfTypes:websiteDataTypes.get() completionHandler:^(NSArray<WKWebsiteDataRecord *> *dataRecords) {
ASSERT_EQ(1u, dataRecords.count);
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:websiteDataTypes.get() forDataRecords:dataRecords completionHandler:^() {
readyToContinue = true;
}];
}];
TestWebKitAPI::Util::run(&readyToContinue);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] fetchDataRecordsOfTypes:websiteDataTypes.get() completionHandler:^(NSArray<WKWebsiteDataRecord *> *dataRecords) {
readyToContinue = true;
ASSERT_EQ(0u, dataRecords.count);
}];
TestWebKitAPI::Util::run(&readyToContinue);
}
static NSString *mainFrameStringPersistence = @"<script> \
function postResult(event) { \
window.webkit.messageHandlers.testHandler.postMessage(event.data); \
} \
addEventListener('message', postResult, false); \
</script> \
<iframe src='iframe://'>";
static const char* iframeBytesPersistence = R"TESTRESOURCE(
<script>
function postResult(result) {
if (window.parent != window.top) {
parent.postMessage(result, '*');
} else {
window.webkit.messageHandlers.testHandler.postMessage(result);
}
}
try {
var request = window.indexedDB.open('IndexedDBThirdPartyFrameHasAccess');
var db = null;
request.onupgradeneeded = function(event) {
db = event.target.result;
var objectStore = db.createObjectStore('TestObjectStore');
var putRequest = objectStore.put('TestValue', 'TestKey');
putRequest.onsuccess = function(event) {
postResult('database is created - put item success');
}
putRequest.onerror = function(event) {
postResult('database is created - put item error: ' + event.target.error.name + ' - ' + event.target.error.message);
}
}
request.onsuccess = function(event) {
if (db)
return;
db = event.target.result;
var objectStore = db.transaction(['TestObjectStore']).objectStore('TestObjectStore');
var getRequest = objectStore.get('TestKey');
getRequest.onsuccess = function(event) {
postResult('database exists - get item success: ' + event.target.result);
}
getRequest.onerror = function(event) {
postResult('database exists - get item error: ' + event.target.error.name + ' - ' + event.target.error.message);
}
}
request.onerror = function(event) {
if (!db) {
postResult('database error: ' + event.target.error.name + ' - ' + event.target.error.message);
}
}
} catch(err) {
postResult('database error: ' + err.name + ' - ' + err.message);
}
</script>
)TESTRESOURCE";
static void loadThirdPartyPageInWebView(WKWebView *webView, NSString *expectedResult)
{
[webView loadHTMLString:mainFrameStringPersistence baseURL:[NSURL URLWithString:@"http://webkit.org"]];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ(expectedResult, (NSString *)[getNextMessage() body]);
}
TEST(IndexedDB, IndexedDBThirdPartyFrameHasAccess)
{
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
auto response = adoptNS([[NSURLResponse alloc] initWithURL:task.request.URL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
[task didReceiveResponse:response.get()];
[task didReceiveData:[NSData dataWithBytes:iframeBytesPersistence length:strlen(iframeBytesPersistence)]];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"iframe"];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(webView.get(), @"database is created - put item success");
auto secondWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(secondWebView.get(), @"database exists - get item success: TestValue");
webView = nil;
secondWebView = nil;
[configuration.get().websiteDataStore _terminateNetworkProcess];
// Third-party IDB storage is stored in the memory of network process.
auto thirdWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(thirdWebView.get(), @"database is created - put item success");
}
TEST(IndexedDB, IndexedDBThirdPartyDataRemoval)
{
auto websiteDataTypes = adoptNS([[NSSet alloc] initWithArray:@[WKWebsiteDataTypeIndexedDBDatabases]]);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:websiteDataTypes.get() modifiedSince:[NSDate distantPast] completionHandler:^() {
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
[WKWebsiteDataStore _allowWebsiteDataRecordsForAllOrigins];
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
auto response = adoptNS([[NSURLResponse alloc] initWithURL:task.request.URL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
[task didReceiveResponse:response.get()];
[task didReceiveData:[NSData dataWithBytes:iframeBytesPersistence length:strlen(iframeBytesPersistence)]];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"iframe"];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(webView.get(), @"database is created - put item success");
auto secondWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
[secondWebView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"iframe://"]]];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ( @"database is created - put item success", (NSString *)[getNextMessage() body]);
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] fetchDataRecordsOfTypes:websiteDataTypes.get() completionHandler:^(NSArray<WKWebsiteDataRecord *> *records) {
EXPECT_EQ(2u, records.count);
NSSortDescriptor *recordDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"displayName" ascending:YES];
NSArray *sortDescriptors = @[recordDescriptor];
NSArray *sortedRecords = [records sortedArrayUsingDescriptors:sortDescriptors];
EXPECT_WK_STREQ("iframe", [[sortedRecords objectAtIndex:0] displayName]);
EXPECT_WK_STREQ("webkit.org", [[sortedRecords objectAtIndex:1] displayName]);
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:websiteDataTypes.get() forDataRecords:records completionHandler:^() {
readyToContinue = true;
}];
}];
TestWebKitAPI::Util::run(&readyToContinue);
auto thirdWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(thirdWebView.get(), @"database is created - put item success");
}
TEST(IndexedDB, IndexedDBThirdPartyStorageLayout)
{
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
NSString *databaseHash = WebCore::SQLiteFileSystem::computeHashForFileName("IndexedDBThirdPartyFrameHasAccess"_s);
NSURL *webkitURL = [NSURL URLWithString:@"http://webkit.org"];
NSURL *iframeURL = [NSURL URLWithString:@"iframe://"];
__block NSString *directoryString = nil;
done = false;
[configuration.get().websiteDataStore _originDirectoryForTesting:iframeURL topOrigin:webkitURL type:WKWebsiteDataTypeIndexedDBDatabases completionHandler:^(NSString *result) {
directoryString = result;
done = true;
}];
TestWebKitAPI::Util::run(&done);
NSURL *webkitIframeRootDirectory = [NSURL fileURLWithPath:directoryString isDirectory:YES];
NSURL *webkitIframeDatabaseDirectory = [webkitIframeRootDirectory URLByAppendingPathComponent:databaseHash];
NSURL *webkitIframeDatabaseFile = [webkitIframeDatabaseDirectory URLByAppendingPathComponent:@"IndexedDB.sqlite3"];
done = false;
[configuration.get().websiteDataStore _originDirectoryForTesting:webkitURL topOrigin:iframeURL type:WKWebsiteDataTypeIndexedDBDatabases completionHandler:^(NSString *result) {
directoryString = result;
done = true;
}];
TestWebKitAPI::Util::run(&done);
NSURL *iframeWebKitRootDirectory = [NSURL fileURLWithPath:directoryString isDirectory:YES];
NSFileManager *fileManager = [NSFileManager defaultManager];
[fileManager removeItemAtURL:webkitIframeRootDirectory error:nil];
[fileManager removeItemAtURL:iframeWebKitRootDirectory error:nil];
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
auto response = adoptNS([[NSURLResponse alloc] initWithURL:task.request.URL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
[task didReceiveResponse:response.get()];
[task didReceiveData:[NSData dataWithBytes:iframeBytesPersistence length:strlen(iframeBytesPersistence)]];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"iframe"];
// Allowing third-party frame to store data on disk.
[[configuration preferences] _setStorageBlockingPolicy:_WKStorageBlockingPolicyAllowAll];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(webView.get(), @"database is created - put item success");
EXPECT_TRUE([fileManager fileExistsAtPath:webkitIframeRootDirectory.path]);
EXPECT_TRUE([fileManager fileExistsAtPath:webkitIframeDatabaseFile.path]);
EXPECT_FALSE([fileManager fileExistsAtPath:iframeWebKitRootDirectory.path]);
}
TEST(IndexedDB, MigrateThirdPartyDataToGeneralStorageDirectory)
{
NSURL *resourceSalt = [[NSBundle mainBundle] URLForResource:@"general-storage-directory" withExtension:@"salt" subdirectory:@"TestWebKitAPI.resources"];
NSURL *resourceDatabase = [[NSBundle mainBundle] URLForResource:@"indexeddb-persistence-third-party" withExtension:@"sqlite3" subdirectory:@"TestWebKitAPI.resources"];
NSURL *generalStorageDirectory = [NSURL fileURLWithPath:[@"~/Library/WebKit/com.apple.WebKit.TestWebKitAPI/CustomWebsiteData/Default" stringByExpandingTildeInPath] isDirectory:YES];
NSURL *webkitOriginDirectory = [generalStorageDirectory URLByAppendingPathComponent:@"EAO66s8JvCWNn4D3YQut5pXfiGF_UXNZGvMGN6aKILg/EAO66s8JvCWNn4D3YQut5pXfiGF_UXNZGvMGN6aKILg"];
NSURL *webkitOriginFile = [webkitOriginDirectory URLByAppendingPathComponent:@"origin"];
NSURL *wrongWebkitIframeDatabaseDirectory = [webkitOriginDirectory URLByAppendingPathComponent:@"IndexedDB/iframe__0"];
NSURL *webkitIframeOriginDirectory = [generalStorageDirectory URLByAppendingPathComponent:@"EAO66s8JvCWNn4D3YQut5pXfiGF_UXNZGvMGN6aKILg/vudvbMlKXj1m6RibnVvc8PdAdcXZsNE6ON_Al7yqOsg"];
NSURL *webkitIframeOriginFile = [webkitIframeOriginDirectory URLByAppendingPathComponent:@"origin"];
NSString *hashedDatabaseName = WebCore::SQLiteFileSystem::computeHashForFileName("IndexedDBThirdPartyFrameHasAccess"_s);
NSURL *webkitIframeDatabaseFile= [NSURL fileURLWithPath:[NSString pathWithComponents:@[webkitIframeOriginDirectory.path, @"IndexedDB", hashedDatabaseName, @"IndexedDB.sqlite3"]]];
NSURL *idbDirectory = [NSURL fileURLWithPath:[@"~/Library/WebKit/com.apple.WebKit.TestWebKitAPI/CustomWebsiteData/IndexedDB" stringByExpandingTildeInPath] isDirectory:YES];
NSURL *oldWebkitIframeDatabaseDirectory = [NSURL fileURLWithPath:[NSString pathWithComponents:@[idbDirectory.path, @"v1/http_webkit.org_0/iframe__0", hashedDatabaseName]]];
NSURL *oldWebkitIframeDatabaseFile = [oldWebkitIframeDatabaseDirectory URLByAppendingPathComponent:@"IndexedDB.sqlite3"];
NSFileManager *fileManager = [NSFileManager defaultManager];
[fileManager removeItemAtURL:generalStorageDirectory error:nil];
[fileManager createDirectoryAtURL:generalStorageDirectory withIntermediateDirectories:YES attributes:nil error:nil];
[fileManager copyItemAtURL:resourceSalt toURL:[generalStorageDirectory URLByAppendingPathComponent:@"salt"] error:nil];
// Create third-party database file on disk.
[fileManager removeItemAtURL:idbDirectory error:nil];
[fileManager createDirectoryAtURL:oldWebkitIframeDatabaseDirectory withIntermediateDirectories:YES attributes:nil error:nil];
[fileManager copyItemAtURL:resourceDatabase toURL:oldWebkitIframeDatabaseFile error:nil];
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
auto response = adoptNS([[NSURLResponse alloc] initWithURL:task.request.URL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
[task didReceiveResponse:response.get()];
[task didReceiveData:[NSData dataWithBytes:iframeBytesPersistence length:strlen(iframeBytesPersistence)]];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"iframe"];
[[configuration preferences] _setStorageBlockingPolicy:_WKStorageBlockingPolicyAllowAll];
auto websiteDataStoreConfiguration = adoptNS([[_WKWebsiteDataStoreConfiguration alloc] init]);
websiteDataStoreConfiguration.get()._indexedDBDatabaseDirectory = idbDirectory;
websiteDataStoreConfiguration.get().generalStorageDirectory = generalStorageDirectory;
websiteDataStoreConfiguration.get().shouldUseCustomStoragePaths = false;
auto websiteDataStore = adoptNS([[WKWebsiteDataStore alloc] _initWithConfiguration:websiteDataStoreConfiguration.get()]);
[configuration setWebsiteDataStore:websiteDataStore.get()];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
// Ensure opening first-party database does not migrate third-party data.
NSString *firstPartyHTMLString = @"<script> \
var request = window.indexedDB.open('TestDatabase'); \
request.onsuccess = function() { \
window.webkit.messageHandlers.testHandler.postMessage('database is created'); \
}; \
request.onerror = function() { \
window.webkit.messageHandlers.testHandler.postMessage('database error'); \
}; \
</script>";
[webView loadHTMLString:firstPartyHTMLString baseURL:[NSURL URLWithString:@"http://webkit.org"]];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ(@"database is created", (NSString *)[getNextMessage() body]);
EXPECT_TRUE([fileManager fileExistsAtPath:webkitOriginFile.path]);
EXPECT_FALSE([fileManager fileExistsAtPath:wrongWebkitIframeDatabaseDirectory.path]);
EXPECT_FALSE([fileManager fileExistsAtPath:webkitIframeOriginFile.path]);
EXPECT_FALSE([fileManager fileExistsAtPath:webkitIframeDatabaseFile.path]);
// Ensure opening third-party database migrates third-party data, and data can be correctly read.
loadThirdPartyPageInWebView(webView.get(), @"database exists - get item success: TestValue");
EXPECT_TRUE([fileManager fileExistsAtPath:webkitIframeOriginFile.path]);
EXPECT_TRUE([fileManager fileExistsAtPath:webkitIframeDatabaseFile.path]);
}
static const char* workerBytes = R"TESTRESOURCE(
try {
var request = indexedDB.open('IndexedDBThirdPartyWorkerHasAccess');
var db = null;
request.onupgradeneeded = function(event) {
db = event.target.result;
self.postMessage('database is created');
}
request.onsuccess = function(event) {
if (db)
return;
self.postMessage('database exists');
}
request.onerror = function(event) {
if (!db) {
self.postMessage('database error: ' + event.target.error.name + ' - ' + event.target.error.message);
}
}
} catch(err) {
self.postMessage('database error: ' + err.name + ' - ' + err.message);
}
)TESTRESOURCE";
static const char* workerFrameBytes = R"TESTRESOURCE(
<script>
var worker = new Worker('worker.js');
worker.onmessage = function(event) {
parent.postMessage(event.data, '*');
};
</script>
)TESTRESOURCE";
TEST(IndexedDB, IndexedDBThirdPartyWorkerHasAccess)
{
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
RetainPtr<NSURLResponse> response;
RetainPtr<NSData> data;
NSURL *requestURL = task.request.URL;
if ([requestURL.absoluteString isEqualToString:@"iframe://"]) {
response = adoptNS([[NSURLResponse alloc] initWithURL:requestURL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
data = [NSData dataWithBytes:workerFrameBytes length:strlen(workerFrameBytes)];
} else {
EXPECT_WK_STREQ("iframe:///worker.js", requestURL.absoluteString);
response = adoptNS([[NSURLResponse alloc] initWithURL:requestURL MIMEType:@"text/javascript" expectedContentLength:0 textEncodingName:nil]);
data = [NSData dataWithBytes:workerBytes length:strlen(workerBytes)];
}
[task didReceiveResponse:response.get()];
[task didReceiveData:data.get()];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"iframe"];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(webView.get(), @"database is created");
auto secondWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(webView.get(), @"database exists");
webView = nil;
secondWebView = nil;
[configuration.get().websiteDataStore _terminateNetworkProcess];
auto thirdWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
loadThirdPartyPageInWebView(thirdWebView.get(), @"database is created");
}
static NSString *getDatabasesString = @"<script> \
function sendMessage(message) { \
window.webkit.messageHandlers.testHandler.postMessage(message); \
} \
function postResult(event) { \
sendMessage(event.data); \
} \
function postDatabases(databases) { \
sendMessage('databases: ' + JSON.stringify(databases)); \
} \
function loadFrame() { \
var frame = document.createElement('iframe'); \
frame.src = 'webkit://'; \
document.body.appendChild(frame); \
} \
addEventListener('message', postResult, false); \
var request = indexedDB.open('IndexedDBGetDatabases'); \
request.onsuccess = function(event) { \
indexedDB.databases().then(postDatabases); \
} \
</script>";
static const char* getDatabasesBytes = R"TESTRESOURCE(
<script>
function postResult(result) {
if (window.self !== window.top)
parent.postMessage(result, '*');
else
window.webkit.messageHandlers.testHandler.postMessage(result);
}
function postDatabases(databases) {
var prefix = 'main frame databases: ';
if (window.self !== window.top)
prefix = 'child frame databases: ';
postResult(prefix + JSON.stringify(databases));
}
indexedDB.databases().then(postDatabases);
</script>
)TESTRESOURCE";
TEST(IndexedDB, IndexedDBGetDatabases)
{
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:[NSSet setWithObjects:WKWebsiteDataTypeIndexedDBDatabases, nil] modifiedSince:[NSDate distantPast] completionHandler:^() {
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
auto handler = adoptNS([[IndexedDBMessageHandler alloc] init]);
auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
[[configuration userContentController] addScriptMessageHandler:handler.get() name:@"testHandler"];
auto schemeHandler = adoptNS([[TestURLSchemeHandler alloc] init]);
[schemeHandler setStartURLSchemeTaskHandler:^(WKWebView *, id<WKURLSchemeTask> task) {
auto response = adoptNS([[NSURLResponse alloc] initWithURL:task.request.URL MIMEType:@"text/html" expectedContentLength:0 textEncodingName:nil]);
[task didReceiveResponse:response.get()];
[task didReceiveData:[NSData dataWithBytes:getDatabasesBytes length:strlen(getDatabasesBytes)]];
[task didFinish];
}];
[configuration setURLSchemeHandler:schemeHandler.get() forURLScheme:@"webkit"];
auto webView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
[webView loadHTMLString:getDatabasesString baseURL:[NSURL URLWithString:@"http://apple.com"]];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ(@"databases: [{\"name\":\"IndexedDBGetDatabases\",\"version\":1}]", [getNextMessage() body]);
[webView evaluateJavaScript:@"loadFrame()" completionHandler:nil];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ(@"child frame databases: []", [getNextMessage() body]);
auto secondWebView = adoptNS([[WKWebView alloc] initWithFrame:NSMakeRect(0, 0, 800, 600) configuration:configuration.get()]);
[secondWebView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"webkit://"]]];
TestWebKitAPI::Util::run(&receivedScriptMessage);
receivedScriptMessage = false;
EXPECT_WK_STREQ(@"main frame databases: []", [getNextMessage() body]);
// Getting databases should not create files on disk.
NSURL *appleURL = [NSURL URLWithString:@"http://apple.com"];
NSURL *webkitURL = [NSURL URLWithString:@"webkit://"];
__block NSString *appleDirectoryString = nil;
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] _originDirectoryForTesting:appleURL topOrigin:appleURL type:WKWebsiteDataTypeIndexedDBDatabases completionHandler:^(NSString *result) {
appleDirectoryString = result;
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
NSURL *appleDirectoryURL = [NSURL fileURLWithPath:appleDirectoryString isDirectory:YES];
readyToContinue = false;
__block NSString *webkitDirectoryString = nil;
[[WKWebsiteDataStore defaultDataStore] _originDirectoryForTesting:webkitURL topOrigin:webkitURL type:WKWebsiteDataTypeIndexedDBDatabases completionHandler:^(NSString *result) {
webkitDirectoryString = result;
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
NSURL *webkitDirectoryURL = [NSURL fileURLWithPath:webkitDirectoryString isDirectory:YES];
__block NSString *appleWebkitDirectoryString = nil;
readyToContinue = false;
[[WKWebsiteDataStore defaultDataStore] _originDirectoryForTesting:webkitURL topOrigin:appleURL type:WKWebsiteDataTypeIndexedDBDatabases completionHandler:^(NSString *result) {
appleWebkitDirectoryString = result;
readyToContinue = true;
}];
TestWebKitAPI::Util::run(&readyToContinue);
NSURL *appleWebkitDirectoryURL = [NSURL fileURLWithPath:appleWebkitDirectoryString isDirectory:YES];
auto defaultFileManager = [NSFileManager defaultManager];
EXPECT_TRUE([defaultFileManager fileExistsAtPath:appleDirectoryURL.path]);
EXPECT_FALSE([defaultFileManager fileExistsAtPath:appleWebkitDirectoryURL.path]);
EXPECT_FALSE([defaultFileManager fileExistsAtPath:webkitDirectoryURL.path]);
}