--- /dev/null
+//
+// 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 <grpc/support/port_platform.h>
+
+#include "src/core/ext/filters/client_channel/service_config.h"
+
+#include <string.h>
+
+#include <grpc/impl/codegen/grpc_types.h>
+#include <grpc/support/alloc.h>
+#include <grpc/support/log.h>
+#include <grpc/support/string_util.h>
+
+#include "src/core/lib/gpr/string.h"
+#include "src/core/lib/json/json.h"
+#include "src/core/lib/slice/slice_hash_table.h"
+#include "src/core/lib/slice/slice_internal.h"
+#include "src/core/lib/slice/slice_string_helpers.h"
+
+namespace grpc_core {
+
+namespace {
+typedef InlinedVector<UniquePtr<ServiceConfig::Parser>,
+ ServiceConfig::kNumPreallocatedParsers>
+ ServiceConfigParserList;
+ServiceConfigParserList* g_registered_parsers;
+} // namespace
+
+RefCountedPtr<ServiceConfig> ServiceConfig::Create(const char* json,
+ grpc_error** error) {
+ UniquePtr<char> service_config_json(gpr_strdup(json));
+ UniquePtr<char> json_string(gpr_strdup(json));
+ GPR_DEBUG_ASSERT(error != nullptr);
+ grpc_json* json_tree = grpc_json_parse_string(json_string.get());
+ if (json_tree == nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "failed to parse JSON for service config");
+ return nullptr;
+ }
+ return MakeRefCounted<ServiceConfig>(
+ std::move(service_config_json), std::move(json_string), json_tree, error);
+}
+
+ServiceConfig::ServiceConfig(UniquePtr<char> service_config_json,
+ UniquePtr<char> json_string, grpc_json* json_tree,
+ grpc_error** error)
+ : service_config_json_(std::move(service_config_json)),
+ json_string_(std::move(json_string)),
+ json_tree_(json_tree) {
+ GPR_DEBUG_ASSERT(error != nullptr);
+ if (json_tree->type != GRPC_JSON_OBJECT || json_tree->key != nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "Malformed service Config JSON object");
+ return;
+ }
+ grpc_error* error_list[2];
+ int error_count = 0;
+ grpc_error* global_error = ParseGlobalParams(json_tree);
+ grpc_error* local_error = ParsePerMethodParams(json_tree);
+ if (global_error != GRPC_ERROR_NONE) {
+ error_list[error_count++] = global_error;
+ }
+ if (local_error != GRPC_ERROR_NONE) {
+ error_list[error_count++] = local_error;
+ }
+ if (error_count > 0) {
+ *error = GRPC_ERROR_CREATE_REFERENCING_FROM_STATIC_STRING(
+ "Service config parsing error", error_list, error_count);
+ GRPC_ERROR_UNREF(global_error);
+ GRPC_ERROR_UNREF(local_error);
+ }
+}
+
+grpc_error* ServiceConfig::ParseGlobalParams(const grpc_json* json_tree) {
+ GPR_DEBUG_ASSERT(json_tree_->type == GRPC_JSON_OBJECT);
+ GPR_DEBUG_ASSERT(json_tree_->key == nullptr);
+ InlinedVector<grpc_error*, 4> error_list;
+ for (size_t i = 0; i < g_registered_parsers->size(); i++) {
+ grpc_error* parser_error = GRPC_ERROR_NONE;
+ auto parsed_obj =
+ (*g_registered_parsers)[i]->ParseGlobalParams(json_tree, &parser_error);
+ if (parser_error != GRPC_ERROR_NONE) {
+ error_list.push_back(parser_error);
+ }
+ parsed_global_configs_.push_back(std::move(parsed_obj));
+ }
+ return GRPC_ERROR_CREATE_FROM_VECTOR("Global Params", &error_list);
+}
+
+grpc_error* ServiceConfig::ParseJsonMethodConfigToServiceConfigVectorTable(
+ const grpc_json* json,
+ SliceHashTable<const ParsedConfigVector*>::Entry* entries, size_t* idx) {
+ auto objs_vector = MakeUnique<ParsedConfigVector>();
+ InlinedVector<grpc_error*, 4> error_list;
+ for (size_t i = 0; i < g_registered_parsers->size(); i++) {
+ grpc_error* parser_error = GRPC_ERROR_NONE;
+ auto parsed_obj =
+ (*g_registered_parsers)[i]->ParsePerMethodParams(json, &parser_error);
+ if (parser_error != GRPC_ERROR_NONE) {
+ error_list.push_back(parser_error);
+ }
+ objs_vector->push_back(std::move(parsed_obj));
+ }
+ parsed_method_config_vectors_storage_.push_back(std::move(objs_vector));
+ const auto* vector_ptr =
+ parsed_method_config_vectors_storage_
+ [parsed_method_config_vectors_storage_.size() - 1]
+ .get();
+ // Construct list of paths.
+ InlinedVector<UniquePtr<char>, 10> paths;
+ for (grpc_json* child = json->child; child != nullptr; child = child->next) {
+ if (child->key == nullptr) continue;
+ if (strcmp(child->key, "name") == 0) {
+ if (child->type != GRPC_JSON_ARRAY) {
+ error_list.push_back(GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error:not of type Array"));
+ goto wrap_error;
+ }
+ for (grpc_json* name = child->child; name != nullptr; name = name->next) {
+ grpc_error* parse_error = GRPC_ERROR_NONE;
+ UniquePtr<char> path = ParseJsonMethodName(name, &parse_error);
+ if (path == nullptr) {
+ error_list.push_back(parse_error);
+ } else {
+ GPR_DEBUG_ASSERT(parse_error == GRPC_ERROR_NONE);
+ paths.push_back(std::move(path));
+ }
+ }
+ }
+ }
+ if (paths.size() == 0) {
+ error_list.push_back(
+ GRPC_ERROR_CREATE_FROM_STATIC_STRING("No names specified"));
+ }
+ // Add entry for each path.
+ for (size_t i = 0; i < paths.size(); ++i) {
+ entries[*idx].key = grpc_slice_from_copied_string(paths[i].get());
+ entries[*idx].value = vector_ptr;
+ ++*idx;
+ }
+wrap_error:
+ return GRPC_ERROR_CREATE_FROM_VECTOR("methodConfig", &error_list);
+}
+
+grpc_error* ServiceConfig::ParsePerMethodParams(const grpc_json* json_tree) {
+ GPR_DEBUG_ASSERT(json_tree_->type == GRPC_JSON_OBJECT);
+ GPR_DEBUG_ASSERT(json_tree_->key == nullptr);
+ SliceHashTable<const ParsedConfigVector*>::Entry* entries = nullptr;
+ size_t num_entries = 0;
+ InlinedVector<grpc_error*, 4> error_list;
+ for (grpc_json* field = json_tree->child; field != nullptr;
+ field = field->next) {
+ if (field->key == nullptr) {
+ error_list.push_back(GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "error:Illegal key value - NULL"));
+ continue;
+ }
+ if (strcmp(field->key, "methodConfig") == 0) {
+ if (entries != nullptr) {
+ GPR_ASSERT(false);
+ }
+ if (field->type != GRPC_JSON_ARRAY) {
+ error_list.push_back(GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:methodConfig error:not of type Array"));
+ }
+ for (grpc_json* method = field->child; method != nullptr;
+ method = method->next) {
+ int count = CountNamesInMethodConfig(method);
+ if (count <= 0) {
+ error_list.push_back(GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:methodConfig error:No names found"));
+ }
+ num_entries += static_cast<size_t>(count);
+ }
+ entries = static_cast<SliceHashTable<const ParsedConfigVector*>::Entry*>(
+ gpr_zalloc(num_entries *
+ sizeof(SliceHashTable<const ParsedConfigVector*>::Entry)));
+ size_t idx = 0;
+ for (grpc_json* method = field->child; method != nullptr;
+ method = method->next) {
+ grpc_error* error = ParseJsonMethodConfigToServiceConfigVectorTable(
+ method, entries, &idx);
+ if (error != GRPC_ERROR_NONE) {
+ error_list.push_back(error);
+ }
+ }
+ // idx might not be equal to num_entries due to parsing errors
+ num_entries = idx;
+ break;
+ }
+ }
+ if (entries != nullptr) {
+ parsed_method_configs_table_ =
+ SliceHashTable<const ParsedConfigVector*>::Create(num_entries, entries,
+ nullptr);
+ gpr_free(entries);
+ }
+ return GRPC_ERROR_CREATE_FROM_VECTOR("Method Params", &error_list);
+}
+
+ServiceConfig::~ServiceConfig() { grpc_json_destroy(json_tree_); }
+
+int ServiceConfig::CountNamesInMethodConfig(grpc_json* json) {
+ int num_names = 0;
+ for (grpc_json* field = json->child; field != nullptr; field = field->next) {
+ if (field->key != nullptr && strcmp(field->key, "name") == 0) {
+ if (field->type != GRPC_JSON_ARRAY) return -1;
+ for (grpc_json* name = field->child; name != nullptr; name = name->next) {
+ if (name->type != GRPC_JSON_OBJECT) return -1;
+ ++num_names;
+ }
+ }
+ }
+ return num_names;
+}
+
+UniquePtr<char> ServiceConfig::ParseJsonMethodName(grpc_json* json,
+ grpc_error** error) {
+ if (json->type != GRPC_JSON_OBJECT) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error:type is not object");
+ return nullptr;
+ }
+ const char* service_name = nullptr;
+ const char* method_name = nullptr;
+ for (grpc_json* child = json->child; child != nullptr; child = child->next) {
+ if (child->key == nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error:Child entry with no key");
+ return nullptr;
+ }
+ if (child->type != GRPC_JSON_STRING) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error:Child entry not of type string");
+ return nullptr;
+ }
+ if (strcmp(child->key, "service") == 0) {
+ if (service_name != nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error: field:service error:Multiple entries");
+ return nullptr; // Duplicate.
+ }
+ if (child->value == nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error: field:service error:empty value");
+ return nullptr;
+ }
+ service_name = child->value;
+ } else if (strcmp(child->key, "method") == 0) {
+ if (method_name != nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error: field:method error:multiple entries");
+ return nullptr; // Duplicate.
+ }
+ if (child->value == nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error: field:method error:empty value");
+ return nullptr;
+ }
+ method_name = child->value;
+ }
+ }
+ if (service_name == nullptr) {
+ *error = GRPC_ERROR_CREATE_FROM_STATIC_STRING(
+ "field:name error: field:service error:not found");
+ return nullptr; // Required field.
+ }
+ char* path;
+ gpr_asprintf(&path, "/%s/%s", service_name,
+ method_name == nullptr ? "*" : method_name);
+ return UniquePtr<char>(path);
+}
+
+const ServiceConfig::ParsedConfigVector*
+ServiceConfig::GetMethodParsedConfigVector(const grpc_slice& path) {
+ if (parsed_method_configs_table_.get() == nullptr) {
+ return nullptr;
+ }
+ const auto* value = parsed_method_configs_table_->Get(path);
+ // If we didn't find a match for the path, try looking for a wildcard
+ // entry (i.e., change "/service/method" to "/service/*").
+ if (value == nullptr) {
+ char* path_str = grpc_slice_to_c_string(path);
+ const char* sep = strrchr(path_str, '/') + 1;
+ const size_t len = (size_t)(sep - path_str);
+ char* buf = (char*)gpr_malloc(len + 2); // '*' and NUL
+ memcpy(buf, path_str, len);
+ buf[len] = '*';
+ buf[len + 1] = '\0';
+ grpc_slice wildcard_path = grpc_slice_from_copied_string(buf);
+ gpr_free(buf);
+ value = parsed_method_configs_table_->Get(wildcard_path);
+ grpc_slice_unref_internal(wildcard_path);
+ gpr_free(path_str);
+ if (value == nullptr) return nullptr;
+ }
+ return *value;
+}
+
+size_t ServiceConfig::RegisterParser(UniquePtr<Parser> parser) {
+ g_registered_parsers->push_back(std::move(parser));
+ return g_registered_parsers->size() - 1;
+}
+
+void ServiceConfig::Init() {
+ GPR_ASSERT(g_registered_parsers == nullptr);
+ g_registered_parsers = New<ServiceConfigParserList>();
+}
+
+void ServiceConfig::Shutdown() {
+ Delete(g_registered_parsers);
+ g_registered_parsers = nullptr;
+}
+
+} // namespace grpc_core