chromium/ios/web/navigation/navigation_item_impl_unittest.mm

// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#import "ios/web/navigation/navigation_item_impl.h"

#import <memory>

#import "base/strings/sys_string_conversions.h"
#import "base/strings/utf_string_conversions.h"
#import "ios/web/navigation/wk_navigation_util.h"
#import "ios/web/public/session/proto/navigation.pb.h"
#import "ios/web/public/session/proto/proto_util.h"
#import "testing/gtest/include/gtest/gtest.h"
#import "testing/gtest_mac.h"
#import "testing/platform_test.h"

namespace web {
namespace {

const char kItemURLString[] = "http://init.test";
static NSString* const kHTTPHeaderKey1 = @"key1";
static NSString* const kHTTPHeaderKey2 = @"key2";
static NSString* const kHTTPHeaderValue1 = @"value1";
static NSString* const kHTTPHeaderValue2 = @"value2";

class NavigationItemTest : public PlatformTest {
 protected:
  void SetUp() override {
    item_.reset(new web::NavigationItemImpl());
    item_->SetOriginalRequestURL(GURL(kItemURLString));
    item_->SetURL(GURL(kItemURLString));
    item_->SetTransitionType(ui::PAGE_TRANSITION_AUTO_BOOKMARK);
    item_->SetTimestamp(base::Time::Now());
    item_->AddHttpRequestHeaders(@{kHTTPHeaderKey1 : kHTTPHeaderValue1});
    item_->SetPostData([@"Test data" dataUsingEncoding:NSUTF8StringEncoding]);
  }

