3 * Copyright 2015 gRPC authors.
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
21 #include <grpc/grpc.h>
22 #include <grpc/support/alloc.h>
24 #include "src/core/ext/filters/client_channel/resolver.h"
25 #include "src/core/ext/filters/client_channel/resolver/dns/c_ares/grpc_ares_wrapper.h"
26 #include "src/core/ext/filters/client_channel/resolver_registry.h"
27 #include "src/core/ext/filters/client_channel/server_address.h"
28 #include "src/core/lib/channel/channel_args.h"
29 #include "src/core/lib/iomgr/resolve_address.h"
30 #include "src/core/lib/iomgr/timer.h"
31 #include "src/core/lib/iomgr/work_serializer.h"
32 #include "test/core/util/test_config.h"
35 static bool g_fail_resolution = true;
36 static std::shared_ptr<grpc_core::WorkSerializer>* g_work_serializer;
38 static void my_resolve_address(const char* addr, const char* /*default_port*/,
39 grpc_pollset_set* /*interested_parties*/,
40 grpc_closure* on_done,
41 grpc_resolved_addresses** addrs) {
43 GPR_ASSERT(0 == strcmp("test", addr));
44 grpc_error* error = GRPC_ERROR_NONE;
45 if (g_fail_resolution) {
46 g_fail_resolution = false;
48 error = GRPC_ERROR_CREATE_FROM_STATIC_STRING("Forced Failure");
51 *addrs = static_cast<grpc_resolved_addresses*>(gpr_malloc(sizeof(**addrs)));
53 (*addrs)->addrs = static_cast<grpc_resolved_address*>(
54 gpr_malloc(sizeof(*(*addrs)->addrs)));
55 (*addrs)->addrs[0].len = 123;
57 grpc_core::ExecCtx::Run(DEBUG_LOCATION, on_done, error);
60 static grpc_address_resolver_vtable test_resolver = {my_resolve_address,
63 static grpc_ares_request* my_dns_lookup_ares_locked(
64 const char* /*dns_server*/, const char* addr, const char* /*default_port*/,
65 grpc_pollset_set* /*interested_parties*/, grpc_closure* on_done,
66 std::unique_ptr<grpc_core::ServerAddressList>* addresses,
67 std::unique_ptr<grpc_core::ServerAddressList>* /*balancer_addresses*/,
68 char** /*service_config_json*/, int /*query_timeout_ms*/,
69 std::shared_ptr<grpc_core::WorkSerializer> /*combiner*/) { // NOLINT
71 GPR_ASSERT(0 == strcmp("test", addr));
72 grpc_error* error = GRPC_ERROR_NONE;
73 if (g_fail_resolution) {
74 g_fail_resolution = false;
76 error = GRPC_ERROR_CREATE_FROM_STATIC_STRING("Forced Failure");
79 *addresses = absl::make_unique<grpc_core::ServerAddressList>();
80 grpc_resolved_address dummy_resolved_address;
81 memset(&dummy_resolved_address, 0, sizeof(dummy_resolved_address));
82 dummy_resolved_address.len = 123;
83 (*addresses)->emplace_back(dummy_resolved_address, nullptr);
85 grpc_core::ExecCtx::Run(DEBUG_LOCATION, on_done, error);
89 static void my_cancel_ares_request_locked(grpc_ares_request* request) {
90 GPR_ASSERT(request == nullptr);
93 static grpc_core::OrphanablePtr<grpc_core::Resolver> create_resolver(
95 std::unique_ptr<grpc_core::Resolver::ResultHandler> result_handler) {
96 grpc_core::ResolverFactory* factory =
97 grpc_core::ResolverRegistry::LookupResolverFactory("dns");
98 grpc_uri* uri = grpc_uri_parse(name, false);
100 grpc_core::ResolverArgs args;
102 args.work_serializer = *g_work_serializer;
103 args.result_handler = std::move(result_handler);
104 grpc_core::OrphanablePtr<grpc_core::Resolver> resolver =
105 factory->CreateResolver(std::move(args));
106 grpc_uri_destroy(uri);
110 class ResultHandler : public grpc_core::Resolver::ResultHandler {
112 struct ResolverOutput {
113 grpc_core::Resolver::Result result;
114 grpc_error* error = nullptr;
117 ResolverOutput() { gpr_event_init(&ev); }
118 ~ResolverOutput() { GRPC_ERROR_UNREF(error); }
121 void SetOutput(ResolverOutput* output) {
122 gpr_atm_rel_store(&output_, reinterpret_cast<gpr_atm>(output));
125 void ReturnResult(grpc_core::Resolver::Result result) override {
126 ResolverOutput* output =
127 reinterpret_cast<ResolverOutput*>(gpr_atm_acq_load(&output_));
128 GPR_ASSERT(output != nullptr);
129 output->result = std::move(result);
130 output->error = GRPC_ERROR_NONE;
131 gpr_event_set(&output->ev, (void*)1);
134 void ReturnError(grpc_error* error) override {
135 ResolverOutput* output =
136 reinterpret_cast<ResolverOutput*>(gpr_atm_acq_load(&output_));
137 GPR_ASSERT(output != nullptr);
138 output->error = error;
139 gpr_event_set(&output->ev, (void*)1);
143 gpr_atm output_ = 0; // ResolverOutput*
146 // interleave waiting for an event with a timer check
147 static bool wait_loop(int deadline_seconds, gpr_event* ev) {
148 while (deadline_seconds) {
149 gpr_log(GPR_DEBUG, "Test: waiting for %d more seconds", deadline_seconds);
150 if (gpr_event_wait(ev, grpc_timeout_seconds_to_deadline(1))) return true;
153 grpc_core::ExecCtx exec_ctx;
154 grpc_timer_check(nullptr);
159 int main(int argc, char** argv) {
160 grpc::testing::TestEnvironment env(argc, argv);
164 auto work_serializer = std::make_shared<grpc_core::WorkSerializer>();
165 g_work_serializer = &work_serializer;
166 grpc_set_resolver_impl(&test_resolver);
167 grpc_dns_lookup_ares_locked = my_dns_lookup_ares_locked;
168 grpc_cancel_ares_request_locked = my_cancel_ares_request_locked;
171 grpc_core::ExecCtx exec_ctx;
172 ResultHandler* result_handler = new ResultHandler();
173 grpc_core::OrphanablePtr<grpc_core::Resolver> resolver = create_resolver(
175 std::unique_ptr<grpc_core::Resolver::ResultHandler>(result_handler));
176 ResultHandler::ResolverOutput output1;
177 result_handler->SetOutput(&output1);
178 resolver->StartLocked();
179 grpc_core::ExecCtx::Get()->Flush();
180 GPR_ASSERT(wait_loop(5, &output1.ev));
181 GPR_ASSERT(output1.result.addresses.empty());
182 GPR_ASSERT(output1.error != GRPC_ERROR_NONE);
184 ResultHandler::ResolverOutput output2;
185 result_handler->SetOutput(&output2);
186 grpc_core::ExecCtx::Get()->Flush();
187 GPR_ASSERT(wait_loop(30, &output2.ev));
188 GPR_ASSERT(!output2.result.addresses.empty());
189 GPR_ASSERT(output2.error == GRPC_ERROR_NONE);
193 gpr_mu_destroy(&g_mu);