pacemaker 3.0.1-16e74fc4da
Scalable High-Availability cluster resource manager
Loading...
Searching...
No Matches
group.c
Go to the documentation of this file.
1/*
2 * Copyright 2004-2025 the Pacemaker project contributors
3 *
4 * The version control history for this file may have further details.
5 *
6 * This source code is licensed under the GNU Lesser General Public License
7 * version 2.1 or later (LGPLv2.1+) WITHOUT ANY WARRANTY.
8 */
9
10#include <crm_internal.h>
11
12#include <stdbool.h> // bool, true, false
13#include <stdint.h>
14
15#include <crm/pengine/status.h>
17#include <crm/common/xml.h>
18#include <crm/common/output.h>
21#include <pe_status_private.h>
22
23typedef struct group_variant_data_s {
24 pcmk_resource_t *last_child; // Last group member
25 uint32_t flags; // Group of enum pcmk__group_flags
27
38{
39 if (group != NULL) {
40 const group_variant_data_t *group_data = NULL;
41
42 CRM_CHECK(pcmk__is_group(group), return NULL);
43 group_data = group->priv->variant_opaque;
44 return group_data->last_child;
45 }
46 return NULL;
47}
48
58bool
60{
61 const group_variant_data_t *group_data = NULL;
62
63 CRM_CHECK(pcmk__is_group(group), return false);
64 group_data = group->priv->variant_opaque;
65 return pcmk_all_flags_set(group_data->flags, flags);
66}
67
77static void
78set_group_flag(pcmk_resource_t *group, const char *option, uint32_t flag,
79 uint32_t wo_bit)
80{
81 const char *value_s = NULL;
82 int value = 0;
83
84 value_s = g_hash_table_lookup(group->priv->meta, option);
85
86 // We don't actually need the null check but it speeds up the common case
87 if ((value_s == NULL) || (crm_str_to_boolean(value_s, &value) < 0)
88 || (value != 0)) {
89 group_variant_data_t *group_data = group->priv->variant_opaque;
90
91 group_data->flags |= flag;
92
93 } else {
94 pcmk__warn_once(wo_bit,
95 "Support for the '%s' group meta-attribute is "
96 "deprecated and will be removed in a future release "
97 "(use a resource set instead)", option);
98 }
99}
100
101static int
102inactive_resources(pcmk_resource_t *rsc)
103{
104 int retval = 0;
105
106 for (GList *gIter = rsc->priv->children;
107 gIter != NULL; gIter = gIter->next) {
108
109 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
110
111 if (!child_rsc->priv->fns->active(child_rsc, true)) {
112 retval++;
113 }
114 }
115
116 return retval;
117}
118
119static void
120group_header(pcmk__output_t *out, int *rc, const pcmk_resource_t *rsc,
121 int n_inactive, bool show_inactive, const char *desc)
122{
123 GString *attrs = NULL;
124
125 if (n_inactive > 0 && !show_inactive) {
126 attrs = g_string_sized_new(64);
127 g_string_append_printf(attrs, "%d member%s inactive", n_inactive,
128 pcmk__plural_s(n_inactive));
129 }
130
131 if (pe__resource_is_disabled(rsc)) {
132 pcmk__add_separated_word(&attrs, 64, "disabled", ", ");
133 }
134
136 pcmk__add_separated_word(&attrs, 64, "maintenance", ", ");
137
138 } else if (!pcmk_is_set(rsc->flags, pcmk__rsc_managed)) {
139 pcmk__add_separated_word(&attrs, 64, "unmanaged", ", ");
140 }
141
142 if (attrs != NULL) {
143 PCMK__OUTPUT_LIST_HEADER(out, FALSE, *rc, "Resource Group: %s (%s)%s%s%s",
144 rsc->id,
145 (const char *) attrs->str, desc ? " (" : "",
146 desc ? desc : "", desc ? ")" : "");
147 g_string_free(attrs, TRUE);
148 } else {
149 PCMK__OUTPUT_LIST_HEADER(out, FALSE, *rc, "Resource Group: %s%s%s%s",
150 rsc->id,
151 desc ? " (" : "", desc ? desc : "",
152 desc ? ")" : "");
153 }
154}
155
156static bool
157skip_child_rsc(pcmk_resource_t *rsc, pcmk_resource_t *child,
158 gboolean parent_passes, GList *only_rsc, uint32_t show_opts)
159{
160 bool star_list = pcmk__list_of_1(only_rsc) &&
161 pcmk__str_eq("*", g_list_first(only_rsc)->data, pcmk__str_none);
162 bool child_filtered = child->priv->fns->is_filtered(child, only_rsc, false);
163 bool child_active = child->priv->fns->active(child, false);
164 bool show_inactive = pcmk_is_set(show_opts, pcmk_show_inactive_rscs);
165
166 /* If the resource is in only_rsc by name (so, ignoring "*") then allow
167 * it regardless of if it's active or not.
168 */
169 if (!star_list && !child_filtered) {
170 return false;
171
172 } else if (!child_filtered && (child_active || show_inactive)) {
173 return false;
174
175 } else if (parent_passes && (child_active || show_inactive)) {
176 return false;
177
178 }
179
180 return true;
181}
182
183bool
185{
186 xmlNode *xml_obj = rsc->priv->xml;
187 xmlNode *xml_native_rsc = NULL;
188 group_variant_data_t *group_data = NULL;
189 const char *clone_id = NULL;
190
191 pcmk__rsc_trace(rsc, "Processing resource %s...", rsc->id);
192
193 group_data = pcmk__assert_alloc(1, sizeof(group_variant_data_t));
194 group_data->last_child = NULL;
195 rsc->priv->variant_opaque = group_data;
196
197 // @COMPAT These are deprecated since 2.1.5
198 set_group_flag(rsc, PCMK_META_ORDERED, pcmk__group_ordered,
200 set_group_flag(rsc, "collocated", pcmk__group_colocated,
202
203 clone_id = crm_element_value(rsc->priv->xml, PCMK__META_CLONE);
204
205 for (xml_native_rsc = pcmk__xe_first_child(xml_obj, PCMK_XE_PRIMITIVE,
206 NULL, NULL);
207 xml_native_rsc != NULL;
208 xml_native_rsc = pcmk__xe_next(xml_native_rsc, PCMK_XE_PRIMITIVE)) {
209
210 pcmk_resource_t *new_rsc = NULL;
211
212 crm_xml_add(xml_native_rsc, PCMK__META_CLONE, clone_id);
213 if (pe__unpack_resource(xml_native_rsc, &new_rsc, rsc,
214 rsc->priv->scheduler) != pcmk_rc_ok) {
215 continue;
216 }
217
218 rsc->priv->children = g_list_append(rsc->priv->children, new_rsc);
219 group_data->last_child = new_rsc;
220 pcmk__rsc_trace(rsc, "Added %s member %s", rsc->id, new_rsc->id);
221 }
222
223 if (rsc->priv->children == NULL) {
224 // Not possible with schema validation enabled
225 free(group_data);
226 rsc->priv->variant_opaque = NULL;
227 pcmk__config_err("Group %s has no members", rsc->id);
228 return FALSE;
229 }
230
231 return TRUE;
232}
233
234bool
235group_active(const pcmk_resource_t *rsc, bool all)
236{
237 gboolean c_all = TRUE;
238 gboolean c_any = FALSE;
239
240 for (GList *gIter = rsc->priv->children;
241 gIter != NULL; gIter = gIter->next) {
242
243 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
244
245 if (child_rsc->priv->fns->active(child_rsc, all)) {
246 c_any = TRUE;
247 } else {
248 c_all = FALSE;
249 }
250 }
251
252 if (c_any == FALSE) {
253 return FALSE;
254 } else if (all && c_all == FALSE) {
255 return FALSE;
256 }
257 return TRUE;
258}
259
260PCMK__OUTPUT_ARGS("group", "uint32_t", "pcmk_resource_t *", "GList *",
261 "GList *")
262int
263pe__group_xml(pcmk__output_t *out, va_list args)
264{
265 uint32_t show_opts = va_arg(args, uint32_t);
266 pcmk_resource_t *rsc = va_arg(args, pcmk_resource_t *);
267 GList *only_node = va_arg(args, GList *);
268 GList *only_rsc = va_arg(args, GList *);
269
270 const char *desc = NULL;
271
272 int rc = pcmk_rc_no_output;
273
274 gboolean parent_passes = pcmk__str_in_list(rsc_printable_id(rsc), only_rsc, pcmk__str_star_matches) ||
275 (strstr(rsc->id, ":") != NULL && pcmk__str_in_list(rsc->id, only_rsc, pcmk__str_star_matches));
276
277 desc = pe__resource_description(rsc, show_opts);
278
279 if (rsc->priv->fns->is_filtered(rsc, only_rsc, true)) {
280 return rc;
281 }
282
283 for (GList *gIter = rsc->priv->children;
284 gIter != NULL; gIter = gIter->next) {
285
286 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
287
288 if (skip_child_rsc(rsc, child_rsc, parent_passes, only_rsc, show_opts)) {
289 continue;
290 }
291
292 if (rc == pcmk_rc_no_output) {
293 char *count = pcmk__itoa(g_list_length(gIter));
294 const char *maintenance = pcmk__flag_text(rsc->flags,
296 const char *managed = pcmk__flag_text(rsc->flags,
298 const char *disabled = pcmk__btoa(pe__resource_is_disabled(rsc));
299
301 PCMK_XA_ID, rsc->id,
303 PCMK_XA_MAINTENANCE, maintenance,
304 PCMK_XA_MANAGED, managed,
305 PCMK_XA_DISABLED, disabled,
307 NULL);
308 free(count);
310 }
311
312 out->message(out, (const char *) child_rsc->priv->xml->name,
313 show_opts, child_rsc, only_node, only_rsc);
314 }
315
316 if (rc == pcmk_rc_ok) {
318 }
319
320 return rc;
321}
322
323PCMK__OUTPUT_ARGS("group", "uint32_t", "pcmk_resource_t *", "GList *",
324 "GList *")
325int
326pe__group_default(pcmk__output_t *out, va_list args)
327{
328 uint32_t show_opts = va_arg(args, uint32_t);
329 pcmk_resource_t *rsc = va_arg(args, pcmk_resource_t *);
330 GList *only_node = va_arg(args, GList *);
331 GList *only_rsc = va_arg(args, GList *);
332
333 const char *desc = NULL;
334 int rc = pcmk_rc_no_output;
335
336 gboolean parent_passes = pcmk__str_in_list(rsc_printable_id(rsc), only_rsc, pcmk__str_star_matches) ||
337 (strstr(rsc->id, ":") != NULL && pcmk__str_in_list(rsc->id, only_rsc, pcmk__str_star_matches));
338
339 bool active = rsc->priv->fns->active(rsc, true);
340 bool partially_active = rsc->priv->fns->active(rsc, false);
341
342 desc = pe__resource_description(rsc, show_opts);
343
344 if (rsc->priv->fns->is_filtered(rsc, only_rsc, true)) {
345 return rc;
346 }
347
348 if (pcmk_is_set(show_opts, pcmk_show_brief)) {
349 GList *rscs = pe__filter_rsc_list(rsc->priv->children, only_rsc);
350
351 if (rscs != NULL) {
352 group_header(out, &rc, rsc, !active && partially_active ? inactive_resources(rsc) : 0,
353 pcmk_is_set(show_opts, pcmk_show_inactive_rscs), desc);
354 pe__rscs_brief_output(out, rscs, show_opts | pcmk_show_inactive_rscs);
355
356 rc = pcmk_rc_ok;
357 g_list_free(rscs);
358 }
359
360 } else {
361 for (GList *gIter = rsc->priv->children;
362 gIter != NULL; gIter = gIter->next) {
363 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
364
365 if (skip_child_rsc(rsc, child_rsc, parent_passes, only_rsc, show_opts)) {
366 continue;
367 }
368
369 group_header(out, &rc, rsc, !active && partially_active ? inactive_resources(rsc) : 0,
370 pcmk_is_set(show_opts, pcmk_show_inactive_rscs), desc);
371 out->message(out, (const char *) child_rsc->priv->xml->name,
372 show_opts, child_rsc, only_node, only_rsc);
373 }
374 }
375
377
378 return rc;
379}
380
381void
383{
384 CRM_CHECK(rsc != NULL, return);
385
386 pcmk__rsc_trace(rsc, "Freeing %s", rsc->id);
387
388 for (GList *gIter = rsc->priv->children;
389 gIter != NULL; gIter = gIter->next) {
390
391 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
392
393 pcmk__assert(child_rsc != NULL);
394 pcmk__rsc_trace(child_rsc, "Freeing child %s", child_rsc->id);
395 pcmk__free_resource(child_rsc);
396 }
397
398 pcmk__rsc_trace(rsc, "Freeing child list");
399 g_list_free(rsc->priv->children);
400
401 common_free(rsc);
402}
403
404enum rsc_role_e
405group_resource_state(const pcmk_resource_t *rsc, bool current)
406{
407 enum rsc_role_e group_role = pcmk_role_unknown;
408
409 for (GList *gIter = rsc->priv->children;
410 gIter != NULL; gIter = gIter->next) {
411
412 pcmk_resource_t *child_rsc = (pcmk_resource_t *) gIter->data;
413 enum rsc_role_e role = child_rsc->priv->fns->state(child_rsc,
414 current);
415
416 if (role > group_role) {
417 group_role = role;
418 }
419 }
420
421 pcmk__rsc_trace(rsc, "%s role: %s", rsc->id, pcmk_role_text(group_role));
422 return group_role;
423}
424
425bool
426pe__group_is_filtered(const pcmk_resource_t *rsc, const GList *only_rsc,
427 bool check_parent)
428{
429 bool passes = false;
430
431 if (check_parent
433 false)),
434 only_rsc, pcmk__str_star_matches)) {
435 passes = true;
436 } else if (pcmk__str_in_list(rsc_printable_id(rsc), only_rsc, pcmk__str_star_matches)) {
437 passes = true;
438 } else if (strstr(rsc->id, ":") != NULL && pcmk__str_in_list(rsc->id, only_rsc, pcmk__str_star_matches)) {
439 passes = true;
440 } else {
441 for (const GList *iter = rsc->priv->children;
442 iter != NULL; iter = iter->next) {
443
444 const pcmk_resource_t *child_rsc = iter->data;
445
446 if (!child_rsc->priv->fns->is_filtered(child_rsc, only_rsc,
447 false)) {
448 passes = true;
449 break;
450 }
451 }
452 }
453
454 return !passes;
455}
456
465unsigned int
467{
468 pcmk__assert(pcmk__is_group(rsc));
469 return 1U;
470}
#define pcmk__assert_alloc(nmemb, size)
Definition internal.h:246
uint64_t flags
Definition remote.c:3
#define pcmk_is_set(g, f)
Convenience alias for pcmk_all_flags_set(), to check single flag.
Definition util.h:80
int pe__unpack_resource(xmlNode *xml_obj, pcmk_resource_t **rsc, pcmk_resource_t *parent, pcmk_scheduler_t *scheduler)
Definition complex.c:686
char data[0]
Definition cpg.c:10
struct group_variant_data_s group_variant_data_t
void group_free(pcmk_resource_t *rsc)
Definition group.c:382
unsigned int pe__group_max_per_node(const pcmk_resource_t *rsc)
Definition group.c:466
bool group_active(const pcmk_resource_t *rsc, bool all)
Definition group.c:235
pcmk_resource_t * pe__last_group_member(const pcmk_resource_t *group)
Definition group.c:37
bool pe__group_is_filtered(const pcmk_resource_t *rsc, const GList *only_rsc, bool check_parent)
Definition group.c:426
enum rsc_role_e group_resource_state(const pcmk_resource_t *rsc, bool current)
Definition group.c:405
bool pe__group_flag_is_set(const pcmk_resource_t *group, uint32_t flags)
Definition group.c:59
bool group_unpack(pcmk_resource_t *rsc)
Definition group.c:184
@ pcmk__group_colocated
@ pcmk__group_ordered
#define CRM_CHECK(expr, failure_action)
Definition logging.h:213
#define pcmk__config_err(fmt...)
@ pcmk__wo_group_coloc
@ pcmk__wo_group_order
#define pcmk__warn_once(wo_flag, fmt...)
#define PCMK_META_ORDERED
Definition options.h:100
#define PCMK__META_CLONE
Control output from tools.
@ pcmk_show_brief
Definition output.h:58
@ pcmk_show_inactive_rscs
Definition output.h:63
void pcmk__output_xml_pop_parent(pcmk__output_t *out)
Definition output_xml.c:566
#define PCMK__OUTPUT_LIST_HEADER(out_obj, cond, retcode, title...)
#define PCMK__OUTPUT_LIST_FOOTER(out_obj, retcode)
const char * pe__resource_description(const pcmk_resource_t *rsc, uint32_t show_opts)
Definition pe_output.c:24
const pcmk_resource_t * pe__const_top_resource(const pcmk_resource_t *rsc, bool include_bundle)
Definition complex.c:1025
int pe__name_and_nvpairs_xml(pcmk__output_t *out, bool is_list, const char *tag_name,...) G_GNUC_NULL_TERMINATED
Definition pe_output.c:619
int pe__group_xml(pcmk__output_t *out, va_list args)
GList * pe__filter_rsc_list(GList *rscs, GList *filter)
Definition utils.c:789
void common_free(pcmk_resource_t *rsc)
Definition complex.c:1042
int pe__rscs_brief_output(pcmk__output_t *out, GList *rsc_list, unsigned int options)
bool pe__resource_is_disabled(const pcmk_resource_t *rsc)
Definition utils.c:728
int pe__group_default(pcmk__output_t *out, va_list args)
@ pcmk__rsc_managed
@ pcmk__rsc_maintenance
void pcmk__free_resource(gpointer user_data)
Definition resources.c:25
@ pcmk_rc_no_output
Definition results.h:128
@ pcmk_rc_ok
Definition results.h:159
#define pcmk__assert(expr)
const char * pcmk_role_text(enum rsc_role_e role)
Get readable description of a resource role.
Definition roles.c:23
rsc_role_e
Definition roles.h:34
@ pcmk_role_unknown
Resource role is unknown.
Definition roles.h:35
#define pcmk__rsc_trace(rsc, fmt, args...)
Cluster status and scheduling.
const char * rsc_printable_id(const pcmk_resource_t *rsc)
Definition utils.c:581
int crm_str_to_boolean(const char *s, int *ret)
Definition strings.c:498
gboolean pcmk__str_in_list(const gchar *s, const GList *lst, uint32_t flags)
Definition strings.c:984
#define pcmk__plural_s(i)
@ pcmk__str_none
@ pcmk__str_star_matches
void pcmk__add_separated_word(GString **list, size_t init_size, const char *word, const char *separator)
Definition strings.c:796
This structure contains everything that makes up a single output formatter.
pcmk_scheduler_t * scheduler
const pcmk__rsc_methods_t * fns
unsigned long long flags
Definition resources.h:69
pcmk__resource_private_t * priv
Definition resources.h:61
enum rsc_role_e(* state)(const pcmk_resource_t *rsc, bool current)
bool(* is_filtered)(const pcmk_resource_t *rsc, const GList *only_rsc, bool check_parent)
bool(* active)(const pcmk_resource_t *rsc, bool all)
Wrappers for and extensions to libxml2.
const char * crm_element_value(const xmlNode *data, const char *name)
Retrieve the value of an XML attribute.
const char * crm_xml_add(xmlNode *node, const char *name, const char *value)
Create an XML attribute with specified name and value.
xmlNode * pcmk__xe_first_child(const xmlNode *parent, const char *node_name, const char *attr_n, const char *attr_v)
Definition xml_element.c:43
xmlNode * pcmk__xe_next(const xmlNode *node, const char *element_name)
#define PCMK_XA_DESCRIPTION
Definition xml_names.h:261
#define PCMK_XE_GROUP
Definition xml_names.h:119
#define PCMK_XA_NUMBER_RESOURCES
Definition xml_names.h:343
#define PCMK_XA_ID
Definition xml_names.h:301
#define PCMK_XA_MAINTENANCE
Definition xml_names.h:321
#define PCMK_XE_PRIMITIVE
Definition xml_names.h:164
#define PCMK_XA_MANAGED
Definition xml_names.h:323
#define PCMK_XA_DISABLED
Definition xml_names.h:265