chromium/ios/chrome/browser/web/model/child_window_open_by_dom_egtest.mm

// Copyright 2016 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/strings/stringprintf.h"
#import "base/strings/sys_string_conversions.h"
#import "base/time/time.h"
#import "components/content_settings/core/common/content_settings.h"
#import "ios/chrome/test/earl_grey/chrome_earl_grey.h"
#import "ios/chrome/test/earl_grey/chrome_matchers.h"
#import "ios/chrome/test/earl_grey/chrome_test_case.h"
#import "ios/chrome/test/earl_grey/scoped_block_popups_pref.h"
#import "ios/testing/earl_grey/earl_grey_test.h"
#import "net/test/embedded_test_server/embedded_test_server.h"
#import "net/test/embedded_test_server/http_request.h"
#import "net/test/embedded_test_server/http_response.h"
#import "net/test/embedded_test_server/request_handler_util.h"

using chrome_test_util::OmniboxText;

namespace {
// Test link text and ids.
NSString* kNamedWindowLink = @"openWindowWithName";
NSString* kUnnamedWindowLink = @"openWindowNoName";

// Web view text that indicates window's closed state.
const char kWindow2NeverOpen[] = "window2.closed: never opened";
const char kWindow1Open[] = "window1.closed: false";
const char kWindow2Open[] = "window2.closed: false";
const char kWindow1Closed[] = "window1.closed: true";
const char kWindow2Closed[] = "window2.closed: true";

// URLs for testWindowOpenWriteAndReload.
const char kWriteReloadPath[] = "/writeReload.html";
const char kSlowPath[] = "/slow.html";
const char kSlowPathContent[] = "Slow Page";
constexpr base::TimeDelta kSlowPathDelay = base::Seconds(3);

// net::EmbeddedTestServer handler for kWriteReloadPath.
std::unique_ptr<net::test_server::HttpResponse> ReloadHandler(
    const net::test_server::HttpRequest& request) {
  auto http_response = std::make_unique<net::test_server::BasicHttpResponse>();
  http_response->set_content_type("text/html");
  http_response->set_content(base::StringPrintf(
      "<html><body><script>function start(){var x = window.open('javascript"
      ":document.write(1)');setTimeout(function(){x.location='%s'}, "
      "500);};</script><input onclick='start()' id='button' value='button' "
      "type='button' /></body></html>",
      kSlowPath));
  return std::move(http_response);
}

// net::EmbeddedTestServer handler for kSlowPath.
std::unique_ptr<net::test_server::HttpResponse> SlowResponseHandler(
    const net::test_server::HttpRequest& request) {
  auto slow_http_response =
      std::make_unique<net::test_server::DelayedHttpResponse>(kSlowPathDelay);
  slow_http_response->set_content_type("text/html");
  slow_http_response->set_content(kSlowPathContent);
  return std::move(slow_http_response);
}

}  // namespace

// Test case for child windows opened by DOM.
@interface ChildWindowOpenByDOMTestCase : ChromeTestCase {
  std::unique_ptr<ScopedBlockPopupsPref> _blockPopupsPref;
}

@end

@implementation ChildWindowOpenByDOMTestCase

- (void)setUp {
  [super setUp];
  _blockPopupsPref =
      std::make_unique<ScopedBlockPopupsPref>(CONTENT_SETTING_ALLOW);
  self.testServer->RegisterDefaultHandler(base::BindRepeating(
      net::test_server::HandlePrefixedRequest, kWriteReloadPath,
      base::BindRepeating(&ReloadHandler)));
  self.testServer->RegisterDefaultHandler(
      base::BindRepeating(net::test_server::HandlePrefixedRequest, kSlowPath,
                          base::BindRepeating(&SlowResponseHandler)));
  GREYAssertTrue(self.testServer->Start(), @"Test server failed to start.");

  // Open the test page. There should only be one tab open.
  const char kChildWindowTestURL[] = "/window_proxy.html";
  [ChromeEarlGrey loadURL:self.testServer->GetURL(kChildWindowTestURL)];
  [ChromeEarlGrey waitForWebStateContainingText:(base::SysNSStringToUTF8(
                                                    kNamedWindowLink))];
  [ChromeEarlGrey waitForMainTabCount:1];
}

