diff --git a/src/rdsn/include/dsn/dist/replication/replica_envs.h b/src/rdsn/include/dsn/dist/replication/replica_envs.h index f4c0d58294..4db367a7fa 100644 --- a/src/rdsn/include/dsn/dist/replication/replica_envs.h +++ b/src/rdsn/include/dsn/dist/replication/replica_envs.h @@ -56,6 +56,7 @@ class replica_envs static const std::string MANUAL_COMPACT_PERIODIC_BOTTOMMOST_LEVEL_COMPACTION; static const std::string BUSINESS_INFO; static const std::string REPLICA_ACCESS_CONTROLLER_ALLOWED_USERS; + static const std::string REPLICA_ACCESS_CONTROLLER_RANGER_POLICIES; static const std::string READ_QPS_THROTTLING; static const std::string READ_SIZE_THROTTLING; static const std::string BACKUP_REQUEST_QPS_THROTTLING; diff --git a/src/rdsn/include/dsn/dist/replication/replication.codes.h b/src/rdsn/include/dsn/dist/replication/replication.codes.h index 2766d4a3a9..ffb8e13c4b 100644 --- a/src/rdsn/include/dsn/dist/replication/replication.codes.h +++ b/src/rdsn/include/dsn/dist/replication/replication.codes.h @@ -127,6 +127,7 @@ MAKE_EVENT_CODE_RPC(RPC_CM_START_MANUAL_COMPACT, TASK_PRIORITY_COMMON) MAKE_EVENT_CODE_RPC(RPC_CM_QUERY_MANUAL_COMPACT_STATUS, TASK_PRIORITY_COMMON) MAKE_EVENT_CODE_RPC(RPC_CM_GET_MAX_REPLICA_COUNT, TASK_PRIORITY_COMMON) MAKE_EVENT_CODE_RPC(RPC_CM_SET_MAX_REPLICA_COUNT, TASK_PRIORITY_COMMON) +MAKE_EVENT_CODE(LPC_USE_RANGER_ACCESS_CONTROL, TASK_PRIORITY_COMMON) #undef CURRENT_THREAD_POOL #define CURRENT_THREAD_POOL THREAD_POOL_META_STATE diff --git a/src/rdsn/include/dsn/utility/error_code.h b/src/rdsn/include/dsn/utility/error_code.h index e5a9169f6b..dc581ef58d 100644 --- a/src/rdsn/include/dsn/utility/error_code.h +++ b/src/rdsn/include/dsn/utility/error_code.h @@ -157,4 +157,8 @@ DEFINE_ERR_CODE(ERR_PARENT_PARTITION_MISUSED) DEFINE_ERR_CODE(ERR_CHILD_NOT_READY) DEFINE_ERR_CODE(ERR_DISK_INSUFFICIENT) DEFINE_ERR_CODE(ERR_RETRY_EXHAUSTED) + +DEFINE_ERR_CODE(ERR_SYNC_RANGER_POLICIES_FAILED) +DEFINE_ERR_CODE(ERR_RANGER_PARSE_ACL) +DEFINE_ERR_CODE(ERR_RANGER_POLICIES_NO_NEED_UPDATE) } // namespace dsn diff --git a/src/rdsn/src/common/replication_common.cpp b/src/rdsn/src/common/replication_common.cpp index 9d42b6fba7..3af885169c 100644 --- a/src/rdsn/src/common/replication_common.cpp +++ b/src/rdsn/src/common/replication_common.cpp @@ -653,6 +653,8 @@ const std::string replica_envs::ROCKSDB_BLOCK_CACHE_ENABLED("replica.rocksdb_blo const std::string replica_envs::BUSINESS_INFO("business.info"); const std::string replica_envs::REPLICA_ACCESS_CONTROLLER_ALLOWED_USERS( "replica_access_controller.allowed_users"); +const std::string replica_envs::REPLICA_ACCESS_CONTROLLER_RANGER_POLICIES( + "replica_access_controller.ranger_policies"); const std::string replica_envs::READ_QPS_THROTTLING("replica.read_throttling"); const std::string replica_envs::READ_SIZE_THROTTLING("replica.read_throttling_by_size"); const std::string diff --git a/src/rdsn/src/meta/app_env_validator.cpp b/src/rdsn/src/meta/app_env_validator.cpp index 8441fb3043..f42e97af8f 100644 --- a/src/rdsn/src/meta/app_env_validator.cpp +++ b/src/rdsn/src/meta/app_env_validator.cpp @@ -207,7 +207,8 @@ void app_env_validator::register_all_validators() {replica_envs::MANUAL_COMPACT_PERIODIC_TRIGGER_TIME, nullptr}, {replica_envs::MANUAL_COMPACT_PERIODIC_TARGET_LEVEL, nullptr}, {replica_envs::MANUAL_COMPACT_PERIODIC_BOTTOMMOST_LEVEL_COMPACTION, nullptr}, - {replica_envs::REPLICA_ACCESS_CONTROLLER_ALLOWED_USERS, nullptr}}; + {replica_envs::REPLICA_ACCESS_CONTROLLER_ALLOWED_USERS, nullptr}, + {replica_envs::REPLICA_ACCESS_CONTROLLER_RANGER_POLICIES, nullptr}}; } } // namespace replication diff --git a/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.cpp b/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.cpp index f47d4b7c66..5bab4e50f5 100644 --- a/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.cpp +++ b/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.cpp @@ -15,13 +15,65 @@ // specific language governing permissions and limitations // under the License. +#include +#include +#include +#include +#include +#include +#include + +// Disable class-memaccess warning to facilitate compilation with gcc>7 +// https://github.com/Tencent/rapidjson/issues/1700 +#pragma GCC diagnostic push +#if defined(__GNUC__) && __GNUC__ >= 8 +#pragma GCC diagnostic ignored "-Wclass-memaccess" +#endif +#include + +#pragma GCC diagnostic pop + +#include #include +#include "common/replication_common.h" +#include +#include "fmt/core.h" #include "meta/meta_options.h" +#include "meta/meta_service.h" +#include +#include "meta/server_state.h" +#include "meta_admin_types.h" #include "ranger_resource_policy_manager.h" +#include "rapidjson/allocators.h" +#include "runtime/ranger/ranger_resource_policy.h" +#include "rapidjson/allocators.h" +#include "runtime/ranger/ranger_resource_policy.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include namespace dsn { namespace ranger { +DSN_DEFINE_uint32("security", + update_ranger_policy_interval_sec, + 5, + "The interval seconds of meta " + "server to pull the latest " + "access control policy from " + "Ranger service."); +DSN_DEFINE_string("ranger", ranger_service_url, "", "Apache Ranger service url."); +DSN_DEFINE_string("ranger", + ranger_service_name, + "", + "The name of the policies defined in the Ranger service."); + #define RETURN_ERR_IF_MISSING_MEMBER(obj, member) \ do { \ if (!obj.IsObject() || !obj.HasMember(member)) { \ @@ -75,9 +127,11 @@ const std::map kAccessTypeMaping({{"READ", access_type {"CONTROL", access_type::kControl}}); } // anonymous namespace +const std::chrono::milliseconds kLoadRangerPolicyRetryDelayMs(10000); + ranger_resource_policy_manager::ranger_resource_policy_manager( dsn::replication::meta_service *meta_svc) - : _meta_svc(meta_svc) // , _local_policy_version(0) + : _meta_svc(meta_svc), _local_policy_version(-1) { _ranger_policy_meta_root = dsn::replication::meta_options::concat_path_unix_style( _meta_svc->cluster_root(), "ranger_policy_meta_root"); @@ -163,5 +217,372 @@ void ranger_resource_policy_manager::parse_policies_from_json(const rapidjson::V policies.emplace_back(pi); } } + +dsn::error_code ranger_resource_policy_manager::update_policies_from_ranger_service() +{ + std::string ranger_policies; + ERR_LOG_AND_RETURN_NOT_OK(pull_policies_from_ranger_service(&ranger_policies), + "Pull Ranger policies failed."); + ddebug("Pull Ranger policies success."); + + auto err_code = load_policies_from_json(ranger_policies); + if (err_code == dsn::ERR_RANGER_POLICIES_NO_NEED_UPDATE) { + ddebug("Skip to update local policies."); + // For the newly created table, its app envs must be empty. This needs to be executed + // periodically to update the table's app envs, regardless of whether the Ranger policy is + // updated or not. + ERR_LOG_AND_RETURN_NOT_OK(sync_policies_to_app_envs(), "Sync policies to app envs failed."); + return dsn::ERR_OK; + } + ERR_LOG_AND_RETURN_NOT_OK(err_code, "Parse Ranger policies failed."); + + start_to_dump_and_sync_policies(); + + return dsn::ERR_OK; +} + +dsn::error_code ranger_resource_policy_manager::pull_policies_from_ranger_service( + std::string *ranger_policies) const +{ + std::string cmd = + fmt::format("curl {}/{}", FLAGS_ranger_service_url, FLAGS_ranger_service_name); + std::stringstream resp; + if (dsn::utils::pipe_execute(cmd.c_str(), resp) != 0) { + return dsn::ERR_SYNC_RANGER_POLICIES_FAILED; + } + + *ranger_policies = resp.str(); + return dsn::ERR_OK; +} + +dsn::error_code ranger_resource_policy_manager::load_policies_from_json(const std::string &data) +{ + // The Ranger policy pulled from Ranger service demo. + /* + { + "serviceName": "PEGASUS1", + "serviceId": 1069, + "policyVersion": 60, + "policyUpdateTime": 1673254471000, + "policies": [{ + "id": 5334, + "guid": "c7918f8c-921a-4f3d-b9d7-bce7009ee5f8", + "isEnabled": true, + "version": 13, + "service": "PEGASUS1", + "name": "all - database", + "policyType": 0, + "policyPriority": 0, + "description": "Policy for all - database", + "isAuditEnabled": true, + "resources": { + "database": { + "values": ["PEGASUS1"], + "isExcludes": false, + "isRecursive": true + } + }, + "policyItems": [{ + "accesses": [{ + "type": "create", + "isAllowed": true + }, { + "type": "drop", + "isAllowed": true + }, { + "type": "control", + "isAllowed": true + }, { + "type": "metadata", + "isAllowed": true + }, { + "type": "list", + "isAllowed": true + }], + "users": ["PEGASUS1"], + "groups": [], + "roles": [], + "conditions": [], + "delegateAdmin": true + }], + "denyPolicyItems": [], + "allowExceptions": [], + "denyExceptions": [], + "dataMaskPolicyItems": [], + "rowFilterPolicyItems": [], + "serviceType": "pegasus", + "options": {}, + "validitySchedules": [], + "policyLabels": [], + "zoneName": "", + "isDenyAllElse": false + }], + "auditMode": "audit-default", + "serviceConfig": {} + } + */ + rapidjson::Document doc; + doc.Parse(data.c_str()); + + // Check if it is needed to update policies. + RETURN_ERR_IF_MISSING_MEMBER(doc, "policyVersion"); + int remote_policy_version = doc["policyVersion"].GetInt(); + if (_local_policy_version == remote_policy_version) { + ddebug_f("Ranger policy version: {}, no need to update.", _local_policy_version); + return dsn::ERR_RANGER_POLICIES_NO_NEED_UPDATE; + } + + if (_local_policy_version > remote_policy_version) { + dwarn_f("Local Ranger policy version ({}) is larger than remote version ({}), please " + "check Ranger services ({}).", + _local_policy_version, + remote_policy_version, + FLAGS_ranger_service_name); + return dsn::ERR_RANGER_POLICIES_NO_NEED_UPDATE; + } + + _local_policy_version = remote_policy_version; + + // Update policies. + _all_resource_policies.clear(); + + // TODO(wanghao): it's optional + // Provide a DATABASE default policy for legacy tables. + // ranger_resource_policy default_database_policy; + // ranger_resource_policy::create_default_database_policy(default_database_policy); + // _all_resource_policies[enum_to_string(resource_type::kDatabase)] = {default_database_policy}; + + RETURN_ERR_IF_MISSING_MEMBER(doc, "policies"); + const rapidjson::Value &policies = doc["policies"]; + RETURN_ERR_IF_NOT_ARRAY(policies); + for (const auto &policy : policies.GetArray()) { + RETURN_ERR_IF_MISSING_MEMBER(policy, "isEnabled"); + // 1. Check if the policy is enabled or not. + if (!policy["isEnabled"].IsBool() || !policy["isEnabled"].GetBool()) { + continue; + } + + // 2. Parse resource type. + RETURN_ERR_IF_MISSING_MEMBER(policy, "resources"); + std::map> values_of_resource_type; + for (const auto &resource : policy["resources"].GetObject()) { + RETURN_ERR_IF_MISSING_MEMBER(resource.value, "values"); + RETURN_ERR_IF_NOT_ARRAY((resource.value)["values"]); + std::unordered_set values; + for (const auto &v : (resource.value)["values"].GetArray()) { + values.insert(v.GetString()); + } + values_of_resource_type.emplace(std::make_pair(resource.name.GetString(), values)); + } + + // 3. Construct ACL policy. + ranger_resource_policy resource_policy; + CONTINUE_IF_MISSING_MEMBER(policy, "name"); + resource_policy.name = policy["name"].GetString(); + + resource_type rt = resource_type::kUnknown; + do { + // TODO(wanghao): refactor the following code + // parse Ranger policies json string into `values_of_resource_type`, distinguish + // resource types by `values_of_resource_type.size()` + if (values_of_resource_type.size() == 1) { + auto iter = values_of_resource_type.find("global"); + if (iter != values_of_resource_type.end()) { + rt = resource_type::kGlobal; + break; + } + iter = values_of_resource_type.find("database"); + if (iter != values_of_resource_type.end()) { + resource_policy.database_names = iter->second; + rt = resource_type::kDatabase; + break; + } + } else if (values_of_resource_type.size() == 2) { + auto iter1 = values_of_resource_type.find("database"); + auto iter2 = values_of_resource_type.find("table"); + if (iter1 != values_of_resource_type.end() && + iter2 != values_of_resource_type.end()) { + resource_policy.database_names = iter1->second; + resource_policy.table_names = iter2->second; + rt = resource_type::kDatabaseTable; + break; + } + } + return dsn::ERR_RANGER_PARSE_ACL; + } while (false); + + parse_policies_from_json(policy["policyItems"], resource_policy.policies.allow_policies); + parse_policies_from_json(policy["denyPolicyItems"], resource_policy.policies.deny_policies); + parse_policies_from_json(policy["allowExceptions"], + resource_policy.policies.allow_policies_exclude); + parse_policies_from_json(policy["denyExceptions"], + resource_policy.policies.deny_policies_exclude); + + // 4. Add the ACL policy. + auto ret = _all_resource_policies.emplace(enum_to_string(rt), + resource_policies({resource_policy})); + if (!ret.second) { + ret.first->second.emplace_back(resource_policy); + } + } + + return dsn::ERR_OK; +} + +void ranger_resource_policy_manager::start_to_dump_and_sync_policies() +{ + ddebug("Start to create Ranger policy meta root on remote storage."); + dsn::task_ptr sync_task = dsn::tasking::create_task( + LPC_USE_RANGER_ACCESS_CONTROL, &_tracker, [this]() { dump_and_sync_policies(); }); + _meta_svc->get_remote_storage()->create_node( + _ranger_policy_meta_root, + LPC_USE_RANGER_ACCESS_CONTROL, + [this, sync_task](dsn::error_code err) { + if (err == dsn::ERR_OK || err == dsn::ERR_NODE_ALREADY_EXIST) { + ddebug("Create Ranger policy meta root succeed."); + sync_task->enqueue(); + return; + } + dassert_f(err == dsn::ERR_TIMEOUT, + "error to create Ranger policy meta root, error = {}", + err); + derror("Create Ranger policy meta root timeout, retry later."); + dsn::tasking::enqueue(LPC_USE_RANGER_ACCESS_CONTROL, + &_tracker, + [this]() { start_to_dump_and_sync_policies(); }, + 0, + kLoadRangerPolicyRetryDelayMs); + }); +} + +void ranger_resource_policy_manager::dump_and_sync_policies() +{ + ddebug("Start to sync Ranger policies to remote storage."); + + dump_policies_to_remote_storage(); + ddebug("Dump Ranger policies to remote storage succeed."); + + update_cached_policies(); + ddebug("Update using resources policies succeed."); + + if (dsn::ERR_OK != sync_policies_to_app_envs()) { + derror("Sync policies to app envs failed."); + } +} + +void ranger_resource_policy_manager::dump_policies_to_remote_storage() +{ + dsn::blob value = json::json_forwarder::encode(_all_resource_policies); + _meta_svc->get_remote_storage()->set_data( + _ranger_policy_meta_root, value, LPC_USE_RANGER_ACCESS_CONTROL, [this](dsn::error_code e) { + if (e == dsn::ERR_OK) { + ddebug_f("Dump Ranger policies to remote storage succeed."); + return; + } + // The return error code is not 'ERR_TIMEOUT', use assert here. + dassert_f(e == dsn::ERR_TIMEOUT, + "error to dump Ranger policies to remote storage, error = {}", + e); + derror("Dump Ranger policies to remote storage timeout, retry later."); + dsn::tasking::enqueue(LPC_USE_RANGER_ACCESS_CONTROL, + &_tracker, + [this]() { dump_policies_to_remote_storage(); }, + 0, + kLoadRangerPolicyRetryDelayMs); + }); +} + +void ranger_resource_policy_manager::update_cached_policies() +{ + { + utils::auto_write_lock l(_global_policies_lock); + _global_policies_cache.swap(_all_resource_policies[enum_to_string(resource_type::kGlobal)]); + // TODO(wanghao): provide a query method + } + { + utils::auto_write_lock l(_database_policies_lock); + _database_policies_cache.swap( + _all_resource_policies[enum_to_string(resource_type::kDatabase)]); + // TODO(wanghao): provide a query method + } +} + +dsn::error_code ranger_resource_policy_manager::sync_policies_to_app_envs() +{ + const auto &table_policies = + _all_resource_policies.find(enum_to_string(resource_type::kDatabaseTable)); + if (table_policies == _all_resource_policies.end()) { + dinfo_f("DATABASE_TABLE level policy is empty, skip to sync app envs."); + return dsn::ERR_OK; + } + + dsn::replication::configuration_list_apps_response list_resp; + dsn::replication::configuration_list_apps_request list_req; + list_req.status = dsn::app_status::AS_AVAILABLE; + _meta_svc->get_server_state()->list_apps(list_req, list_resp); + ERR_LOG_AND_RETURN_NOT_OK(list_resp.err, "list_apps failed."); + for (const auto &app : list_resp.infos) { + std::string database_name = get_database_name_from_app_name(app.app_name); + // Use "*" for table name of invalid Ranger rules to match datdabase resources. + if (database_name.empty()) { + database_name = "*"; + } + std::string table_name = get_table_name_from_app_name(app.app_name); + + auto req = dsn::make_unique(); + req->__set_app_name(app.app_name); + req->__set_keys( + {dsn::replication::replica_envs::REPLICA_ACCESS_CONTROLLER_RANGER_POLICIES}); + bool is_policy_matched = false; + for (const auto &policy : table_policies->second) { + if (policy.database_names.count(database_name) == 0) { + continue; + } + + // if table name does not conform to the naming rules(database_name.table_name), + // database is defined by "*" in ranger for acl matching + if (policy.table_names.count("*") != 0 || policy.table_names.count(table_name) != 0) { + is_policy_matched = true; + req->__set_op(dsn::replication::app_env_operation::type::APP_ENV_OP_SET); + req->__set_values( + {json::json_forwarder::encode(policy.policies).to_string()}); + + dsn::replication::update_app_env_rpc rpc(std::move(req), + LPC_USE_RANGER_ACCESS_CONTROL); + _meta_svc->get_server_state()->set_app_envs(rpc); + ERR_LOG_AND_RETURN_NOT_OK(rpc.response().err, "set_app_envs failed."); + break; + } + } + + // There is no matched policy, clear app Ranger policy + if (!is_policy_matched) { + req->__set_op(dsn::replication::app_env_operation::type::APP_ENV_OP_DEL); + + dsn::replication::update_app_env_rpc rpc(std::move(req), LPC_USE_RANGER_ACCESS_CONTROL); + _meta_svc->get_server_state()->del_app_envs(rpc); + ERR_LOG_AND_RETURN_NOT_OK(rpc.response().err, "del_app_envs failed."); + } + } + + ddebug("Sync policies to app envs succeeded."); + return dsn::ERR_OK; +} + +std::string get_database_name_from_app_name(const std::string &app_name) +{ + std::string prefix = utils::find_string_prefix(app_name, '.'); + if (prefix.empty() || prefix == app_name) { + return std::string(); + } + + return prefix; +} + +std::string get_table_name_from_app_name(const std::string &app_name) +{ + std::string database_name = get_database_name_from_app_name(app_name); + return database_name.empty() ? app_name : app_name.substr(database_name.size() + 1); +} } // namespace ranger } // namespace dsn diff --git a/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.h b/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.h index 1781c3845e..6140834dcd 100644 --- a/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.h +++ b/src/rdsn/src/runtime/ranger/ranger_resource_policy_manager.h @@ -24,7 +24,12 @@ #include "meta/meta_service.h" #include "ranger_resource_policy.h" #include +#include "rapidjson/document.h" +#include +#include +#include #include +#include namespace dsn { @@ -69,12 +74,48 @@ class ranger_resource_policy_manager static void parse_policies_from_json(const rapidjson::Value &data, std::vector &policies); + // Update policies from Ranger service. + dsn::error_code update_policies_from_ranger_service(); + + // Pull policies in JSON format from Ranger service. + dsn::error_code pull_policies_from_ranger_service(std::string *ranger_policies) const; + + // Load policies from JSON formated string. + dsn::error_code load_policies_from_json(const std::string &data); + + // Create the path to save policies in remote storage, and update using resources policies. + void start_to_dump_and_sync_policies(); + + // Sync policies in use from Ranger service. + void dump_and_sync_policies(); + + // Dump policies to remote storage. + void dump_policies_to_remote_storage(); + + // Update the cached global/database resources policies. + void update_cached_policies(); + + // Sync policies to app_envs(REPLICA_ACCESS_CONTROLLER_RANGER_POLICIES). + dsn::error_code sync_policies_to_app_envs(); + private: + dsn::task_tracker _tracker; + // The path where policies to be saved in remote storage. std::string _ranger_policy_meta_root; replication::meta_service *_meta_svc; + // The cache of the global resources policies, it's a subset of '_all_resource_policies'. + utils::rw_lock_nr _global_policies_lock; // [ + resource_policies _global_policies_cache; + // ] + + // The cache of the database resources policies, it's a subset of '_all_resource_policies'. + utils::rw_lock_nr _database_policies_lock; // [ + resource_policies _database_policies_cache; + // ] + // The access type of RPCs which access global level resources. access_type_of_rpc_code _ac_type_of_global_rpcs; @@ -82,7 +123,7 @@ class ranger_resource_policy_manager access_type_of_rpc_code _ac_type_of_database_rpcs; // The Ranger policy version to determine whether to update. - // int _local_policy_version; + int _local_policy_version; // All Ranger ACL policies. all_resource_policies _all_resource_policies; @@ -91,5 +132,15 @@ class ranger_resource_policy_manager FRIEND_TEST(ranger_resource_policy_manager_test, parse_policies_from_json_for_test); }; + +// Try to get the database name of 'app_name'. +// When using Ranger for ACL, the constraint table naming rule is +// "{database_name}.{table_name}", use "." to split database name and table name. +// Return an empty string if 'app_name' is not a valid Ranger rule table name. +std::string get_database_name_from_app_name(const std::string &app_name); + +// Try to get the table_name of 'app_name'. +// Return 'app_name' if 'app_name' is not a valid Ranger rule table name. +std::string get_table_name_from_app_name(const std::string &app_name); } // namespace ranger } // namespace dsn diff --git a/src/rdsn/src/runtime/test/ranger_resource_policy_manager_test.cpp b/src/rdsn/src/runtime/test/ranger_resource_policy_manager_test.cpp index 547c6be107..c98eb70cc4 100644 --- a/src/rdsn/src/runtime/test/ranger_resource_policy_manager_test.cpp +++ b/src/rdsn/src/runtime/test/ranger_resource_policy_manager_test.cpp @@ -202,5 +202,46 @@ TEST(ranger_resource_policy_manager_test, ranger_resource_policy_serialized_test EXPECT_EQ(test.expected_result, actual_result); } } + +TEST(ranger_resource_policy_manager_test, get_database_name_from_app_name_test) +{ + struct test_case + { + std::string app_name; + std::string expected_result; + } tests[] = {{"", ""}, + {".", ""}, + {"...", ""}, + {"database_name.", "database_name"}, + {".table_name", ""}, + {"app_name", ""}, + {"database_name.table_name", "database_name"}, + {"a.b.c", "a"}}; + for (const auto &test : tests) { + auto actual_result = get_database_name_from_app_name(test.app_name); + EXPECT_EQ(test.expected_result, actual_result); + } +} + +TEST(ranger_resource_policy_manager_test, get_table_name_from_app_name_test) +{ + struct test_case + { + std::string app_name; + std::string expected_result; + } tests[] = {{"", ""}, + {".", "."}, + {"...", "..."}, + {"database_name.", ""}, + {".table_name", ".table_name"}, + {"app_name", "app_name"}, + {"database_name.table_name", "table_name"}, + {"a.b.c", "b.c"}}; + for (const auto &test : tests) { + auto actual_result = get_table_name_from_app_name(test.app_name); + EXPECT_EQ(test.expected_result, actual_result); + } +} + } // namespace ranger } // namespace dsn