Puppet Class: tripleo::fencing

Defined in:
manifests/fencing.pp

Overview

Class: tripleo::fencing

Configure Pacemaker fencing devices for TripleO.

Parameters:

config

JSON config of fencing devices, using the following structure:

{
  "devices": [
    {
      "agent": "AGENT_NAME",
      "host_mac": "HOST_MAC_ADDRESS",
      "params": {"PARAM_NAME": "PARAM_VALUE"}
    }
  ]
}

For instance:

{
  "devices": [
    {
      "agent": "fence_xvm",
      "host_mac": "52:54:00:aa:bb:cc",
      "params": {
        "multicast_address": "225.0.0.12",
        "port": "baremetal_0",
        "manage_fw": true,
        "manage_key_file": true,
        "key_file": "/etc/fence_xvm.key",
        "key_file_password": "abcdef"
      }
    }
  ]
}

Defaults to {}

tries

Number of attempts when creating fence devices and constraints. Defaults to 10

try_sleep

Delay (in seconds) between attempts when creating fence devices and constraints. Defaults to 3

deep_compare

Enable deep comparing of resources and bundles When set to true a resource will be compared in full (options, meta parameters,..) to the existing one and in case of difference it will be repushed to the CIB Defaults to false

update_settle_secs

When deep_compare is enabled and puppet updates a resource, this parameter represents the number (in seconds) to wait for the cluster to settle after the resource update. Defaults to 600 (seconds)

watchdog_timeout

Only valid if sbd watchdog fencing is enabled. Pacemaker will assume unseen nodes self-fence within this much time. Defaults to 60 (seconds)

enable_instanceha

(Optional) Boolean driving the Instance HA controlplane configuration Defaults to lookup(‘tripleo::instanceha’, undef, undef, false),

Parameters:

  • config (Any) (defaults to: {})
  • tries (Any) (defaults to: 10)
  • try_sleep (Any) (defaults to: 3)
  • deep_compare (Any) (defaults to: false)
  • update_settle_secs (Any) (defaults to: 600)
  • watchdog_timeout (Any) (defaults to: 60)
  • enable_instanceha (Any) (defaults to: lookup('tripleo::instanceha', undef, undef, false))


67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
# File 'manifests/fencing.pp', line 67

