|  | // Copyright 2021 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #import "base/test/ios/wait_util.h" | 
|  |  | 
|  | #import "base/functional/bind.h" | 
|  | #import "base/ios/ios_util.h" | 
|  | #import "base/strings/sys_string_conversions.h" | 
|  | #import "base/test/ios/wait_util.h" | 
|  | #import "ios/web/js_messaging/java_script_feature_manager.h" | 
|  | #import "ios/web/public/js_messaging/content_world.h" | 
|  | #import "ios/web/public/js_messaging/java_script_feature_util.h" | 
|  | #import "ios/web/public/js_messaging/script_message.h" | 
|  | #import "ios/web/public/js_messaging/web_frames_manager.h" | 
|  | #import "ios/web/public/test/fakes/fake_web_client.h" | 
|  | #import "ios/web/public/test/web_test_with_web_state.h" | 
|  | #import "ios/web/public/test/web_view_content_test_util.h" | 
|  | #import "ios/web/test/fakes/fake_java_script_feature.h" | 
|  | #import "ios/web/web_state/ui/wk_web_view_configuration_provider.h" | 
|  |  | 
|  | using base::test::ios::kWaitForJSCompletionTimeout; | 
|  | using base::test::ios::WaitUntilConditionOrTimeout; | 
|  |  | 
|  | static NSString* kPageHTML = | 
|  | @"<html><body>" | 
|  | "  <div id=\"div\">contents1</div><div id=\"div2\">contents2</div>" | 
|  | "</body></html>"; | 
|  |  | 
|  | namespace web { | 
|  |  | 
|  | // typedef WebTestWithWebState JavaScriptFeatureTest; | 
|  | // Sets up a FakeJavaScriptFeature in the page content world. | 
|  | class JavaScriptFeaturePageContentWorldTest : public WebTestWithWebState { | 
|  | protected: | 
|  | JavaScriptFeaturePageContentWorldTest() | 
|  | : WebTestWithWebState(std::make_unique<web::FakeWebClient>()), | 
|  | feature_(ContentWorld::kPageContentWorld) {} | 
|  |  | 
|  | void SetUp() override { | 
|  | WebTestWithWebState::SetUp(); | 
|  |  | 
|  | static_cast<web::FakeWebClient*>(WebTestWithWebState::GetWebClient()) | 
|  | ->SetJavaScriptFeatures({feature()}); | 
|  | } | 
|  |  | 
|  | WebFrame* GetMainFrame() { | 
|  | return feature()->GetWebFramesManager(web_state())->GetMainWebFrame(); | 
|  | } | 
|  |  | 
|  | FakeJavaScriptFeature* feature() { return &feature_; } | 
|  |  | 
|  | private: | 
|  | FakeJavaScriptFeature feature_; | 
|  | }; | 
|  |  | 
|  | // Tests that a JavaScriptFeature executes its injected JavaScript when | 
|  | // configured in the page content world. | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | JavaScriptFeatureInjectJavaScript) { | 
|  | LoadHtml(kPageHTML); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents1")); | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText( | 
|  | web_state(), kFakeJavaScriptFeatureLoadedText)); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature correctly calls JavaScript functions when | 
|  | // configured in the page content world. | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | JavaScriptFeatureExecuteJavaScript) { | 
|  | LoadHtml(kPageHTML); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents1")); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  |  | 
|  | feature()->ReplaceDivContents(GetMainFrame()); | 
|  |  | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "updated")); | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature receives post messages from JavaScript for | 
|  | // registered names in the page content world. | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | MessageHandlerInPageContentWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | auto parameters = | 
|  | base::Value::List().Append(kFakeJavaScriptFeaturePostMessageReplyValue); | 
|  | feature()->ReplyWithPostMessage(GetMainFrame(), parameters); | 
|  |  | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return feature()->last_received_web_state(); | 
|  | })); | 
|  |  | 
|  | EXPECT_EQ(web_state(), feature()->last_received_web_state()); | 
|  |  | 
|  | ASSERT_TRUE(feature()->last_received_message()->body()); | 
|  | const std::string* reply = | 
|  | feature()->last_received_message()->body()->GetIfString(); | 
|  | ASSERT_TRUE(reply); | 
|  | EXPECT_STREQ(kFakeJavaScriptFeaturePostMessageReplyValue, reply->c_str()); | 
|  | } | 
|  |  | 
|  | // Tests that a page which overrides the window.webkit object does not break the | 
|  | // JavaScriptFeature JS->native messaging system when the feature script is | 
|  | // using `sendWebKitMessage` from ios/web/public/js_messaging/resources/utils.ts | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | MessagingWithOverriddenWebkitObject) { | 
|  | LoadHtml(kPageHTML); | 
|  | ExecuteJavaScript(@"webkit = undefined;"); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | auto parameters = | 
|  | base::Value::List().Append(kFakeJavaScriptFeaturePostMessageReplyValue); | 
|  | feature()->ReplyWithPostMessage(GetMainFrame(), parameters); | 
|  |  | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return feature()->last_received_web_state(); | 
|  | })); | 
|  |  | 
|  | EXPECT_EQ(web_state(), feature()->last_received_web_state()); | 
|  |  | 
|  | ASSERT_TRUE(feature()->last_received_message()->body()); | 
|  | const std::string* reply = | 
|  | feature()->last_received_message()->body()->GetIfString(); | 
|  | ASSERT_TRUE(reply); | 
|  | EXPECT_STREQ(kFakeJavaScriptFeaturePostMessageReplyValue, reply->c_str()); | 
|  | } | 
|  |  | 
|  | // Tests that a page which overrides the window.webkit object does not break the | 
|  | // JavaScriptFeature JS->native messaging system when the feature script is | 
|  | // using `__gCrWeb.common.sendWebKitMessage` | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | MessagingWithOverriddenWebkitObjectCommonJS) { | 
|  | LoadHtml(kPageHTML); | 
|  | ExecuteJavaScript(@"webkit = undefined;"); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | auto parameters = | 
|  | base::Value::List().Append(kFakeJavaScriptFeaturePostMessageReplyValue); | 
|  | feature()->ReplyWithPostMessageCommonJS(GetMainFrame(), parameters); | 
|  |  | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return feature()->last_received_web_state(); | 
|  | })); | 
|  |  | 
|  | EXPECT_EQ(web_state(), feature()->last_received_web_state()); | 
|  |  | 
|  | ASSERT_TRUE(feature()->last_received_message()->body()); | 
|  | const std::string* reply = | 
|  | feature()->last_received_message()->body()->GetIfString(); | 
|  | ASSERT_TRUE(reply); | 
|  | EXPECT_STREQ(kFakeJavaScriptFeaturePostMessageReplyValue, reply->c_str()); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature with | 
|  | // ReinjectionBehavior::kReinjectOnDocumentRecreation re-injects JavaScript in | 
|  | // the page content world. | 
|  | TEST_F(JavaScriptFeaturePageContentWorldTest, | 
|  | ReinjectionBehaviorPageContentWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | __block bool count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | ASSERT_EQ(0ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  |  | 
|  | ExecuteJavaScript(@"invalidFunction();"); | 
|  |  | 
|  | count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | ASSERT_EQ(1ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  |  | 
|  | ASSERT_TRUE(ExecuteJavaScript( | 
|  | @"document.open(); document.write('<p></p>'); document.close(); true;")); | 
|  |  | 
|  | ExecuteJavaScript(@"invalidFunction();"); | 
|  |  | 
|  | count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | EXPECT_EQ(2ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  | } | 
|  |  | 
|  | // Sets up a FakeJavaScriptFeature in an isolated world. | 
|  | class JavaScriptFeatureAnyContentWorldTest : public WebTestWithWebState { | 
|  | protected: | 
|  | JavaScriptFeatureAnyContentWorldTest() | 
|  | : WebTestWithWebState(std::make_unique<web::FakeWebClient>()), | 
|  | feature_(ContentWorld::kIsolatedWorld) {} | 
|  |  | 
|  | void SetUp() override { | 
|  | WebTestWithWebState::SetUp(); | 
|  |  | 
|  | static_cast<web::FakeWebClient*>(WebTestWithWebState::GetWebClient()) | 
|  | ->SetJavaScriptFeatures({feature()}); | 
|  | } | 
|  |  | 
|  | WebFrame* GetMainFrame() { | 
|  | return feature()->GetWebFramesManager(web_state())->GetMainWebFrame(); | 
|  | } | 
|  |  | 
|  | FakeJavaScriptFeature* feature() { return &feature_; } | 
|  |  | 
|  | private: | 
|  | FakeJavaScriptFeature feature_; | 
|  | }; | 
|  |  | 
|  | // Tests that a JavaScriptFeature executes its injected JavaScript when | 
|  | // configured in an isolated world. | 
|  | TEST_F(JavaScriptFeatureAnyContentWorldTest, | 
|  | JavaScriptFeatureInjectJavaScriptIsolatedWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents1")); | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText( | 
|  | web_state(), kFakeJavaScriptFeatureLoadedText)); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature correctly calls JavaScript functions when | 
|  | // configured in an isolated world. | 
|  | TEST_F(JavaScriptFeatureAnyContentWorldTest, | 
|  | JavaScriptFeatureExecuteJavaScriptInIsolatedWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents1")); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  |  | 
|  | feature()->ReplaceDivContents(GetMainFrame()); | 
|  |  | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "updated")); | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature receives post messages from JavaScript for | 
|  | // registered names in an isolated world. | 
|  | TEST_F(JavaScriptFeatureAnyContentWorldTest, MessageHandlerInIsolatedWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | auto parameters = | 
|  | base::Value::List().Append(kFakeJavaScriptFeaturePostMessageReplyValue); | 
|  | feature()->ReplyWithPostMessage(GetMainFrame(), parameters); | 
|  |  | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return feature()->last_received_web_state(); | 
|  | })); | 
|  |  | 
|  | EXPECT_EQ(web_state(), feature()->last_received_web_state()); | 
|  |  | 
|  | ASSERT_TRUE(feature()->last_received_message()->body()); | 
|  | const std::string* reply = | 
|  | feature()->last_received_message()->body()->GetIfString(); | 
|  | ASSERT_TRUE(reply); | 
|  | EXPECT_STREQ(kFakeJavaScriptFeaturePostMessageReplyValue, reply->c_str()); | 
|  | } | 
|  |  | 
|  | // Tests that a JavaScriptFeature with | 
|  | // ReinjectionBehavior::kReinjectOnDocumentRecreation re-injects JavaScript in | 
|  | // an isolated world. | 
|  | TEST_F(JavaScriptFeatureAnyContentWorldTest, ReinjectionBehaviorIsolatedWorld) { | 
|  | LoadHtml(kPageHTML); | 
|  |  | 
|  | ASSERT_FALSE(feature()->last_received_web_state()); | 
|  | ASSERT_FALSE(feature()->last_received_message()); | 
|  |  | 
|  | __block bool count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | ASSERT_EQ(0ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  |  | 
|  | ExecuteJavaScript(@"invalidFunction();"); | 
|  |  | 
|  | count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | ASSERT_EQ(1ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  |  | 
|  | ASSERT_TRUE(ExecuteJavaScript( | 
|  | @"document.open(); document.write('<p></p>'); document.close(); true;")); | 
|  |  | 
|  | ExecuteJavaScript(@"invalidFunction();"); | 
|  |  | 
|  | count_received = false; | 
|  | feature()->GetErrorCount(GetMainFrame(), | 
|  | base::BindOnce(^void(const base::Value* count) { | 
|  | ASSERT_TRUE(count); | 
|  | ASSERT_TRUE(count->is_double()); | 
|  | EXPECT_EQ(2ul, count->GetDouble()); | 
|  | count_received = true; | 
|  | })); | 
|  | ASSERT_TRUE(WaitUntilConditionOrTimeout(kWaitForJSCompletionTimeout, ^bool { | 
|  | return count_received; | 
|  | })); | 
|  | } | 
|  |  | 
|  | // Sets up a FakeJavaScriptFeature in an isolated world using | 
|  | // `ContentWorld::kIsolatedWorld`. | 
|  | class JavaScriptFeatureIsolatedWorldTest : public WebTestWithWebState { | 
|  | protected: | 
|  | JavaScriptFeatureIsolatedWorldTest() | 
|  | : WebTestWithWebState(std::make_unique<web::FakeWebClient>()), | 
|  | feature_(ContentWorld::kIsolatedWorld) {} | 
|  |  | 
|  | void SetUp() override { | 
|  | WebTestWithWebState::SetUp(); | 
|  |  | 
|  | static_cast<web::FakeWebClient*>(WebTestWithWebState::GetWebClient()) | 
|  | ->SetJavaScriptFeatures({feature()}); | 
|  | } | 
|  |  | 
|  | FakeJavaScriptFeature* feature() { return &feature_; } | 
|  |  | 
|  | private: | 
|  | FakeJavaScriptFeature feature_; | 
|  | }; | 
|  |  | 
|  | // Tests that a JavaScriptFeature correctly calls JavaScript functions when | 
|  | // configured in an isolated world only. | 
|  | TEST_F(JavaScriptFeatureIsolatedWorldTest, | 
|  | JavaScriptFeatureExecuteJavaScriptInIsolatedWorldOnly) { | 
|  | LoadHtml(kPageHTML); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents1")); | 
|  | ASSERT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  |  | 
|  | WebFrame* frame = | 
|  | feature()->GetWebFramesManager(web_state())->GetMainWebFrame(); | 
|  | feature()->ReplaceDivContents(frame); | 
|  |  | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "updated")); | 
|  | EXPECT_TRUE(test::WaitForWebViewContainingText(web_state(), "contents2")); | 
|  | } | 
|  |  | 
|  | }  // namespace web |