root/daemons/attrd/attrd_elections.c

/* [previous][next][first][last][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. attrd_election_cb
  2. attrd_election_init
  3. attrd_start_election_if_needed
  4. attrd_election_won
  5. attrd_handle_election_op
  6. attrd_check_for_new_writer
  7. attrd_declare_winner
  8. attrd_remove_voter
  9. attrd_xml_add_writer

   1 /*
   2  * Copyright 2013-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 General Public License version 2
   7  * or later (GPLv2+) WITHOUT ANY WARRANTY.
   8  */
   9 
  10 #include <crm_internal.h>
  11 #include <crm/cluster.h>
  12 #include <crm/cluster/election_internal.h>
  13 #include <crm/common/xml.h>
  14 
  15 #include "pacemaker-attrd.h"
  16 
  17 static char *peer_writer = NULL;
  18 
  19 static void
  20 attrd_election_cb(pcmk_cluster_t *cluster)
     /* [previous][next][first][last][top][bottom][index][help] */
  21 {
  22     attrd_declare_winner();
  23 
  24     /* Update the peers after an election */
  25     attrd_peer_sync(NULL);
  26 
  27     /* After winning an election, update the CIB with the values of all
  28      * attributes as the winner knows them.
  29      */
  30     attrd_write_attributes(attrd_write_all);
  31 }
  32 
  33 void
  34 attrd_election_init(void)
     /* [previous][next][first][last][top][bottom][index][help] */
  35 {
  36     election_init(attrd_cluster, attrd_election_cb);
  37 }
  38 
  39 void
  40 attrd_start_election_if_needed(void)
     /* [previous][next][first][last][top][bottom][index][help] */
  41 {
  42     if ((peer_writer == NULL)
  43         && (election_state(attrd_cluster) != election_in_progress)
  44         && !attrd_shutting_down(false)) {
  45 
  46         crm_info("Starting an election to determine the writer");
  47         election_vote(attrd_cluster);
  48     }
  49 }
  50 
  51 bool
  52 attrd_election_won(void)
     /* [previous][next][first][last][top][bottom][index][help] */
  53 {
  54     return (election_state(attrd_cluster) == election_won);
  55 }
  56 
  57 void
  58 attrd_handle_election_op(const pcmk__node_status_t *peer, xmlNode *xml)
     /* [previous][next][first][last][top][bottom][index][help] */
  59 {
  60     enum election_result rc = 0;
  61     enum election_result previous = election_state(attrd_cluster);
  62 
  63     crm_xml_add(xml, PCMK__XA_SRC, peer->name);
  64 
  65     // Don't become writer if we're shutting down
  66     rc = election_count_vote(attrd_cluster, xml, !attrd_shutting_down(false));
  67 
  68     switch(rc) {
  69         case election_start:
  70             crm_debug("Unsetting writer (was %s) and starting new election",
  71                       peer_writer? peer_writer : "unset");
  72             free(peer_writer);
  73             peer_writer = NULL;
  74             election_vote(attrd_cluster);
  75             break;
  76 
  77         case election_lost:
  78             /* The election API should really distinguish between "we just lost
  79              * to this peer" and "we already lost previously, and we are
  80              * discarding this vote for some reason", but it doesn't.
  81              *
  82              * In the first case, we want to tentatively set the peer writer to
  83              * this peer, even though another peer may eventually win (which we
  84              * will learn via attrd_check_for_new_writer()), so
  85              * attrd_start_election_if_needed() doesn't start a new election.
  86              *
  87              * Approximate a test for that case as best as possible.
  88              */
  89             if ((peer_writer == NULL) || (previous != election_lost)) {
  90                 pcmk__str_update(&peer_writer, peer->name);
  91                 crm_debug("Election lost, presuming %s is writer for now",
  92                           peer_writer);
  93             }
  94             break;
  95 
  96         case election_in_progress:
  97             election_check(attrd_cluster);
  98             break;
  99 
 100         default:
 101             crm_info("Ignoring election op from %s due to error", peer->name);
 102             break;
 103     }
 104 }
 105 
 106 bool
 107 attrd_check_for_new_writer(const pcmk__node_status_t *peer, const xmlNode *xml)
     /* [previous][next][first][last][top][bottom][index][help] */
 108 {
 109     int peer_state = 0;
 110 
 111     crm_element_value_int(xml, PCMK__XA_ATTR_WRITER, &peer_state);
 112     if (peer_state == election_won) {
 113         if ((election_state(attrd_cluster) == election_won)
 114             && !pcmk__str_eq(peer->name, attrd_cluster->priv->node_name,
 115                              pcmk__str_casei)) {
 116             crm_notice("Detected another attribute writer (%s), starting new "
 117                        "election",
 118                        peer->name);
 119             election_vote(attrd_cluster);
 120 
 121         } else if (!pcmk__str_eq(peer->name, peer_writer, pcmk__str_casei)) {
 122             crm_notice("Recorded new attribute writer: %s (was %s)",
 123                        peer->name, pcmk__s(peer_writer, "unset"));
 124             pcmk__str_update(&peer_writer, peer->name);
 125         }
 126     }
 127     return (peer_state == election_won);
 128 }
 129 
 130 void
 131 attrd_declare_winner(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 132 {
 133     crm_notice("Recorded local node as attribute writer (was %s)",
 134                (peer_writer? peer_writer : "unset"));
 135     pcmk__str_update(&peer_writer, attrd_cluster->priv->node_name);
 136 }
 137 
 138 void
 139 attrd_remove_voter(const pcmk__node_status_t *peer)
     /* [previous][next][first][last][top][bottom][index][help] */
 140 {
 141     election_remove(attrd_cluster, peer->name);
 142     if ((peer_writer != NULL)
 143         && pcmk__str_eq(peer->name, peer_writer, pcmk__str_casei)) {
 144 
 145         free(peer_writer);
 146         peer_writer = NULL;
 147         crm_notice("Lost attribute writer %s", peer->name);
 148 
 149         /* Clear any election dampening in effect. Otherwise, if the lost writer
 150          * had just won, the election could fizzle out with no new writer.
 151          */
 152         election_clear_dampening(attrd_cluster);
 153 
 154         /* If the writer received attribute updates during its shutdown, it will
 155          * not have written them to the CIB. Ensure we get a new writer so they
 156          * are written out. This means that every node that sees the writer
 157          * leave will start a new election, but that's better than losing
 158          * attributes.
 159          */
 160         attrd_start_election_if_needed();
 161 
 162     /* If an election is in progress, we need to call election_check(), in case
 163      * this lost peer is the only one that hasn't voted, otherwise the election
 164      * would be pending until it's timed out.
 165      */
 166     } else if (election_state(attrd_cluster) == election_in_progress) {
 167        crm_debug("Checking election status upon loss of voter %s", peer->name);
 168        election_check(attrd_cluster);
 169     }
 170 }
 171 
 172 void
 173 attrd_xml_add_writer(xmlNode *xml)
     /* [previous][next][first][last][top][bottom][index][help] */
 174 {
 175     crm_xml_add_int(xml, PCMK__XA_ATTR_WRITER, election_state(attrd_cluster));
 176 }

/* [previous][next][first][last][top][bottom][index][help] */