| // Copyright 2023 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/manta/orca_provider.h" |
| |
| #include <memory> |
| #include <string> |
| |
| #include "base/strings/stringprintf.h" |
| #include "base/test/bind.h" |
| #include "base/test/metrics/histogram_tester.h" |
| #include "base/test/task_environment.h" |
| #include "base/time/time.h" |
| #include "components/manta/base_provider.h" |
| #include "components/manta/base_provider_test_helper.h" |
| #include "components/manta/manta_status.h" |
| #include "components/manta/proto/rpc_status.pb.h" |
| #include "components/manta/provider_params.h" |
| #include "components/signin/public/base/consent_level.h" |
| #include "components/signin/public/identity_manager/identity_manager.h" |
| #include "components/signin/public/identity_manager/identity_test_environment.h" |
| #include "net/base/net_errors.h" |
| #include "net/http/http_status_code.h" |
| #include "net/http/http_util.h" |
| #include "net/traffic_annotation/network_traffic_annotation_test_helper.h" |
| #include "services/network/public/cpp/weak_wrapper_shared_url_loader_factory.h" |
| #include "services/network/test/test_url_loader_factory.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace manta { |
| |
| namespace { |
| constexpr char kMockEndpoint[] = "https://my-endpoint.com"; |
| } |
| |
| class FakeOrcaProvider : public OrcaProvider, public FakeBaseProvider { |
| public: |
| FakeOrcaProvider( |
| scoped_refptr<network::SharedURLLoaderFactory> test_url_loader_factory, |
| signin::IdentityManager* identity_manager) |
| : BaseProvider(test_url_loader_factory, identity_manager), |
| OrcaProvider(test_url_loader_factory, |
| identity_manager, |
| ProviderParams()), |
| FakeBaseProvider(test_url_loader_factory, identity_manager) {} |
| }; |
| |
| class OrcaProviderTest : public BaseProviderTest { |
| public: |
| OrcaProviderTest() = default; |
| |
| OrcaProviderTest(const OrcaProviderTest&) = delete; |
| OrcaProviderTest& operator=(const OrcaProviderTest&) = delete; |
| |
| ~OrcaProviderTest() override = default; |
| |
| std::unique_ptr<FakeOrcaProvider> CreateOrcaProvider() { |
| return std::make_unique<FakeOrcaProvider>( |
| base::MakeRefCounted<network::WeakWrapperSharedURLLoaderFactory>( |
| &test_url_loader_factory_), |
| identity_test_env_->identity_manager()); |
| } |
| }; |
| |
| // Test OrcaProvider rejects invalid input data. Currently we require the |
| // input must contain a valid tone. |
| TEST_F(OrcaProviderTest, PrepareRequestFailure) { |
| std::map<std::string, std::string> input = {{"data", "simple post data"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, /*response_data=*/"", |
| net::HTTP_OK, net::OK); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kInvalidInput); |
| quit_closure.Run(); |
| })); |
| |
| task_environment_.RunUntilQuit(); |
| } |
| |
| // Test that responses with http_status_code != net::HTTP_OK are captured. |
| TEST_F(OrcaProviderTest, CaptureUnexcpetedStatusCode) { |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, /*response_data=*/"", |
| net::HTTP_BAD_REQUEST, net::OK); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kBackendFailure); |
| quit_closure.Run(); |
| })); |
| task_environment_.RunUntilQuit(); |
| } |
| |
| TEST_F(OrcaProviderTest, CaptureNetError) { |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, /*response_data=*/"", |
| net::HTTP_OK, net::ERR_FAILED); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kNoInternetConnection); |
| quit_closure.Run(); |
| })); |
| task_environment_.RunUntilQuit(); |
| } |
| |
| // Test that malformed proto data can be captured with proper error. |
| TEST_F(OrcaProviderTest, ParseMalformedSerializedProto) { |
| std::string post_data = "{invalid proto"; |
| |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, /*response_data=*/post_data, |
| net::HTTP_OK, net::OK); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kMalformedResponse); |
| EXPECT_EQ(manta_status.message, ""); |
| EXPECT_EQ(manta_status.locale, ""); |
| quit_closure.Run(); |
| })); |
| |
| task_environment_.RunUntilQuit(); |
| } |
| |
| // Test that an unexpected response with a serialized RpcStatus proto can be |
| // handled properly. |
| TEST_F(OrcaProviderTest, ParseRpcStatusFromFailedResponse) { |
| proto::RpcStatus rpc_status; |
| rpc_status.set_code(3); // Will be mapped to kInvalidInput. |
| rpc_status.set_message("foo"); |
| |
| proto::RpcLocalizedMessage localize_message; |
| localize_message.set_message("bar"); |
| localize_message.set_locale("en"); |
| |
| auto* detail = rpc_status.add_details(); |
| detail->set_type_url("type.googleapis.com/google.rpc.LocalizedMessage"); |
| detail->set_value(localize_message.SerializeAsString()); |
| |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, |
| /*response_data=*/rpc_status.SerializeAsString(), |
| net::HTTP_BAD_REQUEST, net::OK); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kInvalidInput); |
| EXPECT_EQ(manta_status.message, "bar"); |
| EXPECT_EQ(manta_status.locale, "en"); |
| quit_closure.Run(); |
| })); |
| |
| task_environment_.RunUntilQuit(); |
| |
| // ErrorInfo.reason can be mapped to kRestrictedCountry and override the manta |
| // status code. |
| proto::RpcErrorInfo error_info; |
| error_info.set_reason("RESTRICTED_COUNTRY"); |
| error_info.set_domain("aratea-pa.googleapis.com"); |
| detail = rpc_status.add_details(); |
| detail->set_type_url("type.googleapis.com/google.rpc.ErrorInfo"); |
| detail->set_value(error_info.SerializeAsString()); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, |
| /*response_data=*/rpc_status.SerializeAsString(), |
| net::HTTP_BAD_REQUEST, net::OK); |
| |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, |
| MantaStatusCode::kRestrictedCountry); |
| EXPECT_EQ(manta_status.message, "bar"); |
| EXPECT_EQ(manta_status.locale, "en"); |
| quit_closure.Run(); |
| })); |
| |
| task_environment_.RunUntilQuit(); |
| } |
| |
| // Test a successful response can be parsed as base::Value::Dict. |
| TEST_F(OrcaProviderTest, ParseSuccessfulResponse) { |
| proto::Response response; |
| proto::OutputData& output_data = *response.add_output_data(); |
| output_data.set_text("foo"); |
| |
| base::HistogramTester histogram_tester; |
| |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| SetEndpointMockResponse(GURL{kMockEndpoint}, |
| /*response_data=*/response.SerializeAsString(), |
| net::HTTP_OK, net::OK); |
| |
| orca_provider->Call( |
| input, |
| base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict response, MantaStatus manta_status) { |
| EXPECT_EQ(manta_status.status_code, MantaStatusCode::kOk); |
| |
| EXPECT_TRUE(response.contains("outputData")); |
| |
| const auto* output_data_list = response.FindList("outputData"); |
| EXPECT_EQ(output_data_list->size(), 1u); |
| |
| const base::Value& front_element = output_data_list->front(); |
| EXPECT_TRUE(front_element.is_dict()); |
| EXPECT_EQ(*(front_element.GetDict().FindString("text")), "foo"); |
| |
| quit_closure.Run(); |
| })); |
| task_environment_.RunUntilQuit(); |
| histogram_tester.ExpectTotalCount("Ash.MantaService.OrcaProvider.TimeCost", |
| 1); |
| } |
| |
| TEST_F(OrcaProviderTest, EmptyResponseAfterIdentityManagerShutdown) { |
| std::unique_ptr<FakeOrcaProvider> orca_provider = CreateOrcaProvider(); |
| |
| identity_test_env_.reset(); |
| |
| std::map<std::string, std::string> input = {{"data", "simple post data"}, |
| {"tone", "SHORTEN"}}; |
| orca_provider->Call( |
| input, base::BindLambdaForTesting( |
| [quit_closure = task_environment_.QuitClosure()]( |
| base::Value::Dict dict, MantaStatus manta_status) { |
| ASSERT_TRUE(dict.empty()); |
| ASSERT_EQ(MantaStatusCode::kNoIdentityManager, |
| manta_status.status_code); |
| quit_closure.Run(); |
| })); |
| task_environment_.RunUntilQuit(); |
| } |
| |
| } // namespace manta |