Skip to content

Commit 935e006

Browse files
Merge pull request #18 from TalkingQuickly/feature/refactor
Feature/refactor
2 parents 99d8aa5 + 7fb6a2d commit 935e006

18 files changed

+110
-71
lines changed

CHANGELOG.md

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
# Changelog
2+
3+
## 5.0.1 (March 2021)
4+
5+
- Adds full support for deploy (but not config creation) without sudo access
6+
- Refactor config files to use single array of hashes with flags
7+
- Refactor Sidekiq and Monit configurations to copy files directly rather than using symlinks to avoid potential root access leak
8+
- Fixes bug where object identifier was outputted in logs rather than filename
9+
- Fixes nginx not being reloaded after `setup_config` due to shared log directory not yet existing
10+
11+
## 5.0.0 (March 2021)
12+
13+
- Full overhaul to support Rails 6 and Ubuntu 20.04

capistrano-cookbook.gemspec

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ Gem::Specification.new do |spec|
1919

2020
spec.add_dependency 'capistrano', '~> 3.16'
2121
spec.add_dependency 'capistrano3-puma', '~> 5.0.4'
22+
spec.add_dependency 'capistrano-sidekiq', '~> 2.0'
2223

2324
spec.add_development_dependency "bundler", "~> 1.5"
2425
spec.add_development_dependency "rake"

lib/capistrano/cookbook.rb

+2-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ module Cookbook
99
require 'capistrano/cookbook/run_tests'
1010
require 'capistrano/cookbook/setup_config'
1111
require 'capistrano/cookbook/create_database'
12-
require 'capistrano/cookbook/systemd'
12+
require 'capistrano/cookbook/puma_systemd'
13+
require 'capistrano/cookbook/sidekiq_systemd'
1314
end
1415
end

lib/capistrano/cookbook/helpers/setup_config_values.rb

+19-32
Original file line numberDiff line numberDiff line change
@@ -1,59 +1,46 @@
11
module Capistrano
22
module Cookbook
33
class SetupConfigValues
4-
def symlinks
5-
fetch(:symlinks) || symlinks_defaults
6-
end
7-
8-
def executable_config_files
9-
fetch(:executable_config_files) || executable_config_files_defaults
10-
end
11-
124
def config_files
135
fetch(:config_files) || config_files_defaults
146
end
157

168
private
179

18-
def symlinks_defaults
10+
def config_files_defaults
1911
base = [
2012
{
21-
source: "log_rotation",
22-
link: "/etc/logrotate.d/{{full_app_name}}"
13+
source: 'log_rotation',
14+
destination: "/etc/logrotate.d/#{fetch(:full_app_name)}",
15+
executable: false,
16+
as_root: true
17+
},
18+
{
19+
source: 'database.example.yml',
20+
destination: "#{shared_path}/config/database.example.yml",
21+
executable: false,
22+
as_root: false
2323
}
2424
]
25+
2526
return base unless sidekiq_enabled?
2627

2728
base + [
2829
{
29-
source: "sidekiq.service.capistrano",
30-
link: "/etc/systemd/system/#{fetch(:sidekiq_service_unit_name)}.service"
30+
source: 'sidekiq.service.capistrano',
31+
destination: "/home/#{fetch(:deploy_user)}/.config/systemd/user/#{fetch(:sidekiq_service_unit_name)}.service",
32+
executable: false,
33+
as_root: false
3134
},
3235
{
3336
source: "sidekiq_monit",
34-
link: "/etc/monit/conf.d/#{fetch(:full_app_name)}_sidekiq.conf"
37+
destination: "/etc/monit/conf.d/#{fetch(:full_app_name)}_sidekiq.conf",
38+
executable: false,
39+
as_root: true
3540
}
3641
]
3742
end
3843

39-
def executable_config_files_defaults
40-
%w(
41-
)
42-
end
43-
44-
def config_files_defaults
45-
base = %w(
46-
database.example.yml
47-
log_rotation
48-
)
49-
return base unless sidekiq_enabled?
50-
51-
base + %w(
52-
sidekiq.service.capistrano
53-
sidekiq_monit
54-
)
55-
end
56-
5744
def sidekiq_enabled?
5845
defined?(Capistrano::Sidekiq) == 'constant' && Capistrano::Sidekiq.class == Class
5946
end

lib/capistrano/cookbook/helpers/template.rb lib/capistrano/cookbook/helpers/smart_template.rb

