chromium/chrome/browser/ash/attestation/attestation_ca_client_unittest.cc

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

#include "chrome/browser/ash/attestation/attestation_ca_client.h"

#include "base/command_line.h"
#include "base/functional/bind.h"
#include "base/test/bind.h"
#include "chrome/test/base/testing_browser_process.h"
#include "chromeos/dbus/constants/dbus_switches.h"
#include "content/public/test/browser_task_environment.h"
#include "net/base/net_errors.h"
#include "net/http/http_status_code.h"
#include "services/network/public/cpp/weak_wrapper_shared_url_loader_factory.h"
#include "services/network/public/mojom/proxy_lookup_client.mojom.h"
#include "services/network/public/mojom/url_response_head.mojom.h"
#include "services/network/test/test_network_context.h"
#include "services/network/test/test_url_loader_factory.h"
#include "services/network/test/test_utils.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"

using testing::_;
using testing::Invoke;
using testing::StrictMock;

namespace ash {
namespace attestation {

namespace {
class MockNetworkContext : public network::TestNetworkContext {
 public:
  MockNetworkContext() {
    ON_CALL(*this, LookUpProxyForURL(_, _, _))
        .WillByDefault(
            Invoke(this, &MockNetworkContext::LookUpProxyForURLInternal));
  }
  ~MockNetworkContext() override = default;
  MOCK_METHOD(void,
              LookUpProxyForURL,
              (const GURL& url,
               const net::NetworkAnonymizationKey& network_anonymization_key,
               mojo::PendingRemote<::network::mojom::ProxyLookupClient>
                   proxy_lookup_client),
              (override));

  void SetProxyPresence(const GURL& url, bool is_present) {
    proxy_presence_table_[url] = is_present;
  }

 private:
  void LookUpProxyForURLInternal(
      const GURL& url,
      const net::NetworkAnonymizationKey& network_anonymization_key,
      mojo::PendingRemote<::network::mojom::ProxyLookupClient>
          proxy_lookup_client) {
    mojo::Remote<::network::mojom::ProxyLookupClient> client(
        std::move(proxy_lookup_client));
    if (proxy_presence_table_.count(url) == 0) {
      client->OnProxyLookupComplete(net::ERR_FAILED,
                                    /*proxy_info=*/std::nullopt);
      return;
    }
    net::ProxyInfo proxy_info;
    if (proxy_presence_table_[url]) {
      proxy_info.UseNamedProxy("named.proxy.com");
    } else {
      proxy_info.UseDirect();
    }
    client->OnProxyLookupComplete(net::OK, std::move(proxy_info));
  }

  std::map<GURL, bool> proxy_presence_table_;
};
}  // namespace

class AttestationCAClientTest : public ::testing::Test {
 public:
  AttestationCAClientTest()
      : test_shared_url_loader_factory_(
            test_url_loader_factory_.GetSafeWeakWrapper()),
        num_invocations_(0),
        result_(false) {}

  ~AttestationCAClientTest() override {}

  void SetUp() override {
    TestingBrowserProcess::GetGlobal()->SetSharedURLLoaderFactory(
        test_shared_url_loader_factory_);

    test_url_loader_factory_.SetInterceptor(base::BindLambdaForTesting(
        [&](const network::ResourceRequest& request) {
          last_resource_request_ = request;
        }));
  }

  void DataCallback(bool result, const std::string& data) {
    ++num_invocations_;
    result_ = result;
    data_ = data;
  }

  void DeleteClientDataCallback(AttestationCAClient* client,
                                bool result,
                                const std::string& data) {
    delete client;
    DataCallback(result, data);
  }

 protected:
  void CheckURLAndSendResponse(GURL expected_url,
                               net::Error error,
                               int response_code) {
    CHECK(test_url_loader_factory_.NumPending() == 1);
    EXPECT_EQ(expected_url, last_resource_request_.url);
    std::string response =
        network::GetUploadData(last_resource_request_) + "_response";
    test_url_loader_factory_.AddResponse(last_resource_request_.url.spec(),
                                         response);
    base::RunLoop().RunUntilIdle();
  }