class tripleo::fencing(
  $config             = {},
  $tries              = 10,
  $try_sleep          = 3,
  $deep_compare       = false,
  $update_settle_secs = 600,
  $watchdog_timeout   = 60,
  $enable_instanceha  = lookup('tripleo::instanceha', undef, undef, false),
) {
  $common_params = {
    'tries' => $tries,
    'try_sleep' => $try_sleep,
    'deep_compare' => $deep_compare,
    'update_settle_secs' => $update_settle_secs
  }

  # check if instanceha is enabled
  if member(lookup('compute_instanceha_short_node_names', undef, undef, []), downcase($::hostname)) {
    $is_compute_instanceha_node = true
  } else {
    $is_compute_instanceha_node = false
  }

  $content = $config['devices']

  # check if the devices: section in fence.yaml contains levels.
  # if it doesn't, assume level=1 and build a hash with the content.
  $all_levels = $content ? {
    Array   => {'level1' => $content},
    default => $content
  }

  # collect the number of stonith levels currently defined for this system
  # and convert it to integer.
  $local_levels = 0 + $facts['stonith_levels']

  # if the number of levels defined on this system is greater than the number in hiera
  # we need to delete the delta.
  if $local_levels > $all_levels.length {
    $begin = $all_levels.length + 1
    range("${begin}", "${local_levels}").each |$level|{
        pacemaker::stonith::level{ "stonith-${level}":
          ensure            => 'absent',
          level             => $level,
          target            => '$(/usr/sbin/crm_node -n)',
          stonith_resources => [''],
          tries             => $tries,
          try_sleep         => $try_sleep,
        }
    }
  }

  $all_levels.each |$index, $levelx_devices |{

    $level = regsubst($index, 'level', '', 'G')
    $all_devices = $levelx_devices

    $xvm_devices = local_fence_devices('fence_xvm', $all_devices)
    create_resources('pacemaker::stonith::fence_xvm', $xvm_devices, $common_params)

    $ironic_devices = local_fence_devices('fence_ironic', $all_devices)
    create_resources('pacemaker::stonith::fence_ironic', $ironic_devices, $common_params)

    $redfish_devices = local_fence_devices('fence_redfish', $all_devices)
    create_resources('pacemaker::stonith::fence_redfish', $redfish_devices, $common_params)

    $ipmilan_devices = local_fence_devices('fence_ipmilan', $all_devices)
    create_resources('pacemaker::stonith::fence_ipmilan', $ipmilan_devices, $common_params)

    $kdump_devices = local_fence_devices('fence_kdump', $all_devices)
    create_resources('pacemaker::stonith::fence_kdump', $kdump_devices, $common_params)

    $kubevirt_devices = local_fence_devices('fence_kubevirt', $all_devices)
    create_resources('pacemaker::stonith::fence_kubevirt', $kubevirt_devices, $common_params)

    $rhev_devices = local_fence_devices('fence_rhevm', $all_devices)
    create_resources('pacemaker::stonith::fence_rhevm', $rhev_devices, $common_params)

    $ucs_devices = local_fence_devices('fence_cisco_ucs', $all_devices)
    create_resources('pacemaker::stonith::fence_cisco_ucs', $ucs_devices, $common_params)

    $data = {
      'xvm' => $xvm_devices, 'ironic' => $ironic_devices, 'redfish' => $redfish_devices,
      'ipmilan' => $ipmilan_devices, 'kdump' => $kdump_devices, 'kubevirt' => $kubevirt_devices,
      'rhevm' => $rhev_devices, 'cisco_ucs' => $ucs_devices
    }

    # let's store the number of stonith devices created for this server.
    # this will be used to detect if there is a least one and fail if
    # instance_ha is configured and puppet is running on a compute node.
    $data_num = [
      length($ironic_devices), length($redfish_devices),
      length($ipmilan_devices), length($kdump_devices), length($rhev_devices)
    ]

    $sum = $data_num.reduce |$memo, $value| { $memo + $value }

    $data.each |$items| {
      $driver = $items[0]
      $driver_devices = $items[1]

      # if there is no valid stonith device and this is a compute-instanceha node we raise an exception
      if $level == '1' and $sum == 0 and $enable_instanceha and $is_compute_instanceha_node {
        fail('Instance HA requires at least one valid stonith device')
      }

      if $driver_devices and length($driver_devices) == 1 {
        $mac = keys($driver_devices)[0]
        $safe_mac = regsubst($mac, ':', '', 'G')
        if ($enable_instanceha and $is_compute_instanceha_node) {
          $stonith_resources = [ "stonith-fence_${driver}-${safe_mac}", 'stonith-fence_compute-fence-nova' ]
        }
        else {
          $stonith_resources = [ "stonith-fence_${driver}-${safe_mac}" ]
        }
        pacemaker::stonith::level{ "stonith-${level}-${safe_mac}":
          level             => $level,
          target            => '$(/usr/sbin/crm_node -n)',
          stonith_resources => $stonith_resources,
          tries             => $tries,
          try_sleep         => $try_sleep,
        }
        Pcmk_stonith<||> -> Pcmk_stonith_level<||>
      }
    }
    # we use the boostrap_node to create the watchdog resource and the stonith
    # topology for all the nodes in the cluster, because the watchdog resource
    # is not per-node but cluster-wide
    $watchdog_devices = local_fence_devices('fence_watchdog', $all_devices)
    if length($watchdog_devices) > 0 {
      # check if this is the bootstrap node
      if downcase($::hostname) == lookup('pacemaker_short_bootstrap_node_name') {
        create_resources('pacemaker::stonith::fence_watchdog', $watchdog_devices, $common_params)
        $stonith_resources = [ 'watchdog' ]
        # if this is the boostrap node we set watchdog as levelX for all
        # the pacemaker nodes
        lookup('pacemaker_short_node_names').each |$node| {
          pacemaker::stonith::level{ "stonith-${level}-watchdog-${node}":
            level             => $level,
            target            => $node,
            stonith_resources => [ 'watchdog' ],
            tries             => $tries,
            try_sleep         => $try_sleep,
          }
        }
        pacemaker::property { 'stonith-watchdog-timeout':
          property => 'stonith-watchdog-timeout',
          value    => $watchdog_timeout,
          tries    => $tries,
        }
        Pcmk_property<||> -> Pcmk_stonith<||> -> Pcmk_stonith_level<||>
      }
    }
  }
}