+18-5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
require 'securerandom'
2+
require 'stringio'
3+
14
# will first try and copy the file:
25
# config/deploy/#{full_app_name}/#{from}.erb
36
# to:
@@ -11,13 +14,12 @@
1114
# ones to be over-ridden
1215
# if the target file name is the same as the source then
1316
# the second parameter can be left out
14-
def smart_template(from, to=nil)
15-
to ||= from
16-
full_to_path = "#{shared_path}/config/#{to}"
17+
def smart_template(from, to, as_root=false)
1718
if from_erb_path = template_file(from)
1819
from_erb = StringIO.new(ERB.new(File.read(from_erb_path)).result(binding))
19-
upload! from_erb, full_to_path
20-
info "copying: #{from_erb} to: #{full_to_path}"
20+
upload!(from_erb, to) unless as_root
21+
sudo_upload!(from_erb, to) if as_root
22+
info "copying: #{from} to: #{to}"
2123
else
2224
error "error #{from} not found"
2325
end
@@ -35,3 +37,14 @@ def template_file(name)
3537
end
3638
return nil
3739
end
40+
41+
def sudo_upload!(file_path, remote_path, mode: '644', owner: 'root:root')
42+
tmp_path = "/tmp/#{SecureRandom.uuid}"
43+
44+
upload!(file_path, tmp_path)
45+
46+
execute(:sudo, :mkdir, '-p', File.dirname(remote_path))
47+
execute(:sudo, :mv, '-f', tmp_path, remote_path)
48+
execute(:sudo, :chmod, mode, remote_path)
49+
execute(:sudo, :chown, owner, remote_path)
50+
end
+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
load File.expand_path("tasks/puma_systemd.cap", File.dirname(__FILE__))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
load File.expand_path("tasks/sidekiq_systemd.cap", File.dirname(__FILE__))

lib/capistrano/cookbook/systemd.rb

-1
This file was deleted.

lib/capistrano/cookbook/tasks/nginx.cap

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ namespace :nginx do
33
desc "#{task } Nginx"
44
task task_name do
55
on roles(:app), in: :sequence, wait: 5 do
6-
sudo "/etc/init.d/nginx #{task_name}"
6+
sudo "systemctl #{task_name} nginx"
77
end
88
end
99
end

lib/capistrano/cookbook/tasks/systemd.cap lib/capistrano/cookbook/tasks/puma_systemd.cap

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ namespace :puma do
66
if fetch(:puma_systemctl_user) == :system
77
sudo "#{fetch(:puma_systemctl_bin)} reload-or-restart #{fetch(:puma_service_unit_name)}"
88
else
9-
execute "#{fetch(:puma_systemctl_bin)}", "--user", "reload", fetch(:puma_service_unit_name)
109
execute :loginctl, "enable-linger", fetch(:puma_lingering_user) if fetch(:puma_enable_lingering)
10+
execute "#{fetch(:puma_systemctl_bin)}", "--user", "reload-or-restart", fetch(:puma_service_unit_name)
1111
end
1212
end
1313
end

lib/capistrano/cookbook/tasks/setup_config.cap

+20-8
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
require 'capistrano/dsl'
22
require 'capistrano/cookbook/helpers/setup_config_values'
33
require 'capistrano/cookbook/helpers/substitute_strings'
4-
require 'capistrano/cookbook/helpers/template'
4+
require 'capistrano/cookbook/helpers/smart_template'
55
require 'capistrano/cookbook/nginx'
66
require 'capistrano/cookbook/monit'
77
require 'securerandom'
@@ -13,22 +13,24 @@ namespace :deploy do
1313
on roles(:app) do
1414
# make the config dir
1515
execute :mkdir, "-p #{shared_path}/config"
16+
execute :mkdir, "-p /home/#{fetch(:deploy_user)}/.config/systemd/user"
1617

1718
# config files to be uploaded to shared/config, see the
1819
# definition of smart_template for details of operation.
1920
conf.config_files.each do |file|
20-
smart_template file
21+
smart_template(file[:source], file[:destination], file[:as_root])
22+
execute(:chmod, "+x #{file[:destination]}") if file[:executable]
2123
end
2224

2325
# which of the above files should be marked as executable
24-
conf.executable_config_files.each do |file|
25-
execute :chmod, "+x #{shared_path}/config/#{file}"
26-
end
26+
# conf.executable_config_files.each do |file|
27+
# execute :chmod, "+x #{shared_path}/config/#{file}"
28+
# end
2729

2830
# symlink stuff which should be... symlinked
29-
conf.symlinks.each do |symlink|
30-
sudo "ln -nfs #{shared_path}/config/#{symlink[:source]} #{sub_strings(symlink[:link])}"
31-
end
31+
# conf.symlinks.each do |symlink|
32+
# sudo "ln -nfs #{shared_path}/config/#{symlink[:source]} #{sub_strings(symlink[:link])}"
33+
# end
3234