  // The NavigationItemImpl instance being tested.
  std::unique_ptr<NavigationItemImpl> item_;
};

// TODO(rohitrao): Add and adapt tests from NavigationEntryImpl.
TEST_F(NavigationItemTest, Dummy) {
  const GURL url("http://init.test");
  item_->SetURL(url);
  EXPECT_TRUE(item_->GetURL().is_valid());
}

#ifndef NDEBUG
// Tests that the debug description is as expected.
TEST_F(NavigationItemTest, Description) {
  item_->SetTitle(u"Title");
  NSString* description = item_->GetDescription();
  EXPECT_TRUE([description containsString:@"url:http://init.test/"]);
  EXPECT_TRUE([description containsString:@"originalurl:http://init.test/"]);
  EXPECT_TRUE([description containsString:@"title:Title"]);
  EXPECT_TRUE([description containsString:@"transition:2"]);
  EXPECT_TRUE([description containsString:@"userAgent:NONE"]);
  EXPECT_TRUE(
      [description containsString:@"is_created_from_hash_change: false"]);
  EXPECT_TRUE([description containsString:@"navigation_initiation_type: 0"]);
  EXPECT_TRUE([description containsString:@"https_upgrade_type: None"]);
}
#endif

// Tests that cloning NavigationItemImpls create copies of data members that are
// objects.
TEST_F(NavigationItemTest, Clone) {
  // Create objects to be copied.
  NSString* postData0 = @"postData0";
  NSMutableData* mutablePostData =
      [[postData0 dataUsingEncoding:NSUTF8StringEncoding] mutableCopy];
  item_->SetPostData(mutablePostData);
  NSString* state0 = @"state0";
  NSMutableString* mutableState = [state0 mutableCopy];
  item_->SetSerializedStateObject(mutableState);

  // Clone.
  std::unique_ptr<web::NavigationItemImpl> clone = item_->Clone();

  // Modify the objects.
  NSString* postData1 = @"postData1";
  [mutablePostData setData:[postData1 dataUsingEncoding:NSUTF8StringEncoding]];
  NSString* state1 = @"state1";
  [mutableState setString:state1];

  // Check that changes occurred in `item_`, but not in `copy`.
  EXPECT_NSEQ([postData1 dataUsingEncoding:NSUTF8StringEncoding],
              item_->GetPostData());
  EXPECT_NSEQ(state1, item_->GetSerializedStateObject());
  EXPECT_NSEQ([postData0 dataUsingEncoding:NSUTF8StringEncoding],
              clone -> GetPostData());
  EXPECT_NSEQ(state0, clone->GetSerializedStateObject());

  // Ensure that HTTP headers are still mutable after the copying.
  clone->AddHttpRequestHeaders(@{});
}

// Tests whether `NavigationItem::AddHttpRequestHeaders()` adds the passed
// headers to the item's request http headers.
TEST_F(NavigationItemTest, AddHttpRequestHeaders) {
  EXPECT_NSEQ(@{kHTTPHeaderKey1 : kHTTPHeaderValue1},
              item_->GetHttpRequestHeaders());

  item_->AddHttpRequestHeaders(@{kHTTPHeaderKey1 : kHTTPHeaderValue2});
  EXPECT_NSEQ(@{kHTTPHeaderKey1 : kHTTPHeaderValue2},
              item_->GetHttpRequestHeaders());

  item_->AddHttpRequestHeaders(@{kHTTPHeaderKey2 : kHTTPHeaderValue1});
  NSDictionary* expected = @{
    kHTTPHeaderKey1 : kHTTPHeaderValue2,
    kHTTPHeaderKey2 : kHTTPHeaderValue1
  };
  EXPECT_NSEQ(expected, item_->GetHttpRequestHeaders());
}

// Tests whether `NavigationItem::AddHttpRequestHeaders()` removes the header
// value associated with the passed key from the item's request http headers.
TEST_F(NavigationItemTest, RemoveHttpRequestHeaderForKey) {
  NSDictionary* httpHeaders = @{
    kHTTPHeaderKey1 : kHTTPHeaderValue1,
    kHTTPHeaderKey2 : kHTTPHeaderValue2
  };
  item_->AddHttpRequestHeaders(httpHeaders);
  EXPECT_NSEQ(httpHeaders, item_->GetHttpRequestHeaders());

  item_->RemoveHttpRequestHeaderForKey(kHTTPHeaderKey1);
  EXPECT_NSEQ(@{kHTTPHeaderKey2 : kHTTPHeaderValue2},
              item_->GetHttpRequestHeaders());

  item_->RemoveHttpRequestHeaderForKey(kHTTPHeaderKey2);
  EXPECT_FALSE(item_->GetHttpRequestHeaders());
}

// Tests the getter, setter, and cloning for the original request URL.
TEST_F(NavigationItemTest, OriginalURL) {
  GURL original_url = GURL(kItemURLString);
  EXPECT_EQ(original_url, item_->GetOriginalRequestURL());
  std::unique_ptr<web::NavigationItemImpl> clone = item_->Clone();
  GURL new_url = GURL("http://new_url.test");
  item_->SetOriginalRequestURL(new_url);
  EXPECT_EQ(new_url, item_->GetOriginalRequestURL());
  EXPECT_EQ(original_url, clone->GetOriginalRequestURL());
}

// Tests the behavior of GetVirtualURL().
TEST_F(NavigationItemTest, VirtualURLTest) {
  // Ensure that GetVirtualURL() returns GetURL() when not set to a custom
  // value.
  GURL original_url = item_->GetURL();
  EXPECT_EQ(original_url, item_->GetVirtualURL());
  // Set the virtual URL and check that the correct value is reported and that
  // GetURL() still reports the original URL.
  GURL new_virtual_url = GURL("http://new_url.test");
  item_->SetVirtualURL(new_virtual_url);
  EXPECT_EQ(new_virtual_url, item_->GetVirtualURL());
  EXPECT_EQ(original_url, item_->GetURL());
}

// Tests setting title longer than kMaxTitleLength.
TEST_F(NavigationItemTest, ExtraLongTitle) {
  item_->SetTitle(base::UTF8ToUTF16(std::string(kMaxTitleLength + 1, 'i')));
  EXPECT_EQ(kMaxTitleLength, item_->GetTitle().size());
}

// Tests NavigationItemImpl::GetDisplayTitleForURL method.
TEST_F(NavigationItemTest, GetDisplayTitleForURL) {
  std::u16string title;

  title = NavigationItemImpl::GetDisplayTitleForURL(GURL("http://foo.org/"));
  EXPECT_EQ("foo.org", base::UTF16ToUTF8(title));

  title = NavigationItemImpl::GetDisplayTitleForURL(GURL("file://foo.org/"));
  EXPECT_EQ("file://foo.org/", base::UTF16ToUTF8(title));

  title = NavigationItemImpl::GetDisplayTitleForURL(GURL("file://foo/1.gz"));
  EXPECT_EQ("1.gz", base::UTF16ToUTF8(title));
}

// Tests NavigationItemImpl::GetTitleForDisplay method
TEST_F(NavigationItemTest, GetTitleForDisplay) {
  item_->SetURL(GURL("file://foo/test.pdf"));
  item_->SetVirtualURL(GURL("testappspecific://foo/"));
  EXPECT_EQ("test.pdf", base::UTF16ToUTF8(item_->GetTitleForDisplay()));

  item_->SetURL(GURL("testappspecific://foo/test.pdf"));
  item_->SetVirtualURL(GURL("testappspecific://foo/test.pdf"));
  EXPECT_EQ("testappspecific://foo/test.pdf",
            base::UTF16ToUTF8(item_->GetTitleForDisplay()));
}

// Tests that RestoreStateFromItem correctly restore the state.
TEST_F(NavigationItemTest, RestoreState) {
  NavigationItemImpl other_item;
  other_item.SetUserAgentType(UserAgentType::DESKTOP);
  other_item.SetURL(GURL("www.otherurl.com"));
  other_item.SetVirtualURL(GURL("www.virtual.com"));

  ASSERT_NE(other_item.GetURL(), item_->GetURL());

  // With a different URL, only the UserAgent should be restored.
  item_->RestoreStateFromItem(&other_item);
  EXPECT_EQ(other_item.GetUserAgentType(), item_->GetUserAgentType());
  EXPECT_NE(other_item.GetVirtualURL(), item_->GetVirtualURL());

  NavigationItemImpl other_item2;
  other_item2.SetUserAgentType(UserAgentType::DESKTOP);
  other_item2.SetURL(item_->GetURL());
  other_item2.SetVirtualURL(GURL("www.virtual.com"));

  // Same URL, everything is restored.
  item_->RestoreStateFromItem(&other_item2);
  EXPECT_EQ(other_item2.GetUserAgentType(), item_->GetUserAgentType());
  EXPECT_EQ(other_item2.GetVirtualURL(), item_->GetVirtualURL());
}

// Tests that NavigationItemImpl round trip correctly when serialized to proto.
TEST_F(NavigationItemTest, NavigationItemImplRoundTrip) {
  NavigationItemImpl original;
  original.SetURL(GURL("http://url.test"));
  original.SetVirtualURL(GURL("http://virtual.test"));
  original.SetReferrer(
      Referrer(GURL("http://referrer.url"), ReferrerPolicyDefault));
  original.SetTimestamp(base::Time::Now());
  original.SetTitle(u"Title");
  original.SetUserAgentType(UserAgentType::DESKTOP);
  original.AddHttpRequestHeaders(@{@"HeaderKey" : @"HeaderValue"});
  original.SetTransitionType(ui::PAGE_TRANSITION_TYPED);

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  NavigationItemImpl decoded(storage);

  EXPECT_EQ(original.GetURL(), decoded.GetURL());
  EXPECT_EQ(original.GetVirtualURL(), decoded.GetVirtualURL());
  EXPECT_EQ(original.GetReferrer(), decoded.GetReferrer());
  EXPECT_EQ(original.GetTimestamp(), decoded.GetTimestamp());
  EXPECT_EQ(original.GetUserAgentType(), decoded.GetUserAgentType());
  EXPECT_NSEQ(original.GetHttpRequestHeaders(),
              decoded.GetHttpRequestHeaders());

  // The page transition type should be ui::PAGE_TRANSITION_RELOAD.
  EXPECT_FALSE(ui::PageTransitionCoreTypeIs(original.GetTransitionType(),
                                            decoded.GetTransitionType()));
  EXPECT_TRUE(ui::PageTransitionCoreTypeIs(ui::PAGE_TRANSITION_RELOAD,
                                           decoded.GetTransitionType()));
}

// Tests that NavigationItemImpl round trip correctly when serialized to proto
// even when the URL is not an HTTP/HTTPS url.
TEST_F(NavigationItemTest, NavigationItemImplRoundTripNonHTTPURL) {
  NavigationItemImpl original;
  original.SetURL(GURL("file:///path/to/file.pdf"));
  original.SetVirtualURL(GURL("http://virtual.test"));

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  NavigationItemImpl decoded(storage);

  EXPECT_NE(original.GetURL(), decoded.GetURL());
  EXPECT_EQ(original.GetVirtualURL(), decoded.GetURL());
  EXPECT_EQ(original.GetVirtualURL(), decoded.GetVirtualURL());
}

// Tests that NavigationItemImpl round trip correctly when serialized to proto
// even when the URL is not an HTTP/HTTPS url, in absence of virtual URL.
TEST_F(NavigationItemTest, NavigationItemImplRoundTripNonHTTPURLNoVirtualURL) {
  NavigationItemImpl original;
  original.SetURL(GURL("testwebui://invalid/"));

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  NavigationItemImpl decoded(storage);

  EXPECT_EQ(original.GetURL(), decoded.GetURL());
  EXPECT_EQ(original.GetVirtualURL(), decoded.GetVirtualURL());
}

// Tests that NavigationItemImpl serialization skips invalid referrer.
TEST_F(NavigationItemTest, SerializationSkipsInvalidReferrer) {
  NavigationItemImpl original;
  original.SetReferrer(Referrer(GURL("invalid"), ReferrerPolicyDefault));

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  EXPECT_FALSE(storage.has_referrer());
}

// Tests that NavigationItemImpl serialization skips HTTP headers if empty.
TEST_F(NavigationItemTest, SerializationSkipsEmptyHTTPHeaders) {
  NavigationItemImpl original;

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  EXPECT_FALSE(storage.has_http_request_headers());
}

// Tests that NavigationItemImpl serialization optimizes the serialization
// of URL and virtual URL when both are equals.
TEST_F(NavigationItemTest, SerializationOptimizesURLStorage) {
  NavigationItemImpl original;
  // Set virtual URL before URL because SetVirtualURL() also optimize the
  // storage of the URLs in NavigationItemImpl by clearing virtual_url_
  // if the value set is equal to url_.
  original.SetVirtualURL(GURL("http://url.test"));
  original.SetURL(GURL("http://url.test"));

  proto::NavigationItemStorage storage;
  original.SerializeToProto(storage);

  EXPECT_FALSE(storage.url().empty());
  EXPECT_TRUE(storage.virtual_url().empty());
}

// Tests correct decoding of the URL and virtual URL when using http: scheme.
TEST_F(NavigationItemTest, DecodeHTTPScheme) {
  web::proto::NavigationItemStorage storage;
  storage.set_url("http://url.test");
  storage.set_virtual_url("http://virtual.test");
  ASSERT_NE(storage.url(), storage.virtual_url());

  NavigationItemImpl navigation_item(storage);
  EXPECT_EQ(GURL(storage.url()), navigation_item.GetURL());
  EXPECT_EQ(GURL(storage.virtual_url()), navigation_item.GetVirtualURL());
}

// Tests correct decoding of the URL and virtual URL when using file: scheme.
TEST_F(NavigationItemTest, DecodeFileScheme) {
  web::proto::NavigationItemStorage storage;
  storage.set_url("file://myfile.test");
  storage.set_virtual_url("http://virtual.test");
  ASSERT_NE(storage.url(), storage.virtual_url());

  NavigationItemImpl navigation_item(storage);
  EXPECT_EQ(GURL(storage.virtual_url()), navigation_item.GetURL());
  EXPECT_EQ(GURL(storage.virtual_url()), navigation_item.GetVirtualURL());
}

// Tests correct decoding of the URL and virtual URL when using blob: scheme.
TEST_F(NavigationItemTest, DecodeBlobScheme) {
  web::proto::NavigationItemStorage storage;
  storage.set_url("blob:myfile.test");
  storage.set_virtual_url("http://virtual.test");
  ASSERT_NE(storage.url(), storage.virtual_url());

  NavigationItemImpl navigation_item(storage);
  EXPECT_EQ(GURL(storage.virtual_url()), navigation_item.GetURL());
  EXPECT_EQ(GURL(storage.virtual_url()), navigation_item.GetVirtualURL());
}

}  // namespace
}  // namespace web