From 11e3e486f1e83afc7c71b990701b04c07faddac1 Mon Sep 17 00:00:00 2001 From: Alexander Fisher Date: Mon, 18 Nov 2019 17:06:49 +0000 Subject: [PATCH 1/2] Add failing acceptance test for service restart In systemd, the service is currently not being restarted after configuration changes. See GH-382 The acceptance test in this commit reconfigures prometheus with the admin API enabled and then attempts to access it. See https://prometheus.io/docs/prometheus/latest/querying/api/#tsdb-admin-apis --- spec/acceptance/prometheus_server_spec.rb | 25 ++++++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/spec/acceptance/prometheus_server_spec.rb b/spec/acceptance/prometheus_server_spec.rb index 937982363..6c58f760d 100644 --- a/spec/acceptance/prometheus_server_spec.rb +++ b/spec/acceptance/prometheus_server_spec.rb @@ -2,7 +2,7 @@ describe 'prometheus server basics' do it 'prometheus server via main class works idempotently with no errors' do - pp = "class{'prometheus': manage_prometheus_server => true, version => '1.5.2' }" + pp = "class{'prometheus': manage_prometheus_server => true }" # Run it twice and test for idempotency apply_manifest(pp, catch_failures: true) @@ -32,6 +32,29 @@ it { is_expected.to be_listening.with('tcp6') } end + it 'does not allow admin API' do # rubocop:disable RSpec/MultipleExpectations + shell('curl -s -XPOST http://127.0.0.1:9090/api/v1/admin/tsdb/snapshot') do |r| + expect(r.stdout).to match(%r{admin APIs disabled}) + expect(r.exit_code).to eq(0) + end + end + + describe 'updating configuration to enable Admin API' do + it 'prometheus server via main class works idempotently with no errors' do + pp = "class{'prometheus': manage_prometheus_server => true, web_enable_admin_api => true }" + + apply_manifest(pp, catch_failures: true) + apply_manifest(pp, catch_changes: true) + end + it 'allows admin API' do # rubocop:disable RSpec/MultipleExpectations + shell('curl -s -XPOST http://127.0.0.1:9090/api/v1/admin/tsdb/snapshot') do |r| + expect(r.stdout).not_to match(%r{admin APIs disabled}) + expect(r.stdout).to match(%r{success}) + expect(r.exit_code).to eq(0) + end + end + end + describe 'prometheus server with options' do it 'is idempotent' do pp = "class{'prometheus::server': version => '2.4.3', external_url => '/test'}" From 9ef3e183804048e478c2b37187e0dd6fb4661f53 Mon Sep 17 00:00:00 2001 From: Alexander Fisher Date: Mon, 18 Nov 2019 17:14:22 +0000 Subject: [PATCH 2/2] Fix systemd unit file not notifying the service The [unreliable](https://puppet.com/docs/puppet/5.5/lang_defaults.html#behavior) resource default ``` File{ notify => Class['prometheus::run_service'] } ``` is replaced by a `$notify` variable that is set on the relevant file resources *and* `systemd::unit_file`. Some care was needed to make sure the reload behaviour wasn't broken. ie If the configuration change is just a new scrape job that is collected, the service should only be reloaded, not restarted. Fixes #382 --- manifests/config.pp | 51 ++++++++++++++++----------------- manifests/scrape_job.pp | 2 +- manifests/server.pp | 2 +- spec/classes/prometheus_spec.rb | 4 +-- 4 files changed, 28 insertions(+), 31 deletions(-) diff --git a/manifests/config.pp b/manifests/config.pp index 78043482d..2700d9d91 100644 --- a/manifests/config.pp +++ b/manifests/config.pp @@ -154,24 +154,22 @@ } $daemon_flags = $flags + $extra_options - # the vast majority of files here are init-files - # so any change there should trigger a full service restart - if $prometheus::server::restart_on_change { - File { - notify => Class['prometheus::run_service'], - } - $systemd_notify = [Exec['prometheus-systemd-reload'], Class['prometheus::run_service']] - } else { - $systemd_notify = Exec['prometheus-systemd-reload'] + # Service files (init-files/systemd unit files) need to trigger a full service restart + # prometheus.yaml and associated scrape file changes should only trigger a reload (and not use $notify) + $notify = $prometheus::server::restart_on_change ? { + true => Class['prometheus::run_service'], + default => undef, } case $prometheus::server::init_style { 'upstart' : { file { '/etc/init/prometheus.conf': + ensure => file, mode => '0444', owner => 'root', group => 'root', content => template('prometheus/prometheus.upstart.erb'), + notify => $notify, } file { '/etc/init.d/prometheus': ensure => link, @@ -179,44 +177,42 @@ owner => 'root', group => 'root', mode => '0755', + notify => $notify, } } 'systemd' : { - include 'systemd' systemd::unit_file {'prometheus.service': content => template('prometheus/prometheus.systemd.erb'), + notify => $notify, } - } - 'sysv', 'redhat' : { - file { '/etc/init.d/prometheus': - mode => '0555', - owner => 'root', - group => 'root', - content => template('prometheus/prometheus.sysv.erb'), + if versioncmp($facts['puppetversion'],'6.1.0') < 0 { + # Puppet 5 doesn't have https://tickets.puppetlabs.com/browse/PUP-3483 + # and camptocamp/systemd only creates this relationship when managing the service + Class['systemd::systemctl::daemon_reload'] -> Class['prometheus::run_service'] } } - 'debian' : { - file { '/etc/init.d/prometheus': - mode => '0555', - owner => 'root', - group => 'root', - content => template('prometheus/prometheus.debian.erb'), + 'sysv', 'redhat', 'debian', 'sles' : { + $content = $prometheus::server::init_style ? { + 'redhat' => template('prometheus/prometheus.sysv.erb'), # redhat and sysv share the same template file + default => template("prometheus/prometheus.${prometheus::server::init_style}.erb"), } - } - 'sles' : { file { '/etc/init.d/prometheus': + ensure => file, mode => '0555', owner => 'root', group => 'root', - content => template('prometheus/prometheus.sles.erb'), + content => $content, + notify => $notify, } } 'launchd' : { file { '/Library/LaunchDaemons/io.prometheus.daemon.plist': + ensure => file, mode => '0644', owner => 'root', group => 'wheel', content => template('prometheus/prometheus.launchd.erb'), + notify => $notify, } } default : { @@ -232,6 +228,7 @@ group => $prometheus::server::group, purge => true, recurse => true, + notify => Class['prometheus::service_reload'], # After purging, a reload is needed } $prometheus::server::collect_scrape_jobs.each |Hash $job_definition| { @@ -265,7 +262,7 @@ if $prometheus::server::manage_config { file { 'prometheus.yaml': - ensure => present, + ensure => file, path => "${prometheus::server::config_dir}/${prometheus::server::configname}", owner => 'root', group => $prometheus::server::group, diff --git a/manifests/scrape_job.pp b/manifests/scrape_job.pp index 15ce1fc27..c2a539931 100644 --- a/manifests/scrape_job.pp +++ b/manifests/scrape_job.pp @@ -31,7 +31,7 @@ }, ]) file { "${collect_dir}/${job_name}_${name}.yaml": - ensure => present, + ensure => file, owner => 'root', group => $prometheus::group, mode => $prometheus::config_mode, diff --git a/manifests/server.pp b/manifests/server.pp index 17942ee9f..53c5a3b16 100644 --- a/manifests/server.pp +++ b/manifests/server.pp @@ -94,6 +94,6 @@ Class['prometheus::install'] -> Class['prometheus::config'] - -> Class['prometheus::run_service'] + -> Class['prometheus::run_service'] # Note: config must *not* be configured here to notify run_service. Some resources in config.pp need to notify service_reload instead -> Class['prometheus::service_reload'] } diff --git a/spec/classes/prometheus_spec.rb b/spec/classes/prometheus_spec.rb index 9b9f5c5c6..249944230 100644 --- a/spec/classes/prometheus_spec.rb +++ b/spec/classes/prometheus_spec.rb @@ -185,7 +185,7 @@ it { is_expected.to contain_file('prometheus.yaml').with( - 'ensure' => 'present', + 'ensure' => 'file', 'path' => '/etc/prometheus/prometheus.yaml', 'owner' => 'root', 'group' => 'prometheus', @@ -296,7 +296,7 @@ } it { is_expected.to contain_file('prometheus.yaml').with( - 'ensure' => 'present', + 'ensure' => 'file', 'path' => '/etc/prometheus/prometheus.yaml', 'owner' => 'root', 'group' => 'prometheus',