diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-17 06:53:20 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-17 06:53:20 +0000 |
commit | e5a812082ae033afb1eed82c0f2df3d0f6bdc93f (patch) | |
tree | a6716c9275b4b413f6c9194798b34b91affb3cc7 /lib/pengine/complex.c | |
parent | Initial commit. (diff) | |
download | pacemaker-e5a812082ae033afb1eed82c0f2df3d0f6bdc93f.tar.xz pacemaker-e5a812082ae033afb1eed82c0f2df3d0f6bdc93f.zip |
Adding upstream version 2.1.6.upstream/2.1.6
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'lib/pengine/complex.c')
-rw-r--r-- | lib/pengine/complex.c | 1174 |
1 files changed, 1174 insertions, 0 deletions
diff --git a/lib/pengine/complex.c b/lib/pengine/complex.c new file mode 100644 index 0000000..f168124 --- /dev/null +++ b/lib/pengine/complex.c @@ -0,0 +1,1174 @@ +/* + * Copyright 2004-2023 the Pacemaker project contributors + * + * The version control history for this file may have further details. + * + * This source code is licensed under the GNU Lesser General Public License + * version 2.1 or later (LGPLv2.1+) WITHOUT ANY WARRANTY. + */ + +#include <crm_internal.h> + +#include <crm/pengine/rules.h> +#include <crm/pengine/internal.h> +#include <crm/msg_xml.h> +#include <crm/common/xml_internal.h> + +#include "pe_status_private.h" + +void populate_hash(xmlNode * nvpair_list, GHashTable * hash, const char **attrs, int attrs_length); + +static pe_node_t *active_node(const pe_resource_t *rsc, unsigned int *count_all, + unsigned int *count_clean); + +resource_object_functions_t resource_class_functions[] = { + { + native_unpack, + native_find_rsc, + native_parameter, + native_print, + native_active, + native_resource_state, + native_location, + native_free, + pe__count_common, + pe__native_is_filtered, + active_node, + }, + { + group_unpack, + native_find_rsc, + native_parameter, + group_print, + group_active, + group_resource_state, + native_location, + group_free, + pe__count_common, + pe__group_is_filtered, + active_node, + }, + { + clone_unpack, + native_find_rsc, + native_parameter, + clone_print, + clone_active, + clone_resource_state, + native_location, + clone_free, + pe__count_common, + pe__clone_is_filtered, + active_node, + }, + { + pe__unpack_bundle, + native_find_rsc, + native_parameter, + pe__print_bundle, + pe__bundle_active, + pe__bundle_resource_state, + native_location, + pe__free_bundle, + pe__count_bundle, + pe__bundle_is_filtered, + pe__bundle_active_node, + } +}; + +static enum pe_obj_types +get_resource_type(const char *name) +{ + if (pcmk__str_eq(name, XML_CIB_TAG_RESOURCE, pcmk__str_casei)) { + return pe_native; + + } else if (pcmk__str_eq(name, XML_CIB_TAG_GROUP, pcmk__str_casei)) { + return pe_group; + + } else if (pcmk__str_eq(name, XML_CIB_TAG_INCARNATION, pcmk__str_casei)) { + return pe_clone; + + } else if (pcmk__str_eq(name, PCMK_XE_PROMOTABLE_LEGACY, pcmk__str_casei)) { + // @COMPAT deprecated since 2.0.0 + return pe_clone; + + } else if (pcmk__str_eq(name, XML_CIB_TAG_CONTAINER, pcmk__str_casei)) { + return pe_container; + } + + return pe_unknown; +} + +static void +dup_attr(gpointer key, gpointer value, gpointer user_data) +{ + add_hash_param(user_data, key, value); +} + +static void +expand_parents_fixed_nvpairs(pe_resource_t * rsc, pe_rule_eval_data_t * rule_data, GHashTable * meta_hash, pe_working_set_t * data_set) +{ + GHashTable *parent_orig_meta = pcmk__strkey_table(free, free); + pe_resource_t *p = rsc->parent; + + if (p == NULL) { + return ; + } + + /* Search all parent resources, get the fixed value of "meta_attributes" set only in the original xml, and stack it in the hash table. */ + /* The fixed value of the lower parent resource takes precedence and is not overwritten. */ + while(p != NULL) { + /* A hash table for comparison is generated, including the id-ref. */ + pe__unpack_dataset_nvpairs(p->xml, XML_TAG_META_SETS, + rule_data, parent_orig_meta, NULL, FALSE, data_set); + p = p->parent; + } + + /* If there is a fixed value of "meta_attributes" of the parent resource, it will be processed. */ + if (parent_orig_meta != NULL) { + GHashTableIter iter; + char *key = NULL; + char *value = NULL; + + g_hash_table_iter_init(&iter, parent_orig_meta); + while (g_hash_table_iter_next(&iter, (gpointer *) &key, (gpointer *) &value)) { + /* Parameters set in the original xml of the parent resource will also try to overwrite the child resource. */ + /* Attributes that already exist in the child lease are not updated. */ + dup_attr(key, value, meta_hash); + } + } + + if (parent_orig_meta != NULL) { + g_hash_table_destroy(parent_orig_meta); + } + + return ; + +} +void +get_meta_attributes(GHashTable * meta_hash, pe_resource_t * rsc, + pe_node_t * node, pe_working_set_t * data_set) +{ + pe_rsc_eval_data_t rsc_rule_data = { + .standard = crm_element_value(rsc->xml, XML_AGENT_ATTR_CLASS), + .provider = crm_element_value(rsc->xml, XML_AGENT_ATTR_PROVIDER), + .agent = crm_element_value(rsc->xml, XML_EXPR_ATTR_TYPE) + }; + + pe_rule_eval_data_t rule_data = { + .node_hash = NULL, + .role = RSC_ROLE_UNKNOWN, + .now = data_set->now, + .match_data = NULL, + .rsc_data = &rsc_rule_data, + .op_data = NULL + }; + + if (node) { + rule_data.node_hash = node->details->attrs; + } + + for (xmlAttrPtr a = pcmk__xe_first_attr(rsc->xml); a != NULL; a = a->next) { + const char *prop_name = (const char *) a->name; + const char *prop_value = crm_element_value(rsc->xml, prop_name); + + add_hash_param(meta_hash, prop_name, prop_value); + } + + pe__unpack_dataset_nvpairs(rsc->xml, XML_TAG_META_SETS, &rule_data, + meta_hash, NULL, FALSE, data_set); + + /* Set the "meta_attributes" explicitly set in the parent resource to the hash table of the child resource. */ + /* If it is already explicitly set as a child, it will not be overwritten. */ + if (rsc->parent != NULL) { + expand_parents_fixed_nvpairs(rsc, &rule_data, meta_hash, data_set); + } + + /* check the defaults */ + pe__unpack_dataset_nvpairs(data_set->rsc_defaults, XML_TAG_META_SETS, + &rule_data, meta_hash, NULL, FALSE, data_set); + + /* If there is "meta_attributes" that the parent resource has not explicitly set, set a value that is not set from rsc_default either. */ + /* The values already set up to this point will not be overwritten. */ + if (rsc->parent) { + g_hash_table_foreach(rsc->parent->meta, dup_attr, meta_hash); + } +} + +void +get_rsc_attributes(GHashTable *meta_hash, const pe_resource_t *rsc, + const pe_node_t *node, pe_working_set_t *data_set) +{ + pe_rule_eval_data_t rule_data = { + .node_hash = NULL, + .role = RSC_ROLE_UNKNOWN, + .now = data_set->now, + .match_data = NULL, + .rsc_data = NULL, + .op_data = NULL + }; + + if (node) { + rule_data.node_hash = node->details->attrs; + } + + pe__unpack_dataset_nvpairs(rsc->xml, XML_TAG_ATTR_SETS, &rule_data, + meta_hash, NULL, FALSE, data_set); + + /* set anything else based on the parent */ + if (rsc->parent != NULL) { + get_rsc_attributes(meta_hash, rsc->parent, node, data_set); + + } else { + /* and finally check the defaults */ + pe__unpack_dataset_nvpairs(data_set->rsc_defaults, XML_TAG_ATTR_SETS, + &rule_data, meta_hash, NULL, FALSE, data_set); + } +} + +static char * +template_op_key(xmlNode * op) +{ + const char *name = crm_element_value(op, "name"); + const char *role = crm_element_value(op, "role"); + char *key = NULL; + + if ((role == NULL) + || pcmk__strcase_any_of(role, RSC_ROLE_STARTED_S, RSC_ROLE_UNPROMOTED_S, + RSC_ROLE_UNPROMOTED_LEGACY_S, NULL)) { + role = RSC_ROLE_UNKNOWN_S; + } + + key = crm_strdup_printf("%s-%s", name, role); + return key; +} + +static gboolean +unpack_template(xmlNode * xml_obj, xmlNode ** expanded_xml, pe_working_set_t * data_set) +{ + xmlNode *cib_resources = NULL; + xmlNode *template = NULL; + xmlNode *new_xml = NULL; + xmlNode *child_xml = NULL; + xmlNode *rsc_ops = NULL; + xmlNode *template_ops = NULL; + const char *template_ref = NULL; + const char *clone = NULL; + const char *id = NULL; + + if (xml_obj == NULL) { + pe_err("No resource object for template unpacking"); + return FALSE; + } + + template_ref = crm_element_value(xml_obj, XML_CIB_TAG_RSC_TEMPLATE); + if (template_ref == NULL) { + return TRUE; + } + + id = ID(xml_obj); + if (id == NULL) { + pe_err("'%s' object must have a id", crm_element_name(xml_obj)); + return FALSE; + } + + if (pcmk__str_eq(template_ref, id, pcmk__str_none)) { + pe_err("The resource object '%s' should not reference itself", id); + return FALSE; + } + + cib_resources = get_xpath_object("//"XML_CIB_TAG_RESOURCES, data_set->input, LOG_TRACE); + if (cib_resources == NULL) { + pe_err("No resources configured"); + return FALSE; + } + + template = pcmk__xe_match(cib_resources, XML_CIB_TAG_RSC_TEMPLATE, + XML_ATTR_ID, template_ref); + if (template == NULL) { + pe_err("No template named '%s'", template_ref); + return FALSE; + } + + new_xml = copy_xml(template); + xmlNodeSetName(new_xml, xml_obj->name); + crm_xml_replace(new_xml, XML_ATTR_ID, id); + + clone = crm_element_value(xml_obj, XML_RSC_ATTR_INCARNATION); + if(clone) { + crm_xml_add(new_xml, XML_RSC_ATTR_INCARNATION, clone); + } + + template_ops = find_xml_node(new_xml, "operations", FALSE); + + for (child_xml = pcmk__xe_first_child(xml_obj); child_xml != NULL; + child_xml = pcmk__xe_next(child_xml)) { + xmlNode *new_child = NULL; + + new_child = add_node_copy(new_xml, child_xml); + + if (pcmk__str_eq((const char *)new_child->name, "operations", pcmk__str_none)) { + rsc_ops = new_child; + } + } + + if (template_ops && rsc_ops) { + xmlNode *op = NULL; + GHashTable *rsc_ops_hash = pcmk__strkey_table(free, NULL); + + for (op = pcmk__xe_first_child(rsc_ops); op != NULL; + op = pcmk__xe_next(op)) { + + char *key = template_op_key(op); + + g_hash_table_insert(rsc_ops_hash, key, op); + } + + for (op = pcmk__xe_first_child(template_ops); op != NULL; + op = pcmk__xe_next(op)) { + + char *key = template_op_key(op); + + if (g_hash_table_lookup(rsc_ops_hash, key) == NULL) { + add_node_copy(rsc_ops, op); + } + + free(key); + } + + if (rsc_ops_hash) { + g_hash_table_destroy(rsc_ops_hash); + } + + free_xml(template_ops); + } + + /*free_xml(*expanded_xml); */ + *expanded_xml = new_xml; + + /* Disable multi-level templates for now */ + /*if(unpack_template(new_xml, expanded_xml, data_set) == FALSE) { + free_xml(*expanded_xml); + *expanded_xml = NULL; + + return FALSE; + } */ + + return TRUE; +} + +static gboolean +add_template_rsc(xmlNode * xml_obj, pe_working_set_t * data_set) +{ + const char *template_ref = NULL; + const char *id = NULL; + + if (xml_obj == NULL) { + pe_err("No resource object for processing resource list of template"); + return FALSE; + } + + template_ref = crm_element_value(xml_obj, XML_CIB_TAG_RSC_TEMPLATE); + if (template_ref == NULL) { + return TRUE; + } + + id = ID(xml_obj); + if (id == NULL) { + pe_err("'%s' object must have a id", crm_element_name(xml_obj)); + return FALSE; + } + + if (pcmk__str_eq(template_ref, id, pcmk__str_none)) { + pe_err("The resource object '%s' should not reference itself", id); + return FALSE; + } + + if (add_tag_ref(data_set->template_rsc_sets, template_ref, id) == FALSE) { + return FALSE; + } + + return TRUE; +} + +static bool +detect_promotable(pe_resource_t *rsc) +{ + const char *promotable = g_hash_table_lookup(rsc->meta, + XML_RSC_ATTR_PROMOTABLE); + + if (crm_is_true(promotable)) { + return TRUE; + } + + // @COMPAT deprecated since 2.0.0 + if (pcmk__str_eq(crm_element_name(rsc->xml), PCMK_XE_PROMOTABLE_LEGACY, + pcmk__str_casei)) { + /* @TODO in some future version, pe_warn_once() here, + * then drop support in even later version + */ + g_hash_table_insert(rsc->meta, strdup(XML_RSC_ATTR_PROMOTABLE), + strdup(XML_BOOLEAN_TRUE)); + return TRUE; + } + return FALSE; +} + +static void +free_params_table(gpointer data) +{ + g_hash_table_destroy((GHashTable *) data); +} + +/*! + * \brief Get a table of resource parameters + * + * \param[in,out] rsc Resource to query + * \param[in] node Node for evaluating rules (NULL for defaults) + * \param[in,out] data_set Cluster working set + * + * \return Hash table containing resource parameter names and values + * (or NULL if \p rsc or \p data_set is NULL) + * \note The returned table will be destroyed when the resource is freed, so + * callers should not destroy it. + */ +GHashTable * +pe_rsc_params(pe_resource_t *rsc, const pe_node_t *node, + pe_working_set_t *data_set) +{ + GHashTable *params_on_node = NULL; + + /* A NULL node is used to request the resource's default parameters + * (not evaluated for node), but we always want something non-NULL + * as a hash table key. + */ + const char *node_name = ""; + + // Sanity check + if ((rsc == NULL) || (data_set == NULL)) { + return NULL; + } + if ((node != NULL) && (node->details->uname != NULL)) { + node_name = node->details->uname; + } + + // Find the parameter table for given node + if (rsc->parameter_cache == NULL) { + rsc->parameter_cache = pcmk__strikey_table(free, free_params_table); + } else { + params_on_node = g_hash_table_lookup(rsc->parameter_cache, node_name); + } + + // If none exists yet, create one with parameters evaluated for node + if (params_on_node == NULL) { + params_on_node = pcmk__strkey_table(free, free); + get_rsc_attributes(params_on_node, rsc, node, data_set); + g_hash_table_insert(rsc->parameter_cache, strdup(node_name), + params_on_node); + } + return params_on_node; +} + +/*! + * \internal + * \brief Unpack a resource's "requires" meta-attribute + * + * \param[in,out] rsc Resource being unpacked + * \param[in] value Value of "requires" meta-attribute + * \param[in] is_default Whether \p value was selected by default + */ +static void +unpack_requires(pe_resource_t *rsc, const char *value, bool is_default) +{ + if (pcmk__str_eq(value, PCMK__VALUE_NOTHING, pcmk__str_casei)) { + + } else if (pcmk__str_eq(value, PCMK__VALUE_QUORUM, pcmk__str_casei)) { + pe__set_resource_flags(rsc, pe_rsc_needs_quorum); + + } else if (pcmk__str_eq(value, PCMK__VALUE_FENCING, pcmk__str_casei)) { + pe__set_resource_flags(rsc, pe_rsc_needs_fencing); + if (!pcmk_is_set(rsc->cluster->flags, pe_flag_stonith_enabled)) { + pcmk__config_warn("%s requires fencing but fencing is disabled", + rsc->id); + } + + } else if (pcmk__str_eq(value, PCMK__VALUE_UNFENCING, pcmk__str_casei)) { + if (pcmk_is_set(rsc->flags, pe_rsc_fence_device)) { + pcmk__config_warn("Resetting \"" XML_RSC_ATTR_REQUIRES "\" for %s " + "to \"" PCMK__VALUE_QUORUM "\" because fencing " + "devices cannot require unfencing", rsc->id); + unpack_requires(rsc, PCMK__VALUE_QUORUM, true); + return; + + } else if (!pcmk_is_set(rsc->cluster->flags, pe_flag_stonith_enabled)) { + pcmk__config_warn("Resetting \"" XML_RSC_ATTR_REQUIRES "\" for %s " + "to \"" PCMK__VALUE_QUORUM "\" because fencing " + "is disabled", rsc->id); + unpack_requires(rsc, PCMK__VALUE_QUORUM, true); + return; + + } else { + pe__set_resource_flags(rsc, + pe_rsc_needs_fencing|pe_rsc_needs_unfencing); + } + + } else { + const char *orig_value = value; + + if (pcmk_is_set(rsc->flags, pe_rsc_fence_device)) { + value = PCMK__VALUE_QUORUM; + + } else if ((rsc->variant == pe_native) + && xml_contains_remote_node(rsc->xml)) { + value = PCMK__VALUE_QUORUM; + + } else if (pcmk_is_set(rsc->cluster->flags, pe_flag_enable_unfencing)) { + value = PCMK__VALUE_UNFENCING; + + } else if (pcmk_is_set(rsc->cluster->flags, pe_flag_stonith_enabled)) { + value = PCMK__VALUE_FENCING; + + } else if (rsc->cluster->no_quorum_policy == no_quorum_ignore) { + value = PCMK__VALUE_NOTHING; + + } else { + value = PCMK__VALUE_QUORUM; + } + + if (orig_value != NULL) { + pcmk__config_err("Resetting '" XML_RSC_ATTR_REQUIRES "' for %s " + "to '%s' because '%s' is not valid", + rsc->id, value, orig_value); + } + unpack_requires(rsc, value, true); + return; + } + + pe_rsc_trace(rsc, "\tRequired to start: %s%s", value, + (is_default? " (default)" : "")); +} + +#ifndef PCMK__COMPAT_2_0 +static void +warn_about_deprecated_classes(pe_resource_t *rsc) +{ + const char *std = crm_element_value(rsc->xml, XML_AGENT_ATTR_CLASS); + + if (pcmk__str_eq(std, PCMK_RESOURCE_CLASS_UPSTART, pcmk__str_none)) { + pe_warn_once(pe_wo_upstart, + "Support for Upstart resources (such as %s) is deprecated " + "and will be removed in a future release of Pacemaker", + rsc->id); + + } else if (pcmk__str_eq(std, PCMK_RESOURCE_CLASS_NAGIOS, pcmk__str_none)) { + pe_warn_once(pe_wo_nagios, + "Support for Nagios resources (such as %s) is deprecated " + "and will be removed in a future release of Pacemaker", + rsc->id); + } +} +#endif + +/*! + * \internal + * \brief Unpack configuration XML for a given resource + * + * Unpack the XML object containing a resource's configuration into a new + * \c pe_resource_t object. + * + * \param[in] xml_obj XML node containing the resource's configuration + * \param[out] rsc Where to store the unpacked resource information + * \param[in] parent Resource's parent, if any + * \param[in,out] data_set Cluster working set + * + * \return Standard Pacemaker return code + * \note If pcmk_rc_ok is returned, \p *rsc is guaranteed to be non-NULL, and + * the caller is responsible for freeing it using its variant-specific + * free() method. Otherwise, \p *rsc is guaranteed to be NULL. + */ +int +pe__unpack_resource(xmlNode *xml_obj, pe_resource_t **rsc, + pe_resource_t *parent, pe_working_set_t *data_set) +{ + xmlNode *expanded_xml = NULL; + xmlNode *ops = NULL; + const char *value = NULL; + const char *id = NULL; + bool guest_node = false; + bool remote_node = false; + + pe_rule_eval_data_t rule_data = { + .node_hash = NULL, + .role = RSC_ROLE_UNKNOWN, + .now = NULL, + .match_data = NULL, + .rsc_data = NULL, + .op_data = NULL + }; + + CRM_CHECK(rsc != NULL, return EINVAL); + CRM_CHECK((xml_obj != NULL) && (data_set != NULL), + *rsc = NULL; + return EINVAL); + + rule_data.now = data_set->now; + + crm_log_xml_trace(xml_obj, "[raw XML]"); + + id = crm_element_value(xml_obj, XML_ATTR_ID); + if (id == NULL) { + pe_err("Ignoring <%s> configuration without " XML_ATTR_ID, + crm_element_name(xml_obj)); + return pcmk_rc_unpack_error; + } + + if (unpack_template(xml_obj, &expanded_xml, data_set) == FALSE) { + return pcmk_rc_unpack_error; + } + + *rsc = calloc(1, sizeof(pe_resource_t)); + if (*rsc == NULL) { + crm_crit("Unable to allocate memory for resource '%s'", id); + return ENOMEM; + } + (*rsc)->cluster = data_set; + + if (expanded_xml) { + crm_log_xml_trace(expanded_xml, "[expanded XML]"); + (*rsc)->xml = expanded_xml; + (*rsc)->orig_xml = xml_obj; + + } else { + (*rsc)->xml = xml_obj; + (*rsc)->orig_xml = NULL; + } + + /* Do not use xml_obj from here on, use (*rsc)->xml in case templates are involved */ + + (*rsc)->parent = parent; + + ops = find_xml_node((*rsc)->xml, "operations", FALSE); + (*rsc)->ops_xml = expand_idref(ops, data_set->input); + + (*rsc)->variant = get_resource_type(crm_element_name((*rsc)->xml)); + if ((*rsc)->variant == pe_unknown) { + pe_err("Ignoring resource '%s' of unknown type '%s'", + id, crm_element_name((*rsc)->xml)); + common_free(*rsc); + *rsc = NULL; + return pcmk_rc_unpack_error; + } + +#ifndef PCMK__COMPAT_2_0 + warn_about_deprecated_classes(*rsc); +#endif + + (*rsc)->meta = pcmk__strkey_table(free, free); + (*rsc)->allowed_nodes = pcmk__strkey_table(NULL, free); + (*rsc)->known_on = pcmk__strkey_table(NULL, free); + + value = crm_element_value((*rsc)->xml, XML_RSC_ATTR_INCARNATION); + if (value) { + (*rsc)->id = crm_strdup_printf("%s:%s", id, value); + add_hash_param((*rsc)->meta, XML_RSC_ATTR_INCARNATION, value); + + } else { + (*rsc)->id = strdup(id); + } + + (*rsc)->fns = &resource_class_functions[(*rsc)->variant]; + + get_meta_attributes((*rsc)->meta, *rsc, NULL, data_set); + (*rsc)->parameters = pe_rsc_params(*rsc, NULL, data_set); // \deprecated + + (*rsc)->flags = 0; + pe__set_resource_flags(*rsc, pe_rsc_runnable|pe_rsc_provisional); + + if (!pcmk_is_set(data_set->flags, pe_flag_maintenance_mode)) { + pe__set_resource_flags(*rsc, pe_rsc_managed); + } + + (*rsc)->rsc_cons = NULL; + (*rsc)->rsc_tickets = NULL; + (*rsc)->actions = NULL; + (*rsc)->role = RSC_ROLE_STOPPED; + (*rsc)->next_role = RSC_ROLE_UNKNOWN; + + (*rsc)->recovery_type = recovery_stop_start; + (*rsc)->stickiness = 0; + (*rsc)->migration_threshold = INFINITY; + (*rsc)->failure_timeout = 0; + + value = g_hash_table_lookup((*rsc)->meta, XML_CIB_ATTR_PRIORITY); + (*rsc)->priority = char2score(value); + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_CRITICAL); + if ((value == NULL) || crm_is_true(value)) { + pe__set_resource_flags(*rsc, pe_rsc_critical); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_NOTIFY); + if (crm_is_true(value)) { + pe__set_resource_flags(*rsc, pe_rsc_notify); + } + + if (xml_contains_remote_node((*rsc)->xml)) { + (*rsc)->is_remote_node = TRUE; + if (g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_CONTAINER)) { + guest_node = true; + } else { + remote_node = true; + } + } + + value = g_hash_table_lookup((*rsc)->meta, XML_OP_ATTR_ALLOW_MIGRATE); + if (crm_is_true(value)) { + pe__set_resource_flags(*rsc, pe_rsc_allow_migrate); + } else if ((value == NULL) && remote_node) { + /* By default, we want remote nodes to be able + * to float around the cluster without having to stop all the + * resources within the remote-node before moving. Allowing + * migration support enables this feature. If this ever causes + * problems, migration support can be explicitly turned off with + * allow-migrate=false. + */ + pe__set_resource_flags(*rsc, pe_rsc_allow_migrate); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_MANAGED); + if (value != NULL && !pcmk__str_eq("default", value, pcmk__str_casei)) { + if (crm_is_true(value)) { + pe__set_resource_flags(*rsc, pe_rsc_managed); + } else { + pe__clear_resource_flags(*rsc, pe_rsc_managed); + } + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_MAINTENANCE); + if (crm_is_true(value)) { + pe__clear_resource_flags(*rsc, pe_rsc_managed); + pe__set_resource_flags(*rsc, pe_rsc_maintenance); + } + if (pcmk_is_set(data_set->flags, pe_flag_maintenance_mode)) { + pe__clear_resource_flags(*rsc, pe_rsc_managed); + pe__set_resource_flags(*rsc, pe_rsc_maintenance); + } + + if (pe_rsc_is_clone(pe__const_top_resource(*rsc, false))) { + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_UNIQUE); + if (crm_is_true(value)) { + pe__set_resource_flags(*rsc, pe_rsc_unique); + } + if (detect_promotable(*rsc)) { + pe__set_resource_flags(*rsc, pe_rsc_promotable); + } + } else { + pe__set_resource_flags(*rsc, pe_rsc_unique); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_RESTART); + if (pcmk__str_eq(value, "restart", pcmk__str_casei)) { + (*rsc)->restart_type = pe_restart_restart; + pe_rsc_trace((*rsc), "%s dependency restart handling: restart", + (*rsc)->id); + pe_warn_once(pe_wo_restart_type, + "Support for restart-type is deprecated and will be removed in a future release"); + + } else { + (*rsc)->restart_type = pe_restart_ignore; + pe_rsc_trace((*rsc), "%s dependency restart handling: ignore", + (*rsc)->id); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_MULTIPLE); + if (pcmk__str_eq(value, "stop_only", pcmk__str_casei)) { + (*rsc)->recovery_type = recovery_stop_only; + pe_rsc_trace((*rsc), "%s multiple running resource recovery: stop only", + (*rsc)->id); + + } else if (pcmk__str_eq(value, "block", pcmk__str_casei)) { + (*rsc)->recovery_type = recovery_block; + pe_rsc_trace((*rsc), "%s multiple running resource recovery: block", + (*rsc)->id); + + } else if (pcmk__str_eq(value, "stop_unexpected", pcmk__str_casei)) { + (*rsc)->recovery_type = recovery_stop_unexpected; + pe_rsc_trace((*rsc), "%s multiple running resource recovery: " + "stop unexpected instances", + (*rsc)->id); + + } else { // "stop_start" + if (!pcmk__str_eq(value, "stop_start", + pcmk__str_casei|pcmk__str_null_matches)) { + pe_warn("%s is not a valid value for " XML_RSC_ATTR_MULTIPLE + ", using default of \"stop_start\"", value); + } + (*rsc)->recovery_type = recovery_stop_start; + pe_rsc_trace((*rsc), "%s multiple running resource recovery: " + "stop/start", (*rsc)->id); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_STICKINESS); + if (value != NULL && !pcmk__str_eq("default", value, pcmk__str_casei)) { + (*rsc)->stickiness = char2score(value); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_FAIL_STICKINESS); + if (value != NULL && !pcmk__str_eq("default", value, pcmk__str_casei)) { + (*rsc)->migration_threshold = char2score(value); + if ((*rsc)->migration_threshold < 0) { + /* @TODO We use 1 here to preserve previous behavior, but this + * should probably use the default (INFINITY) or 0 (to disable) + * instead. + */ + pe_warn_once(pe_wo_neg_threshold, + XML_RSC_ATTR_FAIL_STICKINESS + " must be non-negative, using 1 instead"); + (*rsc)->migration_threshold = 1; + } + } + + if (pcmk__str_eq(crm_element_value((*rsc)->xml, XML_AGENT_ATTR_CLASS), + PCMK_RESOURCE_CLASS_STONITH, pcmk__str_casei)) { + pe__set_working_set_flags(data_set, pe_flag_have_stonith_resource); + pe__set_resource_flags(*rsc, pe_rsc_fence_device); + } + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_REQUIRES); + unpack_requires(*rsc, value, false); + + value = g_hash_table_lookup((*rsc)->meta, XML_RSC_ATTR_FAIL_TIMEOUT); + if (value != NULL) { + // Stored as seconds + (*rsc)->failure_timeout = (int) (crm_parse_interval_spec(value) / 1000); + } + + if (remote_node) { + GHashTable *params = pe_rsc_params(*rsc, NULL, data_set); + + /* Grabbing the value now means that any rules based on node attributes + * will evaluate to false, so such rules should not be used with + * reconnect_interval. + * + * @TODO Evaluate per node before using + */ + value = g_hash_table_lookup(params, XML_REMOTE_ATTR_RECONNECT_INTERVAL); + if (value) { + /* reconnect delay works by setting failure_timeout and preventing the + * connection from starting until the failure is cleared. */ + (*rsc)->remote_reconnect_ms = crm_parse_interval_spec(value); + /* we want to override any default failure_timeout in use when remote + * reconnect_interval is in use. */ + (*rsc)->failure_timeout = (*rsc)->remote_reconnect_ms / 1000; + } + } + + get_target_role(*rsc, &((*rsc)->next_role)); + pe_rsc_trace((*rsc), "%s desired next state: %s", (*rsc)->id, + (*rsc)->next_role != RSC_ROLE_UNKNOWN ? role2text((*rsc)->next_role) : "default"); + + if ((*rsc)->fns->unpack(*rsc, data_set) == FALSE) { + (*rsc)->fns->free(*rsc); + *rsc = NULL; + return pcmk_rc_unpack_error; + } + + if (pcmk_is_set(data_set->flags, pe_flag_symmetric_cluster)) { + // This tag must stay exactly the same because it is tested elsewhere + resource_location(*rsc, NULL, 0, "symmetric_default", data_set); + } else if (guest_node) { + /* remote resources tied to a container resource must always be allowed + * to opt-in to the cluster. Whether the connection resource is actually + * allowed to be placed on a node is dependent on the container resource */ + resource_location(*rsc, NULL, 0, "remote_connection_default", data_set); + } + + pe_rsc_trace((*rsc), "%s action notification: %s", (*rsc)->id, + pcmk_is_set((*rsc)->flags, pe_rsc_notify)? "required" : "not required"); + + (*rsc)->utilization = pcmk__strkey_table(free, free); + + pe__unpack_dataset_nvpairs((*rsc)->xml, XML_TAG_UTILIZATION, &rule_data, + (*rsc)->utilization, NULL, FALSE, data_set); + + if (expanded_xml) { + if (add_template_rsc(xml_obj, data_set) == FALSE) { + (*rsc)->fns->free(*rsc); + *rsc = NULL; + return pcmk_rc_unpack_error; + } + } + return pcmk_rc_ok; +} + +gboolean +is_parent(pe_resource_t *child, pe_resource_t *rsc) +{ + pe_resource_t *parent = child; + + if (parent == NULL || rsc == NULL) { + return FALSE; + } + while (parent->parent != NULL) { + if (parent->parent == rsc) { + return TRUE; + } + parent = parent->parent; + } + return FALSE; +} + +pe_resource_t * +uber_parent(pe_resource_t * rsc) +{ + pe_resource_t *parent = rsc; + + if (parent == NULL) { + return NULL; + } + while (parent->parent != NULL && parent->parent->variant != pe_container) { + parent = parent->parent; + } + return parent; +} + +/*! + * \internal + * \brief Get the topmost parent of a resource as a const pointer + * + * \param[in] rsc Resource to check + * \param[in] include_bundle If true, go all the way to bundle + * + * \return \p NULL if \p rsc is NULL, \p rsc if \p rsc has no parent, + * the bundle if \p rsc is bundled and \p include_bundle is true, + * otherwise the topmost parent of \p rsc up to a clone + */ +const pe_resource_t * +pe__const_top_resource(const pe_resource_t *rsc, bool include_bundle) +{ + const pe_resource_t *parent = rsc; + + if (parent == NULL) { + return NULL; + } + while (parent->parent != NULL) { + if (!include_bundle && (parent->parent->variant == pe_container)) { + break; + } + parent = parent->parent; + } + return parent; +} + +void +common_free(pe_resource_t * rsc) +{ + if (rsc == NULL) { + return; + } + + pe_rsc_trace(rsc, "Freeing %s %d", rsc->id, rsc->variant); + + g_list_free(rsc->rsc_cons); + g_list_free(rsc->rsc_cons_lhs); + g_list_free(rsc->rsc_tickets); + g_list_free(rsc->dangling_migrations); + + if (rsc->parameter_cache != NULL) { + g_hash_table_destroy(rsc->parameter_cache); + } + if (rsc->meta != NULL) { + g_hash_table_destroy(rsc->meta); + } + if (rsc->utilization != NULL) { + g_hash_table_destroy(rsc->utilization); + } + + if ((rsc->parent == NULL) && pcmk_is_set(rsc->flags, pe_rsc_orphan)) { + free_xml(rsc->xml); + rsc->xml = NULL; + free_xml(rsc->orig_xml); + rsc->orig_xml = NULL; + + /* if rsc->orig_xml, then rsc->xml is an expanded xml from a template */ + } else if (rsc->orig_xml) { + free_xml(rsc->xml); + rsc->xml = NULL; + } + if (rsc->running_on) { + g_list_free(rsc->running_on); + rsc->running_on = NULL; + } + if (rsc->known_on) { + g_hash_table_destroy(rsc->known_on); + rsc->known_on = NULL; + } + if (rsc->actions) { + g_list_free(rsc->actions); + rsc->actions = NULL; + } + if (rsc->allowed_nodes) { + g_hash_table_destroy(rsc->allowed_nodes); + rsc->allowed_nodes = NULL; + } + g_list_free(rsc->fillers); + g_list_free(rsc->rsc_location); + pe_rsc_trace(rsc, "Resource freed"); + free(rsc->id); + free(rsc->clone_name); + free(rsc->allocated_to); + free(rsc->variant_opaque); + free(rsc->pending_task); + free(rsc); +} + +/*! + * \internal + * \brief Count a node and update most preferred to it as appropriate + * + * \param[in] rsc An active resource + * \param[in] node A node that \p rsc is active on + * \param[in,out] active This will be set to \p node if \p node is more + * preferred than the current value + * \param[in,out] count_all If not NULL, this will be incremented + * \param[in,out] count_clean If not NULL, this will be incremented if \p node + * is online and clean + * + * \return true if the count should continue, or false if sufficiently known + */ +bool +pe__count_active_node(const pe_resource_t *rsc, pe_node_t *node, + pe_node_t **active, unsigned int *count_all, + unsigned int *count_clean) +{ + bool keep_looking = false; + bool is_happy = false; + + CRM_CHECK((rsc != NULL) && (node != NULL) && (active != NULL), + return false); + + is_happy = node->details->online && !node->details->unclean; + + if (count_all != NULL) { + ++*count_all; + } + if ((count_clean != NULL) && is_happy) { + ++*count_clean; + } + if ((count_all != NULL) || (count_clean != NULL)) { + keep_looking = true; // We're counting, so go through entire list + } + + if (rsc->partial_migration_source != NULL) { + if (node->details == rsc->partial_migration_source->details) { + *active = node; // This is the migration source + } else { + keep_looking = true; + } + } else if (!pcmk_is_set(rsc->flags, pe_rsc_needs_fencing)) { + if (is_happy && ((*active == NULL) || !(*active)->details->online + || (*active)->details->unclean)) { + *active = node; // This is the first clean node + } else { + keep_looking = true; + } + } + if (*active == NULL) { + *active = node; // This is the first node checked + } + return keep_looking; +} + +// Shared implementation of resource_object_functions_t:active_node() +static pe_node_t * +active_node(const pe_resource_t *rsc, unsigned int *count_all, + unsigned int *count_clean) +{ + pe_node_t *active = NULL; + + if (count_all != NULL) { + *count_all = 0; + } + if (count_clean != NULL) { + *count_clean = 0; + } + if (rsc == NULL) { + return NULL; + } + for (GList *iter = rsc->running_on; iter != NULL; iter = iter->next) { + if (!pe__count_active_node(rsc, (pe_node_t *) iter->data, &active, + count_all, count_clean)) { + break; // Don't waste time iterating if we don't have to + } + } + return active; +} + +/*! + * \brief + * \internal Find and count active nodes according to "requires" + * + * \param[in] rsc Resource to check + * \param[out] count If not NULL, will be set to count of active nodes + * + * \return An active node (or NULL if resource is not active anywhere) + * + * \note This is a convenience wrapper for active_node() where the count of all + * active nodes or only clean active nodes is desired according to the + * "requires" meta-attribute. + */ +pe_node_t * +pe__find_active_requires(const pe_resource_t *rsc, unsigned int *count) +{ + if (rsc == NULL) { + if (count != NULL) { + *count = 0; + } + return NULL; + + } else if (pcmk_is_set(rsc->flags, pe_rsc_needs_fencing)) { + return rsc->fns->active_node(rsc, count, NULL); + + } else { + return rsc->fns->active_node(rsc, NULL, count); + } +} + +void +pe__count_common(pe_resource_t *rsc) +{ + if (rsc->children != NULL) { + for (GList *item = rsc->children; item != NULL; item = item->next) { + ((pe_resource_t *) item->data)->fns->count(item->data); + } + + } else if (!pcmk_is_set(rsc->flags, pe_rsc_orphan) + || (rsc->role > RSC_ROLE_STOPPED)) { + rsc->cluster->ninstances++; + if (pe__resource_is_disabled(rsc)) { + rsc->cluster->disabled_resources++; + } + if (pcmk_is_set(rsc->flags, pe_rsc_block)) { + rsc->cluster->blocked_resources++; + } + } +} + +/*! + * \internal + * \brief Update a resource's next role + * + * \param[in,out] rsc Resource to be updated + * \param[in] role Resource's new next role + * \param[in] why Human-friendly reason why role is changing (for logs) + */ +void +pe__set_next_role(pe_resource_t *rsc, enum rsc_role_e role, const char *why) +{ + CRM_ASSERT((rsc != NULL) && (why != NULL)); + if (rsc->next_role != role) { + pe_rsc_trace(rsc, "Resetting next role for %s from %s to %s (%s)", + rsc->id, role2text(rsc->next_role), role2text(role), why); + rsc->next_role = role; + } +} |