| /* |
| * 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]); |
| } |