// Tests that multiple calls to window.open() with the same window name returns
// the same window object.
- (void)test2ChildWindowsWithName {
  // Open two windows with the same name.
  [ChromeEarlGrey tapWebStateElementWithID:kNamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  [ChromeEarlGrey tapWebStateElementWithID:kNamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Check that they're the same window.
  [ChromeEarlGrey tapWebStateElementWithID:@"compareNamedWindows"];
  const char kWindowsEqualText[] = "named windows equal: true";
  [ChromeEarlGrey waitForWebStateContainingText:kWindowsEqualText];
}

// Tests that multiple calls to window.open() with no window name passed in
// returns a unique window object each time.
- (void)test2ChildWindowsWithoutName {
  // Open two unnamed windows.
  [ChromeEarlGrey tapWebStateElementWithID:kUnnamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  [ChromeEarlGrey tapWebStateElementWithID:kUnnamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:3];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Check that they aren't the same window object.
  [ChromeEarlGrey tapWebStateElementWithID:@"compareUnnamedWindows"];
  std::string kWindowsEqualText = "unnamed windows equal: false";
  [ChromeEarlGrey waitForWebStateContainingText:kWindowsEqualText];
}

// Tests that calling window.open() with a name returns a different window
// object than a subsequent call to window.open() without a name.
- (void)testChildWindowsWithAndWithoutName {
  // Open a named window.
  [ChromeEarlGrey tapWebStateElementWithID:kNamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Open an unnamed window.
  [ChromeEarlGrey tapWebStateElementWithID:kUnnamedWindowLink];
  [ChromeEarlGrey waitForMainTabCount:3];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Check that they aren't the same window object.
  [ChromeEarlGrey tapWebStateElementWithID:@"compareNamedAndUnnamedWindows"];
  const char kWindowsEqualText[] = "named and unnamed equal: false";
  [ChromeEarlGrey waitForWebStateContainingText:kWindowsEqualText];
}

// Tests that window.closed is correctly set to true when the corresponding tab
// is closed. Verifies that calling window.open() multiple times with the same
// name returns the same window object, and thus closing the corresponding tab
// results in window.closed being set to true for all references to the window
// object for that tab.
- (void)testWindowClosedWithName {
  [ChromeEarlGrey tapWebStateElementWithID:@"openWindowWithName"];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Check that named window 1 is opened and named window 2 isn't.
  const char kCheckWindow1Link[] = "checkNamedWindow1Closed";
  [ChromeEarlGrey waitForWebStateContainingText:kCheckWindow1Link];
  [ChromeEarlGrey
      tapWebStateElementWithID:[NSString
                                   stringWithUTF8String:kCheckWindow1Link]];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow1Open];
  NSString* kCheckWindow2Link = @"checkNamedWindow2Closed";
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2NeverOpen];

  // Open another window with the same name. Check that named window 2 is now
  // opened.
  [ChromeEarlGrey tapWebStateElementWithID:@"openWindowWithName"];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2Open];

  // Close the opened window. Check that named window 1 and 2 are both closed.
  [ChromeEarlGrey closeTabAtIndex:1];
  [ChromeEarlGrey waitForMainTabCount:1];
  [ChromeEarlGrey
      tapWebStateElementWithID:[NSString
                                   stringWithUTF8String:kCheckWindow1Link]];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow1Closed];
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2Closed];
}

// Tests that closing a tab will set window.closed to true for only
// corresponding window object and not for any other window objects.
- (void)testWindowClosedWithoutName {
  [ChromeEarlGrey tapWebStateElementWithID:@"openWindowNoName"];
  [ChromeEarlGrey waitForMainTabCount:2];
  [ChromeEarlGrey selectTabAtIndex:0];

  // Check that unnamed window 1 is opened and unnamed window 2 isn't.
  const char kCheckWindow1Link[] = "checkUnnamedWindow1Closed";
  [ChromeEarlGrey waitForWebStateContainingText:kCheckWindow1Link];
  [ChromeEarlGrey
      tapWebStateElementWithID:[NSString
                                   stringWithUTF8String:kCheckWindow1Link]];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow1Open];
  NSString* kCheckWindow2Link = @"checkUnnamedWindow2Closed";
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2NeverOpen];

  // Open another unnamed window. Check that unnamed window 2 is now opened.
  [ChromeEarlGrey tapWebStateElementWithID:@"openWindowNoName"];
  [ChromeEarlGrey waitForMainTabCount:3];
  [ChromeEarlGrey selectTabAtIndex:0];
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2Open];

  // Close the first opened window. Check that unnamed window 1 is closed and
  // unnamed window 2 is still open.
  [ChromeEarlGrey closeTabAtIndex:1];
  [ChromeEarlGrey
      tapWebStateElementWithID:[NSString
                                   stringWithUTF8String:kCheckWindow1Link]];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow1Closed];
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2Open];

  // Close the second opened window. Check that unnamed window 2 is closed.
  [ChromeEarlGrey closeTabAtIndex:1];
  [ChromeEarlGrey tapWebStateElementWithID:kCheckWindow2Link];
  [ChromeEarlGrey waitForWebStateContainingText:kWindow2Closed];
}

// Tests that reloading a window.open with a document.write does not leave a
// dangling pending item. This is a regression test from crbug.com/1011898
- (void)testWindowOpenWriteAndReload {
  [ChromeEarlGrey loadURL:self.testServer->GetURL(kWriteReloadPath)];
  [ChromeEarlGrey tapWebStateElementWithID:@"button"];
  [ChromeEarlGrey reload];
  [ChromeEarlGrey waitForWebStateContainingText:kSlowPathContent
                                        timeout:kSlowPathDelay * 2];

  GURL slowURL = self.testServer->GetURL(kSlowPath);
  [[EarlGrey selectElementWithMatcher:OmniboxText(slowURL.GetContent())]
      assertWithMatcher:grey_notNil()];
}

@end