3335
if File.exists?(File.join('config', 'master.key'))
3436
upload! File.join('config', 'master.key'), File.join(shared_path, 'config', 'master.key')
@@ -41,13 +43,23 @@ end
4143
# remove the default nginx configuration as it will tend to conflict with our configs
4244
before 'deploy:setup_config', 'nginx:remove_default_vhost'
4345

46+
# make sure that shared directories etc exist before running otherwise the
47+
# initial nginx reload won't work because of the nginx log file directory path
48+
# not existing
49+
before 'deploy:setup_config', 'deploy:check:directories'
50+
before 'deploy:setup_config', 'deploy:check:linked_dirs'
51+
4452
# After setup config has generated and setup initial files, run the Capistrano Puma
4553
# tasks responsible for uploading config files. Note that `setup_config` creates overrides
4654
# for these in `config/deploy/templates` so we're not using the default ones from the gem
4755
after 'deploy:setup_config', 'puma:config'
4856
after 'deploy:setup_config', 'puma:nginx_config'
4957
after 'deploy:setup_config', 'puma:monit:config'
5058
after 'deploy:setup_config', 'puma:systemd:config'
59+
after 'deploy:setup_config', 'puma:systemd:enable'
60+
61+
# Enable the sidekiq systemd service so that it's started automatically on (re)boot
62+
after 'deploy:setup_config', 'sidekiq:systemd:enable' if (defined?(Capistrano::Sidekiq) == 'constant' && Capistrano::Sidekiq.class == Class)
5163

5264
# reload nginx to it will pick up any modified vhosts from setup_config
5365
after 'deploy:setup_config', 'nginx:reload'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
namespace :sidekiq do
2+
namespace :systemd do
3+
desc 'Install systemd sidekiq service'
4+
task :enable do
5+
on roles fetch(:sidekiq_roles) do |role|
6+
if fetch(:sidekiq_service_unit_user) == :system
7+
execute :sudo, :systemctl, "enable", fetch(:sidekiq_service_unit_name)
8+
else
9+
execute :systemctl, "--user", "enable", fetch(:sidekiq_service_unit_name)
10+
execute :loginctl, "enable-linger", fetch(:sidekiq_systemctl_user) if fetch(:sidekiq_enable_lingering)
11+
end
12+
end
13+
end
14+
end
15+
end

lib/capistrano/cookbook/version.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
module Capistrano
22
module Cookbook
3-
VERSION = "5.0.0"
3+
VERSION = "5.0.1"
44
end
55
end

lib/generators/capistrano/reliably_deploying_rails/templates/deploy.rb.erb

+8-9
Original file line numberDiff line numberDiff line change
@@ -13,20 +13,19 @@ set :rbenv_ruby, '3.0.0'
1313
set :rbenv_prefix, "RBENV_ROOT=#{fetch(:rbenv_path)} RBENV_VERSION=#{fetch(:rbenv_ruby)} #{fetch(:rbenv_path)}/bin/rbenv exec"
1414
set :rbenv_map_bins, %w{rake gem bundle ruby rails}
1515

16-
<% if @generate_sidekiq %>
17-
# Setup sidekiq, make sure we run sidekiq as our deployment user, otherwise
18-
# it will default to root which a) is insecture and b) will lead to lots of
19-
# strange permissions issues
20-
set :sidekiq_service_unit_user, :system
21-
set :sidekiq_user, fetch(:deploy_user)
22-
<% end %>
23-
2416
# setup puma to operate in clustered mode, required for zero downtime deploys
2517
set :puma_preload_app, false
2618
set :puma_init_active_record, true
2719
set :puma_workers, 3
20+
set :puma_systemctl_user, fetch(:deploy_user)
21+
set :puma_enable_lingering, true
22+
23+
<% if @generate_sidekiq %>
24+
set :sidekiq_systemctl_user, fetch(:deploy_user)
25+
set :sidekiq_enable_lingering, true
26+
<% end %>
2827

29-
# how many old releases do we want to keep, not much
28+
# how many old releases do we want to keep
3029
set :keep_releases, 5
3130

3231
# Directories that should be linked to the shared folder

lib/generators/capistrano/reliably_deploying_rails/templates/production.rb.erb

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ set :full_app_name, "#{fetch(:application)}_#{fetch(:stage)}"
1414
<% if @generate_sidekiq %>
1515
# Name sidekiq systemd service after the app and stage name so that
1616
# multiple apps and stages can co-exist on the same machine if needed
17-
set :sidekiq_service_unit_name, "#{fetch(:full_app_name)}_sidekiq"
17+
set :sidekiq_service_unit_name, "sidekiq_#{fetch(:full_app_name)}"
1818
<% end %>
1919