  void SendResponse(net::Error error, net::HttpStatusCode response_code) {
    CHECK(test_url_loader_factory_.NumPending() == 1);
    auto url_response_head = network::CreateURLResponseHead(response_code);
    network::URLLoaderCompletionStatus completion_status(error);
    std::string response =
        network::GetUploadData(last_resource_request_) + "_response";

    test_url_loader_factory_.AddResponse(last_resource_request_.url,
                                         std::move(url_response_head), response,
                                         completion_status);
    base::RunLoop().RunUntilIdle();
  }

  content::BrowserTaskEnvironment task_environment_;

  network::TestURLLoaderFactory test_url_loader_factory_;
  scoped_refptr<network::SharedURLLoaderFactory>
      test_shared_url_loader_factory_;

  network::ResourceRequest last_resource_request_;

  // For use with DataCallback.
  int num_invocations_;
  bool result_;
  std::string data_;
};

TEST_F(AttestationCAClientTest, EnrollRequest) {
  AttestationCAClient client;
  client.SendEnrollRequest(
      "enroll", base::BindOnce(&AttestationCAClientTest::DataCallback,
                               base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://chromeos-ca.gstatic.com/enroll"),
                          net::OK, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("enroll_response", data_);
}

TEST_F(AttestationCAClientTest, CertificateRequest) {
  AttestationCAClient client;
  client.SendCertificateRequest(
      "certificate", base::BindOnce(&AttestationCAClientTest::DataCallback,
                                    base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://chromeos-ca.gstatic.com/sign"), net::OK,
                          net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("certificate_response", data_);
}

TEST_F(AttestationCAClientTest, CertificateRequestNetworkFailure) {
  AttestationCAClient client;
  client.SendCertificateRequest(
      "certificate", base::BindOnce(&AttestationCAClientTest::DataCallback,
                                    base::Unretained(this)));
  SendResponse(net::ERR_FAILED, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_FALSE(result_);
  EXPECT_EQ("", data_);
}

TEST_F(AttestationCAClientTest, CertificateRequestHttpError) {
  AttestationCAClient client;
  client.SendCertificateRequest(
      "certificate", base::BindOnce(&AttestationCAClientTest::DataCallback,
                                    base::Unretained(this)));
  SendResponse(net::OK, net::HTTP_NOT_FOUND);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_FALSE(result_);
  EXPECT_EQ("", data_);
}

TEST_F(AttestationCAClientTest, DeleteOnCallback) {
  AttestationCAClient* client = new AttestationCAClient();
  client->SendCertificateRequest(
      "certificate",
      base::BindOnce(&AttestationCAClientTest::DeleteClientDataCallback,
                     base::Unretained(this), client));
  SendResponse(net::OK, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("certificate_response", data_);
}

class AttestationCAClientAttestationServerTest
    : public AttestationCAClientTest {};

TEST_F(AttestationCAClientAttestationServerTest, DefaultEnrollRequest) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "default");
  AttestationCAClient client;
  client.SendEnrollRequest(
      "enroll", base::BindOnce(&AttestationCAClientTest::DataCallback,
                               base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://chromeos-ca.gstatic.com/enroll"),
                          net::OK, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("enroll_response", data_);
}

TEST_F(AttestationCAClientAttestationServerTest, DefaultCertificateRequest) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "default");
  AttestationCAClient client;
  client.SendCertificateRequest(
      "certificate", base::BindOnce(&AttestationCAClientTest::DataCallback,
                                    base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://chromeos-ca.gstatic.com/sign"), net::OK,
                          net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("certificate_response", data_);
}

TEST_F(AttestationCAClientAttestationServerTest, TestEnrollRequest) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "test");
  AttestationCAClient client;
  client.SendEnrollRequest(
      "enroll", base::BindOnce(&AttestationCAClientTest::DataCallback,
                               base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://asbestos-qa.corp.google.com/enroll"),
                          net::OK, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("enroll_response", data_);
}

TEST_F(AttestationCAClientAttestationServerTest, TestCertificateRequest) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "test");
  AttestationCAClient client;
  client.SendCertificateRequest(
      "certificate", base::BindOnce(&AttestationCAClientTest::DataCallback,
                                    base::Unretained(this)));
  CheckURLAndSendResponse(GURL("https://asbestos-qa.corp.google.com/sign"),
                          net::OK, net::HTTP_OK);

  EXPECT_EQ(1, num_invocations_);
  EXPECT_TRUE(result_);
  EXPECT_EQ("certificate_response", data_);
}

TEST_F(AttestationCAClientAttestationServerTest, ProxyPresentForDefaultCA) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "default");
  StrictMock<MockNetworkContext> network_context;
  network_context.SetProxyPresence(GURL("https://chromeos-ca.gstatic.com"),
                                   true);
  EXPECT_CALL(network_context,
              LookUpProxyForURL(GURL("https://chromeos-ca.gstatic.com"), _, _));
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present = false;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present));
  base::RunLoop().RunUntilIdle();
  EXPECT_TRUE(is_proxy_present);
}

TEST_F(AttestationCAClientAttestationServerTest, ProxyPresentForTestCA) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "test");
  StrictMock<MockNetworkContext> network_context;
  network_context.SetProxyPresence(GURL("https://asbestos-qa.corp.google.com"),
                                   true);
  EXPECT_CALL(
      network_context,
      LookUpProxyForURL(GURL("https://asbestos-qa.corp.google.com"), _, _));
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present = false;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present));
  base::RunLoop().RunUntilIdle();
  EXPECT_TRUE(is_proxy_present);
}

