/* * * Copyright 2015 gRPC authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ #include #include "src/core/ext/filters/client_channel/resolver_registry.h" #include #include #include #include namespace grpc_core { namespace { class RegistryState { public: RegistryState() : default_prefix_(gpr_strdup("dns:///")) {} void SetDefaultPrefix(const char* default_resolver_prefix) { GPR_ASSERT(default_resolver_prefix != nullptr); GPR_ASSERT(*default_resolver_prefix != '\0'); default_prefix_.reset(gpr_strdup(default_resolver_prefix)); } void RegisterResolverFactory(UniquePtr factory) { for (size_t i = 0; i < factories_.size(); ++i) { GPR_ASSERT(strcmp(factories_[i]->scheme(), factory->scheme()) != 0); } factories_.push_back(std::move(factory)); } ResolverFactory* LookupResolverFactory(const char* scheme) const { for (size_t i = 0; i < factories_.size(); ++i) { if (strcmp(scheme, factories_[i]->scheme()) == 0) { return factories_[i].get(); } } return nullptr; } // Returns the factory for the scheme of \a target. If \a target does // not parse as a URI, prepends \a default_prefix_ and tries again. // If URI parsing is successful (in either attempt), sets \a uri to // point to the parsed URI. // If \a default_prefix_ needs to be prepended, sets \a canonical_target // to the canonical target string. ResolverFactory* FindResolverFactory(const char* target, grpc_uri** uri, char** canonical_target) const { GPR_ASSERT(uri != nullptr); *uri = grpc_uri_parse(target, 1); ResolverFactory* factory = *uri == nullptr ? nullptr : LookupResolverFactory((*uri)->scheme); if (factory == nullptr) { grpc_uri_destroy(*uri); gpr_asprintf(canonical_target, "%s%s", default_prefix_.get(), target); *uri = grpc_uri_parse(*canonical_target, 1); factory = *uri == nullptr ? nullptr : LookupResolverFactory((*uri)->scheme); if (factory == nullptr) { grpc_uri_destroy(grpc_uri_parse(target, 0)); grpc_uri_destroy(grpc_uri_parse(*canonical_target, 0)); gpr_log(GPR_ERROR, "don't know how to resolve '%s' or '%s'", target, *canonical_target); } } return factory; } private: // We currently support 10 factories without doing additional // allocation. This number could be raised if there is a case where // more factories are needed and the additional allocations are // hurting performance (which is unlikely, since these allocations // only occur at gRPC initialization time). InlinedVector, 10> factories_; UniquePtr default_prefix_; }; static RegistryState* g_state = nullptr; } // namespace // // ResolverRegistry::Builder // void ResolverRegistry::Builder::InitRegistry() { if (g_state == nullptr) g_state = New(); } void ResolverRegistry::Builder::ShutdownRegistry() { Delete(g_state); g_state = nullptr; } void ResolverRegistry::Builder::SetDefaultPrefix( const char* default_resolver_prefix) { InitRegistry(); g_state->SetDefaultPrefix(default_resolver_prefix); } void ResolverRegistry::Builder::RegisterResolverFactory( UniquePtr factory) { InitRegistry(); g_state->RegisterResolverFactory(std::move(factory)); } // // ResolverRegistry // ResolverFactory* ResolverRegistry::LookupResolverFactory(const char* scheme) { GPR_ASSERT(g_state != nullptr); return g_state->LookupResolverFactory(scheme); } OrphanablePtr ResolverRegistry::CreateResolver( const char* target, const grpc_channel_args* args, grpc_pollset_set* pollset_set, grpc_combiner* combiner) { GPR_ASSERT(g_state != nullptr); grpc_uri* uri = nullptr; char* canonical_target = nullptr; ResolverFactory* factory = g_state->FindResolverFactory(target, &uri, &canonical_target); ResolverArgs resolver_args; resolver_args.uri = uri; resolver_args.args = args; resolver_args.pollset_set = pollset_set; resolver_args.combiner = combiner; OrphanablePtr resolver = factory == nullptr ? nullptr : factory->CreateResolver(resolver_args); grpc_uri_destroy(uri); gpr_free(canonical_target); return resolver; } UniquePtr ResolverRegistry::GetDefaultAuthority(const char* target) { GPR_ASSERT(g_state != nullptr); grpc_uri* uri = nullptr; char* canonical_target = nullptr; ResolverFactory* factory = g_state->FindResolverFactory(target, &uri, &canonical_target); UniquePtr authority = factory == nullptr ? nullptr : factory->GetDefaultAuthority(uri); grpc_uri_destroy(uri); gpr_free(canonical_target); return authority; } UniquePtr ResolverRegistry::AddDefaultPrefixIfNeeded(const char* target) { GPR_ASSERT(g_state != nullptr); grpc_uri* uri = nullptr; char* canonical_target = nullptr; g_state->FindResolverFactory(target, &uri, &canonical_target); grpc_uri_destroy(uri); return UniquePtr(canonical_target == nullptr ? gpr_strdup(target) : canonical_target); } } // namespace grpc_core