2020
server '<%= @production_server_address %>', user: 'deploy', roles: %w{web app db}, primary: true
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,4 @@
1-
# Monit configuration for Puma
2-
# Service name: <%= puma_monit_service_name %>
3-
#
41
check process <%= puma_monit_service_name %>
52
with pidfile "<%= fetch(:puma_pid) %>"
6-
start program = "/usr/bin/systemctl start <%= fetch(:puma_service_unit_name) %>"
7-
stop program = "/usr/bin/systemctl stop <%= fetch(:puma_service_unit_name) %>"
3+
start program = "/bin/bash -c 'XDG_RUNTIME_DIR=/run/user/$(id -u) /usr/bin/systemctl start --user <%= fetch(:puma_service_unit_name) %>'" as uid "<%= fetch(:puma_systemctl_user) %>" and gid "<%= fetch(:puma_systemctl_user) %>"
4+
stop program = "/bin/bash -c 'XDG_RUNTIME_DIR=/run/user/$(id -u) /usr/bin/systemctl stop --user <%= fetch(:puma_service_unit_name) %>'" as uid "<%= fetch(:puma_systemctl_user) %>" and gid "<%= fetch(:puma_systemctl_user) %>"

lib/generators/capistrano/reliably_deploying_rails/templates/sidekiq.service.capistrano.erb

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
<% # Adapted from: https://github.com/seuros/capistrano-sidekiq/blob/master/lib/generators/capistrano/sidekiq/systemd/templates/sidekiq.service.capistrano.erb %>
2-
31
[Unit]
42
Description=sidekiq for <%= "#{fetch(:application)} (#{fetch(:stage)})" %>
53
After=syslog.target network.target
@@ -12,7 +10,7 @@ ExecReload=/bin/kill -TSTP $MAINPID
1210
ExecStop=/bin/kill -TERM $MAINPID
1311
<%="StandardOutput=append:#{fetch(:sidekiq_log)}" if fetch(:sidekiq_log) %>
1412
<%="StandardError=append:#{fetch(:sidekiq_error_log)}" if fetch(:sidekiq_error_log) %>
15-
<%="User=#{fetch(:sidekiq_user)}" if fetch(:sidekiq_user) %>
13+
<%="User=#{fetch(:sidekiq_user)}" if (fetch(:sidekiq_user) && (fetch(:puma_systemctl_user) == :system)) %>
1614
<%="EnvironmentFile=#{fetch(:sidekiq_service_unit_env_file)}" if fetch(:sidekiq_service_unit_env_file) %>
1715
<% fetch(:sidekiq_service_unit_env_vars, []).each do |environment_variable| %>
1816
<%="Environment=#{environment_variable}" %>
@@ -27,4 +25,6 @@ Restart=on-failure
2725
SyslogIdentifier=sidekiq_<%= fetch(:application) %>_<%= fetch(:stage) %>
2826

2927
[Install]
30-
WantedBy=default.target
28+
WantedBy=<%=(fetch(:sidekiq_systemctl_user) == :system) ? "multi-user.target" : "default.target"%>
29+
30+
<% # Adapted from: https://github.com/seuros/capistrano-sidekiq/blob/master/lib/generators/capistrano/sidekiq/systemd/templates/sidekiq.service.capistrano.erb %>
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
check process <%= fetch(:sidekiq_service_unit_name) %> matching "sidekiq.*<%= fetch(:full_app_name) %>"
2-
start program = "/usr/bin/systemctl start <%= fetch(:sidekiq_service_unit_name) %>"
3-
stop program = "/usr/bin/systemctl stop <%= fetch(:sidekiq_service_unit_name) %>"
2+
start program = "/bin/bash -c 'XDG_RUNTIME_DIR=/run/user/$(id -u) /usr/bin/systemctl start --user <%= fetch(:sidekiq_service_unit_name) %>'" as uid "<%= fetch(:sidekiq_systemctl_user) %>" and gid "<%= fetch(:sidekiq_systemctl_user) %>"
3+
stop program = "/bin/bash -c 'XDG_RUNTIME_DIR=/run/user/$(id -u) /usr/bin/systemctl stop --user <%= fetch(:sidekiq_service_unit_name) %>'" as uid "<%= fetch(:sidekiq_systemctl_user) %>" and gid "<%= fetch(:sidekiq_systemctl_user) %>"
44
group <%= fetch(:sidekiq_monit_group) || fetch(:full_app_name) %>-sidekiq

0 commit comments

Comments
 (0)