/src/pacemaker/include/crm/pengine/internal.h
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | * Copyright 2004-2024 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 | | #ifndef PCMK__CRM_PENGINE_INTERNAL__H |
11 | | #define PCMK__CRM_PENGINE_INTERNAL__H |
12 | | |
13 | | #include <stdbool.h> |
14 | | #include <stdint.h> |
15 | | #include <string.h> |
16 | | #include <crm/common/xml.h> |
17 | | #include <crm/pengine/status.h> |
18 | | #include <crm/pengine/remote_internal.h> |
19 | | #include <crm/common/internal.h> |
20 | | #include <crm/common/options_internal.h> |
21 | | #include <crm/common/output_internal.h> |
22 | | #include <crm/common/scheduler_internal.h> |
23 | | |
24 | | #ifdef __cplusplus |
25 | | extern "C" { |
26 | | #endif |
27 | | |
28 | | const char *pe__resource_description(const pcmk_resource_t *rsc, |
29 | | uint32_t show_opts); |
30 | | |
31 | | bool pe__clone_is_ordered(const pcmk_resource_t *clone); |
32 | | int pe__set_clone_flag(pcmk_resource_t *clone, enum pcmk__clone_flags flag); |
33 | | bool pe__clone_flag_is_set(const pcmk_resource_t *clone, uint32_t flags); |
34 | | |
35 | | bool pe__group_flag_is_set(const pcmk_resource_t *group, uint32_t flags); |
36 | | pcmk_resource_t *pe__last_group_member(const pcmk_resource_t *group); |
37 | | |
38 | | const pcmk_resource_t *pe__const_top_resource(const pcmk_resource_t *rsc, |
39 | | bool include_bundle); |
40 | | |
41 | | int pe__clone_max(const pcmk_resource_t *clone); |
42 | | int pe__clone_node_max(const pcmk_resource_t *clone); |
43 | | int pe__clone_promoted_max(const pcmk_resource_t *clone); |
44 | | int pe__clone_promoted_node_max(const pcmk_resource_t *clone); |
45 | | void pe__create_clone_notifications(pcmk_resource_t *clone); |
46 | | void pe__free_clone_notification_data(pcmk_resource_t *clone); |
47 | | void pe__create_clone_notif_pseudo_ops(pcmk_resource_t *clone, |
48 | | pcmk_action_t *start, |
49 | | pcmk_action_t *started, |
50 | | pcmk_action_t *stop, |
51 | | pcmk_action_t *stopped); |
52 | | |
53 | | pcmk_action_t *pe__new_rsc_pseudo_action(pcmk_resource_t *rsc, const char *task, |
54 | | bool optional, bool runnable); |
55 | | |
56 | | void pe__create_promotable_pseudo_ops(pcmk_resource_t *clone, |
57 | | bool any_promoting, bool any_demoting); |
58 | | |
59 | | bool pe_can_fence(const pcmk_scheduler_t *scheduler, const pcmk_node_t *node); |
60 | | |
61 | | char *native_parameter(pcmk_resource_t *rsc, pcmk_node_t *node, gboolean create, |
62 | | const char *name, pcmk_scheduler_t *scheduler); |
63 | | pcmk_node_t *native_location(const pcmk_resource_t *rsc, GList **list, |
64 | | uint32_t target); |
65 | | void native_add_running(pcmk_resource_t *rsc, pcmk_node_t *node, |
66 | | pcmk_scheduler_t *scheduler, gboolean failed); |
67 | | |
68 | | gboolean native_unpack(pcmk_resource_t *rsc, pcmk_scheduler_t *scheduler); |
69 | | gboolean group_unpack(pcmk_resource_t *rsc, pcmk_scheduler_t *scheduler); |
70 | | gboolean clone_unpack(pcmk_resource_t *rsc, pcmk_scheduler_t *scheduler); |
71 | | gboolean pe__unpack_bundle(pcmk_resource_t *rsc, pcmk_scheduler_t *scheduler); |
72 | | |
73 | | pcmk_resource_t *native_find_rsc(pcmk_resource_t *rsc, const char *id, |
74 | | const pcmk_node_t *node, int flags); |
75 | | |
76 | | gboolean native_active(pcmk_resource_t *rsc, gboolean all); |
77 | | gboolean group_active(pcmk_resource_t *rsc, gboolean all); |
78 | | gboolean clone_active(pcmk_resource_t *rsc, gboolean all); |
79 | | gboolean pe__bundle_active(pcmk_resource_t *rsc, gboolean all); |
80 | | |
81 | | gchar *pcmk__native_output_string(const pcmk_resource_t *rsc, const char *name, |
82 | | const pcmk_node_t *node, uint32_t show_opts, |
83 | | const char *target_role, bool show_nodes); |
84 | | |
85 | | int pe__name_and_nvpairs_xml(pcmk__output_t *out, bool is_list, const char *tag_name, |
86 | | ...) G_GNUC_NULL_TERMINATED; |
87 | | char *pe__node_display_name(pcmk_node_t *node, bool print_detail); |
88 | | |
89 | | |
90 | | // Clone notifications (pe_notif.c) |
91 | | void pe__order_notifs_after_fencing(const pcmk_action_t *action, |
92 | | pcmk_resource_t *rsc, |
93 | | pcmk_action_t *stonith_op); |
94 | | |
95 | | |
96 | | // Resource output methods |
97 | | int pe__clone_xml(pcmk__output_t *out, va_list args); |
98 | | int pe__clone_default(pcmk__output_t *out, va_list args); |
99 | | int pe__group_xml(pcmk__output_t *out, va_list args); |
100 | | int pe__group_default(pcmk__output_t *out, va_list args); |
101 | | int pe__bundle_xml(pcmk__output_t *out, va_list args); |
102 | | int pe__bundle_html(pcmk__output_t *out, va_list args); |
103 | | int pe__bundle_text(pcmk__output_t *out, va_list args); |
104 | | int pe__node_html(pcmk__output_t *out, va_list args); |
105 | | int pe__node_text(pcmk__output_t *out, va_list args); |
106 | | int pe__node_xml(pcmk__output_t *out, va_list args); |
107 | | int pe__resource_xml(pcmk__output_t *out, va_list args); |
108 | | int pe__resource_html(pcmk__output_t *out, va_list args); |
109 | | int pe__resource_text(pcmk__output_t *out, va_list args); |
110 | | |
111 | | void native_free(pcmk_resource_t *rsc); |
112 | | void group_free(pcmk_resource_t *rsc); |
113 | | void clone_free(pcmk_resource_t *rsc); |
114 | | void pe__free_bundle(pcmk_resource_t *rsc); |
115 | | |
116 | | enum rsc_role_e native_resource_state(const pcmk_resource_t *rsc, |
117 | | gboolean current); |
118 | | enum rsc_role_e group_resource_state(const pcmk_resource_t *rsc, |
119 | | gboolean current); |
120 | | enum rsc_role_e clone_resource_state(const pcmk_resource_t *rsc, |
121 | | gboolean current); |
122 | | enum rsc_role_e pe__bundle_resource_state(const pcmk_resource_t *rsc, |
123 | | gboolean current); |
124 | | |
125 | | void pe__count_common(pcmk_resource_t *rsc); |
126 | | void pe__count_bundle(pcmk_resource_t *rsc); |
127 | | |
128 | | void common_free(pcmk_resource_t *rsc); |
129 | | |
130 | | pcmk_node_t *pe__copy_node(const pcmk_node_t *this_node); |
131 | | time_t get_effective_time(pcmk_scheduler_t *scheduler); |
132 | | |
133 | | /* Failure handling utilities (from failcounts.c) */ |
134 | | |
135 | | int pe_get_failcount(const pcmk_node_t *node, pcmk_resource_t *rsc, |
136 | | time_t *last_failure, uint32_t flags, |
137 | | const xmlNode *xml_op); |
138 | | |
139 | | pcmk_action_t *pe__clear_failcount(pcmk_resource_t *rsc, |
140 | | const pcmk_node_t *node, const char *reason, |
141 | | pcmk_scheduler_t *scheduler); |
142 | | |
143 | | /* Functions for finding/counting a resource's active nodes */ |
144 | | |
145 | | bool pe__count_active_node(const pcmk_resource_t *rsc, pcmk_node_t *node, |
146 | | pcmk_node_t **active, unsigned int *count_all, |
147 | | unsigned int *count_clean); |
148 | | |
149 | | pcmk_node_t *pe__find_active_requires(const pcmk_resource_t *rsc, |
150 | | unsigned int *count); |
151 | | |
152 | | /* Binary like operators for lists of nodes */ |
153 | | GHashTable *pe__node_list2table(const GList *list); |
154 | | |
155 | | pcmk_action_t *get_pseudo_op(const char *name, pcmk_scheduler_t *scheduler); |
156 | | gboolean order_actions(pcmk_action_t *first, pcmk_action_t *then, |
157 | | uint32_t flags); |
158 | | |
159 | | void pe__show_node_scores_as(const char *file, const char *function, |
160 | | int line, bool to_log, const pcmk_resource_t *rsc, |
161 | | const char *comment, GHashTable *nodes, |
162 | | pcmk_scheduler_t *scheduler); |
163 | | |
164 | | #define pe__show_node_scores(level, rsc, text, nodes, scheduler) \ |
165 | | pe__show_node_scores_as(__FILE__, __func__, __LINE__, \ |
166 | | (level), (rsc), (text), (nodes), (scheduler)) |
167 | | |
168 | | GHashTable *pcmk__unpack_action_meta(pcmk_resource_t *rsc, |
169 | | const pcmk_node_t *node, |
170 | | const char *action_name, guint interval_ms, |
171 | | const xmlNode *action_config); |
172 | | GHashTable *pcmk__unpack_action_rsc_params(const xmlNode *action_xml, |
173 | | GHashTable *node_attrs, |
174 | | pcmk_scheduler_t *data_set); |
175 | | xmlNode *pcmk__find_action_config(const pcmk_resource_t *rsc, |
176 | | const char *action_name, guint interval_ms, |
177 | | bool include_disabled); |
178 | | |
179 | | enum pcmk__requires pcmk__action_requires(const pcmk_resource_t *rsc, |
180 | | const char *action_name); |
181 | | |
182 | | enum pcmk__on_fail pcmk__parse_on_fail(const pcmk_resource_t *rsc, |
183 | | const char *action_name, |
184 | | guint interval_ms, const char *value); |
185 | | |
186 | | enum rsc_role_e pcmk__role_after_failure(const pcmk_resource_t *rsc, |
187 | | const char *action_name, |
188 | | enum pcmk__on_fail on_fail, |
189 | | GHashTable *meta); |
190 | | |
191 | | pcmk_action_t *custom_action(pcmk_resource_t *rsc, char *key, const char *task, |
192 | | const pcmk_node_t *on_node, gboolean optional, |
193 | | pcmk_scheduler_t *scheduler); |
194 | | |
195 | | #define delete_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_DELETE, 0) |
196 | | #define stop_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_STOP, 0) |
197 | | #define reload_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_RELOAD_AGENT, 0) |
198 | | #define start_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_START, 0) |
199 | | #define promote_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_PROMOTE, 0) |
200 | | #define demote_key(rsc) pcmk__op_key((rsc)->id, PCMK_ACTION_DEMOTE, 0) |
201 | | |
202 | | #define delete_action(rsc, node, optional) \ |
203 | | custom_action((rsc), delete_key(rsc), PCMK_ACTION_DELETE, \ |
204 | | (node), (optional), (rsc)->priv->scheduler) |
205 | | |
206 | | #define stop_action(rsc, node, optional) \ |
207 | | custom_action((rsc), stop_key(rsc), PCMK_ACTION_STOP, \ |
208 | | (node), (optional), (rsc)->priv->scheduler) |
209 | | |
210 | | #define start_action(rsc, node, optional) \ |
211 | | custom_action((rsc), start_key(rsc), PCMK_ACTION_START, \ |
212 | | (node), (optional), (rsc)->priv->scheduler) |
213 | | |
214 | | #define promote_action(rsc, node, optional) \ |
215 | | custom_action((rsc), promote_key(rsc), PCMK_ACTION_PROMOTE, \ |
216 | | (node), (optional), (rsc)->priv->scheduler) |
217 | | |
218 | | #define demote_action(rsc, node, optional) \ |
219 | | custom_action((rsc), demote_key(rsc), PCMK_ACTION_DEMOTE, \ |
220 | | (node), (optional), (rsc)->priv->scheduler) |
221 | | |
222 | | pcmk_action_t *find_first_action(const GList *input, const char *uuid, |
223 | | const char *task, const pcmk_node_t *on_node); |
224 | | |
225 | | enum pcmk__action_type get_complex_task(const pcmk_resource_t *rsc, |
226 | | const char *name); |
227 | | |
228 | | GList *find_actions(GList *input, const char *key, const pcmk_node_t *on_node); |
229 | | GList *find_actions_exact(GList *input, const char *key, |
230 | | const pcmk_node_t *on_node); |
231 | | GList *pe__resource_actions(const pcmk_resource_t *rsc, const pcmk_node_t *node, |
232 | | const char *task, bool require_node); |
233 | | |
234 | | extern void pe_free_action(pcmk_action_t *action); |
235 | | |
236 | | void resource_location(pcmk_resource_t *rsc, const pcmk_node_t *node, int score, |
237 | | const char *tag, pcmk_scheduler_t *scheduler); |
238 | | |
239 | | int pe__is_newer_op(const xmlNode *xml_a, const xmlNode *xml_b); |
240 | | extern gint sort_op_by_callid(gconstpointer a, gconstpointer b); |
241 | | gboolean get_target_role(const pcmk_resource_t *rsc, enum rsc_role_e *role); |
242 | | void pe__set_next_role(pcmk_resource_t *rsc, enum rsc_role_e role, |
243 | | const char *why); |
244 | | |
245 | | extern void destroy_ticket(gpointer data); |
246 | | pcmk__ticket_t *ticket_new(const char *ticket_id, pcmk_scheduler_t *scheduler); |
247 | | |
248 | | // Resources for manipulating resource names |
249 | | const char *pe_base_name_end(const char *id); |
250 | | char *clone_strip(const char *last_rsc_id); |
251 | | char *clone_zero(const char *last_rsc_id); |
252 | | |
253 | | static inline bool |
254 | | pe_base_name_eq(const pcmk_resource_t *rsc, const char *id) |
255 | 0 | { |
256 | 0 | if (id && rsc && rsc->id) { |
257 | 0 | // Number of characters in rsc->id before any clone suffix |
258 | 0 | size_t base_len = pe_base_name_end(rsc->id) - rsc->id + 1; |
259 | 0 |
|
260 | 0 | return (strlen(id) == base_len) && !strncmp(id, rsc->id, base_len); |
261 | 0 | } |
262 | 0 | return false; |
263 | 0 | } |
264 | | |
265 | | int pe__target_rc_from_xml(const xmlNode *xml_op); |
266 | | |
267 | | gint pe__cmp_node_name(gconstpointer a, gconstpointer b); |
268 | | bool is_set_recursive(const pcmk_resource_t *rsc, long long flag, bool any); |
269 | | |
270 | | pcmk__op_digest_t *pe__calculate_digests(pcmk_resource_t *rsc, const char *task, |
271 | | guint *interval_ms, |
272 | | const pcmk_node_t *node, |
273 | | const xmlNode *xml_op, |
274 | | GHashTable *overrides, |
275 | | bool calc_secure, |
276 | | pcmk_scheduler_t *scheduler); |
277 | | |
278 | | void pe__free_digests(gpointer ptr); |
279 | | |
280 | | pcmk__op_digest_t *rsc_action_digest_cmp(pcmk_resource_t *rsc, |
281 | | const xmlNode *xml_op, |
282 | | pcmk_node_t *node, |
283 | | pcmk_scheduler_t *scheduler); |
284 | | |
285 | | pcmk_action_t *pe_fence_op(pcmk_node_t *node, const char *op, bool optional, |
286 | | const char *reason, bool priority_delay, |
287 | | pcmk_scheduler_t *scheduler); |
288 | | void trigger_unfencing(pcmk_resource_t *rsc, pcmk_node_t *node, |
289 | | const char *reason, pcmk_action_t *dependency, |
290 | | pcmk_scheduler_t *scheduler); |
291 | | |
292 | | char *pe__action2reason(const pcmk_action_t *action, |
293 | | enum pcmk__action_flags flag); |
294 | | void pe_action_set_reason(pcmk_action_t *action, const char *reason, |
295 | | bool overwrite); |
296 | | void pe__add_action_expected_result(pcmk_action_t *action, int expected_result); |
297 | | |
298 | | void pe__set_resource_flags_recursive(pcmk_resource_t *rsc, uint64_t flags); |
299 | | void pe__clear_resource_flags_recursive(pcmk_resource_t *rsc, uint64_t flags); |
300 | | void pe__clear_resource_flags_on_all(pcmk_scheduler_t *scheduler, |
301 | | uint64_t flag); |
302 | | |
303 | | int pe__rscs_brief_output(pcmk__output_t *out, GList *rsc_list, unsigned int options); |
304 | | void pe_fence_node(pcmk_scheduler_t *scheduler, pcmk_node_t *node, |
305 | | const char *reason, bool priority_delay); |
306 | | |
307 | | pcmk_node_t *pe_create_node(const char *id, const char *uname, const char *type, |
308 | | int score, pcmk_scheduler_t *scheduler); |
309 | | |
310 | | int pe__common_output_text(pcmk__output_t *out, const pcmk_resource_t *rsc, |
311 | | const char *name, const pcmk_node_t *node, |
312 | | unsigned int options); |
313 | | int pe__common_output_html(pcmk__output_t *out, const pcmk_resource_t *rsc, |
314 | | const char *name, const pcmk_node_t *node, |
315 | | unsigned int options); |
316 | | |
317 | | GList *pe__bundle_containers(const pcmk_resource_t *bundle); |
318 | | |
319 | | int pe__bundle_max(const pcmk_resource_t *rsc); |
320 | | bool pe__node_is_bundle_instance(const pcmk_resource_t *bundle, |
321 | | const pcmk_node_t *node); |
322 | | pcmk_resource_t *pe__bundled_resource(const pcmk_resource_t *rsc); |
323 | | const pcmk_resource_t *pe__get_rsc_in_container(const pcmk_resource_t *instance); |
324 | | pcmk_resource_t *pe__first_container(const pcmk_resource_t *bundle); |
325 | | void pe__foreach_bundle_replica(pcmk_resource_t *bundle, |
326 | | bool (*fn)(pcmk__bundle_replica_t *, void *), |
327 | | void *user_data); |
328 | | void pe__foreach_const_bundle_replica(const pcmk_resource_t *bundle, |
329 | | bool (*fn)(const pcmk__bundle_replica_t *, |
330 | | void *), |
331 | | void *user_data); |
332 | | pcmk_resource_t *pe__find_bundle_replica(const pcmk_resource_t *bundle, |
333 | | const pcmk_node_t *node); |
334 | | bool pe__bundle_needs_remote_name(pcmk_resource_t *rsc); |
335 | | const char *pe__add_bundle_remote_name(pcmk_resource_t *rsc, xmlNode *xml, |
336 | | const char *field); |
337 | | bool pe__is_universal_clone(const pcmk_resource_t *rsc, |
338 | | const pcmk_scheduler_t *scheduler); |
339 | | void pe__add_param_check(const xmlNode *rsc_op, pcmk_resource_t *rsc, |
340 | | pcmk_node_t *node, enum pcmk__check_parameters, |
341 | | pcmk_scheduler_t *scheduler); |
342 | | void pe__foreach_param_check(pcmk_scheduler_t *scheduler, |
343 | | void (*cb)(pcmk_resource_t*, pcmk_node_t*, |
344 | | const xmlNode*, |
345 | | enum pcmk__check_parameters)); |
346 | | void pe__free_param_checks(pcmk_scheduler_t *scheduler); |
347 | | |
348 | | bool pe__shutdown_requested(const pcmk_node_t *node); |
349 | | void pe__update_recheck_time(time_t recheck, pcmk_scheduler_t *scheduler, |
350 | | const char *reason); |
351 | | |
352 | | /*! |
353 | | * \internal |
354 | | * \brief Register xml formatting message functions. |
355 | | * |
356 | | * \param[in,out] out Output object to register messages with |
357 | | */ |
358 | | void pe__register_messages(pcmk__output_t *out); |
359 | | |
360 | | void pe__unpack_dataset_nvpairs(const xmlNode *xml_obj, const char *set_name, |
361 | | const pe_rule_eval_data_t *rule_data, |
362 | | GHashTable *hash, const char *always_first, |
363 | | pcmk_scheduler_t *scheduler); |
364 | | |
365 | | bool pe__resource_is_disabled(const pcmk_resource_t *rsc); |
366 | | void pe__clear_resource_history(pcmk_resource_t *rsc, const pcmk_node_t *node); |
367 | | |
368 | | GList *pe__rscs_with_tag(pcmk_scheduler_t *scheduler, const char *tag_name); |
369 | | GList *pe__unames_with_tag(pcmk_scheduler_t *scheduler, const char *tag_name); |
370 | | bool pe__rsc_has_tag(pcmk_scheduler_t *scheduler, const char *rsc, |
371 | | const char *tag); |
372 | | bool pe__uname_has_tag(pcmk_scheduler_t *scheduler, const char *node, |
373 | | const char *tag); |
374 | | |
375 | | bool pe__rsc_running_on_only(const pcmk_resource_t *rsc, |
376 | | const pcmk_node_t *node); |
377 | | bool pe__rsc_running_on_any(pcmk_resource_t *rsc, GList *node_list); |
378 | | GList *pe__filter_rsc_list(GList *rscs, GList *filter); |
379 | | GList * pe__build_node_name_list(pcmk_scheduler_t *scheduler, const char *s); |
380 | | GList * pe__build_rsc_list(pcmk_scheduler_t *scheduler, const char *s); |
381 | | |
382 | | bool pcmk__rsc_filtered_by_node(pcmk_resource_t *rsc, GList *only_node); |
383 | | |
384 | | gboolean pe__bundle_is_filtered(const pcmk_resource_t *rsc, GList *only_rsc, |
385 | | gboolean check_parent); |
386 | | gboolean pe__clone_is_filtered(const pcmk_resource_t *rsc, GList *only_rsc, |
387 | | gboolean check_parent); |
388 | | gboolean pe__group_is_filtered(const pcmk_resource_t *rsc, GList *only_rsc, |
389 | | gboolean check_parent); |
390 | | gboolean pe__native_is_filtered(const pcmk_resource_t *rsc, GList *only_rsc, |
391 | | gboolean check_parent); |
392 | | |
393 | | xmlNode *pe__failed_probe_for_rsc(const pcmk_resource_t *rsc, const char *name); |
394 | | |
395 | | const char *pe__clone_child_id(const pcmk_resource_t *rsc); |
396 | | |
397 | | int pe__sum_node_health_scores(const pcmk_node_t *node, int base_health); |
398 | | int pe__node_health(pcmk_node_t *node); |
399 | | |
400 | | static inline enum pcmk__health_strategy |
401 | | pe__health_strategy(pcmk_scheduler_t *scheduler) |
402 | 0 | { |
403 | 0 | const char *strategy = pcmk__cluster_option(scheduler->priv->options, |
404 | 0 | PCMK_OPT_NODE_HEALTH_STRATEGY); |
405 | 0 |
|
406 | 0 | return pcmk__parse_health_strategy(strategy); |
407 | 0 | } |
408 | | |
409 | | #ifdef __cplusplus |
410 | | } |
411 | | #endif |
412 | | |
413 | | #endif // PCMK__CRM_PENGINE_INTERNAL__H |