/* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ /** * Description of config variables, including their min, max, default * values can be printed (--configinfo). This can also be printed * in xml format (--xml). * * Config can be retrieved from only one of the following sources: ** 1) config stored at mgmd (default) ** 2) config stored at a data node (--config_from_node=) *** (Note: *** Node numbers less than 1 give error: *** "Given value is not a valid node number." *** Non-data node numbers give error: *** "Node is not a data node.") ** 3) my.cnf (--mycnf=) ** 4) config.file (--config_file= * * Config variables are displayed from only one of the following * sections of the retrieved config: ** CFG_SECTION_NODE (default, or --nodes) ** CFG_SECTION_CONNECTION (--connections) ** CFG_SECTION_SYSTEM (--system) */ /** * Examples: * Get config from mgmd (default): ** Display results from section CFG_SECTION_NODE (default) *** ndb_config --nodes --query=nodeid --type=ndbd --host=local1 *** ndb_config --query=nodeid,host * ** Display results from section CFG_SECTION_SYSTEM *** ndb_config --system --query=ConfigGenerationNumber * ** Display results from section CFG_SECTION_CONNECTION *** ndb_config --connections --query=type * * Get config from eg. node 2, which is a data node: * ** ndb_config --config_from_node=2 --system --query=ConfigGenerationNumber ** ndb_config --config_from_node=2 --connections --query=type ** ndb_config --config_from_node=2 --query=id,NoOfFragmentLogFiles * ** Get config from eg. node 2 and display results for node 2 only: *** ndb_config --config_from_node=2 --query=id,NoOfFragmentLogFiles --nodeid=2 */ #include #include #include #include #include "../src/mgmapi/mgmapi_configuration.hpp" #include "../src/mgmsrv/ConfigInfo.hpp" #include #include static int g_verbose = 0; static int try_reconnect = 3; static int g_nodes, g_connections, g_system, g_section; static const char * g_query = 0; static int g_nodeid = 0; static const char * g_type = 0; static const char * g_host = 0; static const char * g_field_delimiter=","; static const char * g_row_delimiter=" "; static const char * g_config_file = 0; static int g_mycnf = 0; static int g_configinfo = 0; static int g_xml = 0; static int g_config_from_node = 0; const char *load_default_groups[]= { "mysql_cluster",0 }; typedef ndb_mgm_configuration_iterator Iter; static struct my_option my_long_options[] = { NDB_STD_OPTS("ndb_config"), { "nodes", NDB_OPT_NOSHORT, "Print nodes", (uchar**) &g_nodes, (uchar**) &g_nodes, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "connections", NDB_OPT_NOSHORT, "Print connections", (uchar**) &g_connections, (uchar**) &g_connections, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "system", NDB_OPT_NOSHORT, "Print system", (uchar**) &g_system, (uchar**) &g_system, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "query", 'q', "Query option(s)", (uchar**) &g_query, (uchar**) &g_query, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "host", NDB_OPT_NOSHORT, "Host", (uchar**) &g_host, (uchar**) &g_host, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "type", NDB_OPT_NOSHORT, "Type of node/connection", (uchar**) &g_type, (uchar**) &g_type, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "id", NDB_OPT_NOSHORT, "Nodeid", (uchar**) &g_nodeid, (uchar**) &g_nodeid, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "nodeid", NDB_OPT_NOSHORT, "Nodeid", (uchar**) &g_nodeid, (uchar**) &g_nodeid, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "fields", 'f', "Field separator", (uchar**) &g_field_delimiter, (uchar**) &g_field_delimiter, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "rows", 'r', "Row separator", (uchar**) &g_row_delimiter, (uchar**) &g_row_delimiter, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "config-file", NDB_OPT_NOSHORT, "Path to config.ini", (uchar**) &g_config_file, (uchar**) &g_config_file, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, { "mycnf", NDB_OPT_NOSHORT, "Read config from my.cnf", (uchar**) &g_mycnf, (uchar**) &g_mycnf, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "configinfo", NDB_OPT_NOSHORT, "Print configinfo", (uchar**) &g_configinfo, (uchar**) &g_configinfo, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "xml", NDB_OPT_NOSHORT, "Print configinfo in xml format", (uchar**) &g_xml, (uchar**) &g_xml, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, { "config_from_node", NDB_OPT_NOSHORT, "Use current config from node with given nodeid", (uchar**) &g_config_from_node, (uchar**) &g_config_from_node, 0, GET_INT, REQUIRED_ARG, INT_MIN, INT_MIN, 0, 0, 0, 0}, { 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; static void short_usage_sub(void) { ndb_short_usage_sub(NULL); } static void usage() { char desc[] = "This program will retreive config options for a ndb cluster\n"; puts(desc); ndb_usage(short_usage_sub, load_default_groups, my_long_options); } /** * Match/Apply framework */ struct Match { int m_key; BaseString m_value; Match() {} virtual int eval(const Iter&); virtual ~Match() {} }; struct HostMatch : public Match { HostMatch() {} virtual int eval(const Iter&); }; struct Apply { Apply() {} Apply(int val) { m_key = val;} int m_key; virtual int apply(const Iter&); virtual ~Apply() {} }; struct NodeTypeApply : public Apply { NodeTypeApply() {} virtual int apply(const Iter&); }; struct ConnectionTypeApply : public Apply { ConnectionTypeApply() {} virtual int apply(const Iter&); }; static int parse_query(Vector&, int &argc, char**& argv); static int parse_where(Vector&, int &argc, char**& argv); static int eval(const Iter&, const Vector&); static int apply(const Iter&, const Vector&); static ndb_mgm_configuration* fetch_configuration(int from_node); static ndb_mgm_configuration* load_configuration(); int main(int argc, char** argv){ NDB_INIT(argv[0]); ndb_opt_set_usage_funcs(short_usage_sub, usage); ndb_load_defaults(NULL,load_default_groups,&argc,&argv); int ho_error; if ((ho_error=handle_options(&argc, &argv, my_long_options, ndb_std_get_one_option))) exit(255); if (g_configinfo) { ConfigInfo info; if (g_xml) info.print_xml(); else info.print(); exit(0); } if ((g_nodes && g_connections) || (g_system && (g_nodes || g_connections))) { fprintf(stderr, "Error: Only one of the section-options: --nodes, --connections, --system is allowed.\n"); exit(255); } /* There is no explicit option for the user to set * 'retrieving config from mgmd', but this is the default. * Therefore will not contradict with other sources. */ if ((g_config_file && g_mycnf) || ((g_config_from_node != INT_MIN) && (g_config_file || g_mycnf))) { fprintf(stderr, "Error: Config should be retrieved from only one of the following sources:\n"); fprintf(stderr, "\tconfig stored at mgmd (default),\n"); fprintf(stderr, "\tconfig stored at a data node (--config_from_node=), \n"); fprintf(stderr, "\tmy.cnf(--mycnf=),\n"); fprintf(stderr, "\tconfig.file (--config_file=).\n"); exit(255); } g_section = CFG_SECTION_NODE; //default if (g_connections) g_section = CFG_SECTION_CONNECTION; else if (g_system) g_section = CFG_SECTION_SYSTEM; ndb_mgm_configuration * conf = 0; if (g_config_file || g_mycnf) conf = load_configuration(); else conf = fetch_configuration(g_config_from_node); if (conf == 0) { exit(255); } Vector select_list; Vector where_clause; if(strcmp(g_row_delimiter, "\\n") == 0) g_row_delimiter = "\n"; if(strcmp(g_field_delimiter, "\\n") == 0) g_field_delimiter = "\n"; if(parse_query(select_list, argc, argv)) { exit(0); } if(parse_where(where_clause, argc, argv)) { exit(0); } Iter iter(* conf, g_section); bool prev= false; iter.first(); for(iter.first(); iter.valid(); iter.next()) { if(eval(iter, where_clause)) { if(prev) printf("%s", g_row_delimiter); prev= true; apply(iter, select_list); } } printf("\n"); return 0; } static int parse_query(Vector& select, int &argc, char**& argv) { if(g_query) { BaseString q(g_query); Vector list; q.split(list, ","); for(unsigned i = 0; i& where, int &argc, char**& argv) { Match m; if(g_host) { HostMatch *tmp = new HostMatch; tmp->m_key = CFG_NODE_HOST; tmp->m_value.assfmt("%s", g_host); where.push_back(tmp); } if(g_type) { m.m_key = CFG_TYPE_OF_SECTION; m.m_value.assfmt("%d", ndb_mgm_match_node_type(g_type)); where.push_back(new Match(m)); } if(g_nodeid) { m.m_key = CFG_NODE_ID; m.m_value.assfmt("%d", g_nodeid); where.push_back(new Match(m)); } return 0; } template class Vector; template class Vector; static int eval(const Iter& iter, const Vector& where) { for(unsigned i = 0; ieval(iter) == 0) return 0; } return 1; } static int apply(const Iter& iter, const Vector& list) { for(unsigned i = 0; iapply(iter); if(i + 1 != list.size()) printf("%s", g_field_delimiter); } return 0; } int Match::eval(const Iter& iter) { Uint32 val32; Uint64 val64; const char* valc; if (iter.get(m_key, &val32) == 0) { if(atoi(m_value.c_str()) != (int)val32) return 0; } else if(iter.get(m_key, &val64) == 0) { if(my_strtoll(m_value.c_str(), (char **)NULL, 10) != (long long)val64) return 0; } else if(iter.get(m_key, &valc) == 0) { if(strcmp(m_value.c_str(), valc) != 0) return 0; } else { return 0; } return 1; } int HostMatch::eval(const Iter& iter) { const char* valc; if(iter.get(m_key, &valc) == 0) { struct hostent *h1, *h2, copy1; char *addr1; h1 = gethostbyname(m_value.c_str()); if (h1 == NULL) { return 0; } // gethostbyname returns a pointer to a static structure // so we need to copy the results before doing the next call memcpy(©1, h1, sizeof(struct hostent)); addr1 = (char *)malloc(copy1.h_length); NdbAutoPtr tmp_aptr(addr1); memcpy(addr1, h1->h_addr, copy1.h_length); h2 = gethostbyname(valc); if (h2 == NULL) { return 0; } if (copy1.h_addrtype != h2->h_addrtype) { return 0; } if (copy1.h_length != h2->h_length) { return 0; } return 0 == memcmp(addr1, h2->h_addr, copy1.h_length); } return 0; } int Apply::apply(const Iter& iter) { Uint32 val32; Uint64 val64; const char* valc; if (iter.get(m_key, &val32) == 0) { printf("%u", val32); } else if(iter.get(m_key, &val64) == 0) { printf("%llu", val64); } else if(iter.get(m_key, &valc) == 0) { printf("%s", valc); } return 0; } int NodeTypeApply::apply(const Iter& iter) { Uint32 val32; if (iter.get(CFG_TYPE_OF_SECTION, &val32) == 0) { printf("%s", ndb_mgm_get_node_type_alias_string((ndb_mgm_node_type)val32, 0)); } return 0; } int ConnectionTypeApply::apply(const Iter& iter) { Uint32 val32; if (iter.get(CFG_TYPE_OF_SECTION, &val32) == 0) { switch (val32) { case CONNECTION_TYPE_TCP: printf("tcp"); break; case CONNECTION_TYPE_SCI: printf("sci"); break; case CONNECTION_TYPE_SHM: printf("shm"); break; default: printf(""); break; } } return 0; } static ndb_mgm_configuration* fetch_configuration(int from_node) { ndb_mgm_configuration* conf = 0; NdbMgmHandle mgm = ndb_mgm_create_handle(); if(mgm == NULL) { fprintf(stderr, "Cannot create handle to management server.\n"); return 0; } ndb_mgm_set_error_stream(mgm, stderr); if (ndb_mgm_set_connectstring(mgm, opt_ndb_connectstring)) { fprintf(stderr, "* %5d: %s\n", ndb_mgm_get_latest_error(mgm), ndb_mgm_get_latest_error_msg(mgm)); fprintf(stderr, "* %s", ndb_mgm_get_latest_error_desc(mgm)); goto noconnect; } if(ndb_mgm_connect(mgm, try_reconnect-1, 5, 1)) { fprintf(stderr, "Connect failed"); fprintf(stderr, " code: %d, msg: %s\n", ndb_mgm_get_latest_error(mgm), ndb_mgm_get_latest_error_msg(mgm)); goto noconnect; } else if(g_verbose) { fprintf(stderr, "Connected to %s:%d\n", ndb_mgm_get_connected_host(mgm), ndb_mgm_get_connected_port(mgm)); } if (from_node == INT_MIN) { // from_node option is not requested. // Retrieve config from the default src: mgmd conf = ndb_mgm_get_configuration(mgm, 0); } else if (from_node < 1) { fprintf(stderr, "Invalid node number %d is given for --config_from_node.\n", from_node); goto noconnect; } else { // Retrieve config from the given data node conf = ndb_mgm_get_configuration_from_node(mgm, from_node); } if(conf == 0) { fprintf(stderr, "Could not get configuration, "); fprintf(stderr, "error code: %d, error msg: %s\n", ndb_mgm_get_latest_error(mgm), ndb_mgm_get_latest_error_msg(mgm)); } else if(g_verbose) { fprintf(stderr, "Fetched configuration\n"); } ndb_mgm_disconnect(mgm); noconnect: ndb_mgm_destroy_handle(&mgm); return conf; } #include "../src/mgmsrv/Config.hpp" #include extern EventLogger *g_eventLogger; static ndb_mgm_configuration* load_configuration() { g_eventLogger->removeAllHandlers(); g_eventLogger->createConsoleHandler(ndberr); g_eventLogger->setCategory("ndb_config"); InitConfigFileParser parser; if (g_config_file) { if (g_verbose) fprintf(stderr, "Using config.ini : %s\n", g_config_file); Config* conf = parser.parseConfig(g_config_file); if (conf) return conf->m_configValues; return 0; } if (g_verbose) fprintf(stderr, "Using my.cnf\n"); Config* conf = parser.parse_mycnf(); if (conf) return conf->m_configValues; return 0; }