TEST_F(AttestationCAClientAttestationServerTest, ProxyNotPresentForDefaultCA) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "default");
  StrictMock<MockNetworkContext> network_context;
  network_context.SetProxyPresence(GURL("https://chromeos-ca.gstatic.com"),
                                   false);
  EXPECT_CALL(network_context,
              LookUpProxyForURL(GURL("https://chromeos-ca.gstatic.com"), _, _));
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present = true;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present));
  base::RunLoop().RunUntilIdle();
  EXPECT_FALSE(is_proxy_present);
}

TEST_F(AttestationCAClientAttestationServerTest, ProxyNotPresentForTestCA) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "test");
  StrictMock<MockNetworkContext> network_context;
  network_context.SetProxyPresence(GURL("https://asbestos-qa.corp.google.com"),
                                   false);
  EXPECT_CALL(
      network_context,
      LookUpProxyForURL(GURL("https://asbestos-qa.corp.google.com"), _, _));
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present = true;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present));
  base::RunLoop().RunUntilIdle();
  EXPECT_FALSE(is_proxy_present);
}

TEST_F(AttestationCAClientAttestationServerTest,
       ProxyAssumedToBePresentUponError) {
  StrictMock<MockNetworkContext> network_context;
  EXPECT_CALL(network_context, LookUpProxyForURL(_, _, _));
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present = false;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present));
  base::RunLoop().RunUntilIdle();
  EXPECT_TRUE(is_proxy_present);
}

TEST_F(AttestationCAClientAttestationServerTest, CheckProxyMultipleCalls) {
  base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
      chromeos::switches::kAttestationServer, "default");
  StrictMock<MockNetworkContext> network_context;
  network_context.SetProxyPresence(GURL("https://chromeos-ca.gstatic.com"),
                                   true);
  EXPECT_CALL(network_context,
              LookUpProxyForURL(GURL("https://chromeos-ca.gstatic.com"), _, _))
      .Times(2);
  AttestationCAClient client;
  client.set_network_context_for_testing(&network_context);
  bool is_proxy_present1 = false;
  bool is_proxy_present2 = false;
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present1));
  client.CheckIfAnyProxyPresent(base::BindOnce(
      [](bool* result, bool is_present) { *result = is_present; },
      &is_proxy_present2));
  base::RunLoop().RunUntilIdle();
  EXPECT_TRUE(is_proxy_present1);
  EXPECT_TRUE(is_proxy_present2);
}

}  // namespace attestation
}  // namespace ash