Skip to content

Commit

Permalink
Improve cs_clone
Browse files Browse the repository at this point in the history
- Stop cs_clone before removing (crm provider)
- Use resource name as clone id (pcs provider) - closes voxpupuli#149
- Add acceptance tests and improve unit tests

Signed-off-by: Julien Pivotto <[email protected]>
  • Loading branch information
roidelapluie committed Sep 15, 2016
1 parent 16b15f6 commit 0f20634
Show file tree
Hide file tree
Showing 8 changed files with 197 additions and 137 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@
- Removed legacy configuration sections: amf, aisexec, logging.logger\_subsys
(#345)
- Fix two\_nodes behaviour with expected\_votes = 2 introduced in 3.0.0 (#246)
- Cs\_clones are now stopped before being removed (crm provider) (#367)
- Cs\_clones now use the resource name as clone id (pcs provider) (#149, #367)

### Deprecation notes

Expand Down
69 changes: 4 additions & 65 deletions lib/puppet/provider/cs_clone/crm.rb
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@
commands crm: 'crm'
commands crm_attribute: 'crm_attribute'

mk_resource_methods

def self.instances
block_until_ready

Expand Down Expand Up @@ -66,76 +68,13 @@ def create
# Unlike create we actually immediately delete the item.
def destroy
debug('Removing clone')
cmd = [command(:crm), '-w', 'resource', 'stop', @resource[:name]]
PuppetX::Voxpupuli::Corosync::Provider::Crmsh.run_command_in_cib(cmd, @resource[:cib], false)
cmd = [command(:crm), 'configure', 'delete', @resource[:name]]
PuppetX::Voxpupuli::Corosync::Provider::Crmsh.run_command_in_cib(cmd, @resource[:cib])
@property_hash.clear
end

#
# Getter that obtains the our service that should have been populated by
# prefetch or instances (depends on if your using puppet resource or not).
def primitive
@property_hash[:primitive]
end

# Getter that obtains the our clone_max that should have been populated by
# prefetch or instances (depends on if your using puppet resource or not).
def clone_max
@property_hash[:clone_max]
end

def clone_node_max
@property_hash[:clone_node_max]
end

def notify_clones
@property_hash[:notify_clones]
end

def globally_unique
@property_hash[:globally_unique]
end

def ordered
@property_hash[:ordered]
end

def interleave
@property_hash[:interleave]
end

# Our setters. Setters are used when the
# resource already exists so we just update the current value in the property
# hash and doing this marks it to be flushed.

def primitive=(should)
@property_hash[:primitive] = should
end

def clone_max=(should)
@property_hash[:clone_max] = should
end

def clone_node_max=(should)
@property_hash[:clone_node_max] = should
end

def notify_clones=(should)
@property_hash[:notify_clones] = should
end

def globally_unique=(should)
@property_hash[:globally_unique] = should
end

def ordered=(should)
@property_hash[:ordered] = should
end

def interleave=(should)
@property_hash[:interleave] = should
end

# Flush is triggered on anything that has been detected as being
# modified in the property_hash. It generates a temporary file with
# the updates that need to be made. The temporary file is then used
Expand Down
83 changes: 17 additions & 66 deletions lib/puppet/provider/cs_clone/pcs.rb
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,24 @@
desc 'Provider to add, delete, manipulate primitive clones.'

commands pcs: 'pcs'
commands cibadmin: 'cibadmin'

mk_resource_methods

defaultfor operatingsystem: [:fedora, :centos, :redhat]

def change_clone_id(primitive, id, cib)
xpath = "/cib/configuration/resources/clone[descendant::primitive[@id='#{primitive}']]"
cmd = [command(:cibadmin), '--query', '--xpath', xpath]
raw, = PuppetX::Voxpupuli::Corosync::Provider::Pcs.run_command_in_cib(cmd, cib)
doc = REXML::Document.new(raw)
if doc.root.attributes['id'] != id
doc.root.attributes['id'] = id
cmd = [command(:cibadmin), '--replace', '--xpath', xpath, '--xml-text', doc.to_s.chop]
PuppetX::Voxpupuli::Corosync::Provider::Pcs.run_command_in_cib(cmd, cib)
end
end

def self.instances
block_until_ready

Expand Down Expand Up @@ -48,7 +63,7 @@ def self.instances
# of actually doing the work.
def create
@property_hash = {
name: @resource[:primitive] + '-clone',
name: @resource[:name],
ensure: :present,
primitive: @resource[:primitive],
clone_max: @resource[:clone_max],
Expand All @@ -68,71 +83,6 @@ def destroy
@property_hash.clear
end

#
# Getter that obtains the our service that should have been populated by
# prefetch or instances (depends on if your using puppet resource or not).
def primitive
@property_hash[:primitive]
end

# Getter that obtains the our clone_max that should have been populated by
# prefetch or instances (depends on if your using puppet resource or not).
def clone_max
@property_hash[:clone_max]
end

def clone_node_max
@property_hash[:clone_node_max]
end

def notify_clones
@property_hash[:notify_clones]
end

def globally_unique
@property_hash[:globally_unique]
end

def ordered
@property_hash[:ordered]
end

def interleave
@property_hash[:interleave]
end

# Our setters. Setters are used when the
# resource already exists so we just update the current value in the property
# hash and doing this marks it to be flushed.

def primitive=(should)
@property_hash[:primitive] = should
end

def clone_max=(should)
@property_hash[:clone_max] = should
end

def clone_node_max=(should)
@property_hash[:clone_node_max] = should
end

def notify_clones=(should)
@property_hash[:notify_clones] = should
end

def globally_unique=(should)
@property_hash[:globally_unique] = should
end

def ordered=(should)
@property_hash[:ordered] = should
end

def interleave=(should)
@property_hash[:interleave] = should
end

def exists?
@property_hash[:existing_resource] == :true
end
Expand Down Expand Up @@ -160,6 +110,7 @@ def flush
cmd << "ordered=#{@property_hash[:ordered]}" if @property_hash[:ordered]
cmd << "interleave=#{@property_hash[:interleave]}" if @property_hash[:interleave]
PuppetX::Voxpupuli::Corosync::Provider::Pcs.run_command_in_cib(cmd, @resource[:cib])
change_clone_id(@property_hash[:primitive], @property_hash[:name], @resource[:cib])
end
end
end
7 changes: 6 additions & 1 deletion lib/puppet/type/cs_clone.rb
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
ensurable

newparam(:name) do
desc "Identifier of the location entry. This value needs to be unique
desc "Identifier of the clone entry. This value needs to be unique
across the entire Corosync/Pacemaker configuration since it doesn't have
the concept of name spaces per type."

Expand Down Expand Up @@ -90,4 +90,9 @@ def unmunge_cs_primitive(name)

name
end

validate do
return if self[:ensure] == :absent
raise Puppet::Error, 'primitive is mandatory' unless self[:primitive]
end
end
90 changes: 90 additions & 0 deletions spec/acceptance/cs_clone_spec.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
#! /usr/bin/env ruby -S rspec
require 'spec_helper_acceptance'

describe 'corosync' do
cert = '-----BEGIN CERTIFICATE-----
MIIDVzCCAj+gAwIBAgIJAJNCo5ZPmKegMA0GCSqGSIb3DQEBBQUAMEIxCzAJBgNV
BAYTAlhYMRUwEwYDVQQHDAxEZWZhdWx0IENpdHkxHDAaBgNVBAoME0RlZmF1bHQg
Q29tcGFueSBMdGQwHhcNMTUwMjI2MjI1MjU5WhcNMTUwMzI4MjI1MjU5WjBCMQsw
CQYDVQQGEwJYWDEVMBMGA1UEBwwMRGVmYXVsdCBDaXR5MRwwGgYDVQQKDBNEZWZh
dWx0IENvbXBhbnkgTHRkMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA
uCPPbDgErGUVs1pKqv59OatjCEU4P9QcmhDYFR7RBN8m08mIqd+RTuiHUKj6C9Rk
vWQ5bYrGQo/+4E0ziAUuUzzITlpIYLVltca6eBhKUqO3Cd0NMRVc2k4nx5948nwv
9FVOIfOOY6BN2ALglfBfLnhObbzJjs6OSZ7bUCpXVPV01t/61Jj3jQ3+R8b7AaoR
mw7j0uWaFimKt/uag1qqKGw3ilieMhHlG0Da5x9WLi+5VIM0t1rcpR58LLXVvXZB
CrQBucm2xhZsz7R76Ai+NL8zhhyzCZidZ2NtJ3E1wzppcSDAfNrru+rcFSlZ4YG+
lMCqZ1aqKWVXmb8+Vg7IkQIDAQABo1AwTjAdBgNVHQ4EFgQULxI68KhZwEF5Q9al
xZmFDR+Beu4wHwYDVR0jBBgwFoAULxI68KhZwEF5Q9alxZmFDR+Beu4wDAYDVR0T
BAUwAwEB/zANBgkqhkiG9w0BAQUFAAOCAQEAsa0YKPixD6VmDo3pal2qqichHbdT
hUONk2ozzRoaibVocqKx2T6Ho23wb/lDlRUu4K4DMO663uumzI9lNoOewa0MuW1D
J52cejAMVsP3ROOdxBv0HZIVVJ8NLBHNLFOHJEDtvzogLVplzmo59vPAdmQo6eIV
japvs+0tdy9iwHj3z1ZME2Ntm/5TzG537e7Hb2zogatM9aBTUAWlZ1tpoaXuTH52
J76GtqoIOh+CTeY/BMwBotdQdgeR0zvjE9FuLWkhTmRtVFhbVIzJbFlFuYq5d3LH
NWyN0RsTXFaqowV1/HSyvfD7LoF/CrmN5gOAM3Ierv/Ti9uqGVhdGBd/kw=='
File.open('/tmp/ca.pem', 'w') { |f| f.write(cert) }
it 'with defaults' do
pp = <<-EOS
file { '/tmp/ca.pem':
ensure => file,
content => '#{cert}'
} ->
class { 'corosync':
multicast_address => '224.0.0.1',
authkey => '/tmp/ca.pem',
bind_address => '127.0.0.1',
set_votequorum => true,
quorum_members => ['127.0.0.1'],
}
cs_property { 'stonith-enabled' :
value => 'false',
} ->
cs_primitive { 'duncan_vip':
primitive_class => 'ocf',
primitive_type => 'IPaddr2',
provided_by => 'heartbeat',
parameters => { 'ip' => '172.16.210.101', 'cidr_netmask' => '24' },
operations => { 'monitor' => { 'interval' => '10s' } },
}
EOS

apply_manifest(pp, catch_failures: true, debug: true, trace: true)
apply_manifest(pp, catch_changes: true, debug: false, trace: true)
end

describe service('corosync') do
it { is_expected.to be_running }
end

it 'creates a clone' do
pp = <<-EOS
cs_clone { 'duncan_vip_clone':
ensure => present,
primitive => 'duncan_vip',
}
EOS
apply_manifest(pp, catch_failures: true, debug: true, trace: true)
apply_manifest(pp, catch_changes: true, debug: false, trace: true)
command = 'cibadmin --query | grep duncan_vip_clone'
shell(command) do |r|
expect(r.stdout).to match(%r{<clone})
end
end

it 'deletes a clone' do
pp = <<-EOS
cs_clone { 'duncan_vip_clone':
ensure => absent,
}
EOS
apply_manifest(pp, catch_failures: true, debug: true, trace: true)
apply_manifest(pp, catch_changes: true, debug: false, trace: true)
command = 'cibadmin --query | grep duncan_vip_clone'
assert_raises(Beaker::Host::CommandFailure) do
shell(command)
end
end

after :all do
cleanup_cs_resources
end
end
1 change: 1 addition & 0 deletions spec/spec_helper_corosync.rb
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ def not_expect_commands(patterns)
shared_context 'pcs' do
before do
described_class.stubs(:command).with(:pcs).returns 'pcs'
described_class.stubs(:command).with(:cibadmin).returns 'cibadmin'
described_class.expects(:block_until_ready).returns(nil).at_most(1)
end
end
Expand Down
Loading

0 comments on commit 0f20634

Please sign in to comment.