Compare commits
210 Commits
v1.7.1
...
proxy-0.5.
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
acd4b85044 | ||
|
|
e71bfcbadd | ||
|
|
567309596a | ||
|
|
b89ec2bf63 | ||
|
|
3172adca30 | ||
|
|
04d21f45bb | ||
|
|
eabd57350c | ||
|
|
487f6f5f53 | ||
|
|
d98500982d | ||
|
|
8693e968c1 | ||
|
|
6ab5fc9459 | ||
|
|
6fc2915884 | ||
|
|
afa6898a82 | ||
|
|
384b36d158 | ||
|
|
6df169a4fb | ||
|
|
ab109afc52 | ||
|
|
a6a48c456c | ||
|
|
a4e5dbe5d4 | ||
|
|
56e90906b1 | ||
|
|
6e65968bdc | ||
|
|
85f1e14b97 | ||
|
|
2c829a4824 | ||
|
|
45a58f7e15 | ||
|
|
834b343ded | ||
|
|
9fe1821cae | ||
|
|
1d7c9fec1d | ||
|
|
a6b983de06 | ||
|
|
3ec4ad2ea5 | ||
|
|
63f854ea18 | ||
|
|
fd0cdc1ca1 | ||
|
|
d218264b69 | ||
|
|
684f7ac148 | ||
|
|
8bcd896242 | ||
|
|
600bbd77ef | ||
|
|
34effef70a | ||
|
|
e07ac070aa | ||
|
|
46c0836cd4 | ||
|
|
bd54c74682 | ||
|
|
f183419f7a | ||
|
|
190dbd1ea3 | ||
|
|
d6eda3d741 | ||
|
|
0fe6a17a91 | ||
|
|
7f15fd143f | ||
|
|
434490bd0c | ||
|
|
267b526438 | ||
|
|
1f721739d6 | ||
|
|
6c51e596ae | ||
|
|
7f31510aec | ||
|
|
e8ff233e81 | ||
|
|
a316e51eda | ||
|
|
bf91d6c1ca | ||
|
|
a84ee6315f | ||
|
|
3c39086613 | ||
|
|
8b965b0a31 | ||
|
|
d2672c771e | ||
|
|
24031fefb0 | ||
|
|
35fe9c154d | ||
|
|
b8972a6833 | ||
|
|
d7d6fa34b0 | ||
|
|
c21757f747 | ||
|
|
cb73c730f9 | ||
|
|
109339189a | ||
|
|
33834a266a | ||
|
|
e1016b2469 | ||
|
|
a40b644145 | ||
|
|
ccb7424197 | ||
|
|
2125327d54 | ||
|
|
f4d309c5cc | ||
|
|
5bca8015bc | ||
|
|
27a7b339a6 | ||
|
|
dcd4778dd9 | ||
|
|
6f2eaed398 | ||
|
|
e9d480b514 | ||
|
|
2fdc59a3aa | ||
|
|
b33c999125 | ||
|
|
2056351c38 | ||
|
|
9c2d5f83f7 | ||
|
|
f347ef7e44 | ||
|
|
63ebeda489 | ||
|
|
13bdf50ceb | ||
|
|
bd6558630f | ||
|
|
53903ddcd2 | ||
|
|
55756fa6f3 | ||
|
|
fe0c656de5 | ||
|
|
418d8045d8 | ||
|
|
d63ff8f251 | ||
|
|
eab717e0cf | ||
|
|
66d5e25834 | ||
|
|
6bbbd81da1 | ||
|
|
876eebc7c5 | ||
|
|
dc1bbac3c8 | ||
|
|
045aa7d167 | ||
|
|
0660895e75 | ||
|
|
debdf00cca | ||
|
|
9089c41f30 | ||
|
|
c9946808b1 | ||
|
|
deb2a6d298 | ||
|
|
0cb69a84f5 | ||
|
|
aa630f156a | ||
|
|
63d0b5ddfa | ||
|
|
06f4caa866 | ||
|
|
5aa3d1aeb0 | ||
|
|
a4d668cd39 | ||
|
|
7156c80f34 | ||
|
|
aed2ef99d0 | ||
|
|
57cbf7cdb5 | ||
|
|
b99c044327 | ||
|
|
8ad6a0ed16 | ||
|
|
8b62e2694a | ||
|
|
be1df4356a | ||
|
|
8210e8e768 | ||
|
|
9b96ef2412 | ||
|
|
1522d94ac9 | ||
|
|
a68294c384 | ||
|
|
31a347c285 | ||
|
|
3d502ab12d | ||
|
|
5226d52f8a | ||
|
|
9deb8af4a0 | ||
|
|
068aaa0bd0 | ||
|
|
a726a86a17 | ||
|
|
b2e1a4d4c1 | ||
|
|
9ade79fc84 | ||
|
|
79731da619 | ||
|
|
0ae8046905 | ||
|
|
d5ecca0fd4 | ||
|
|
0c6a593554 | ||
|
|
3f37fea7c3 | ||
|
|
7daaabd4d4 | ||
|
|
fcdef5fa06 | ||
|
|
5480b40ba3 | ||
|
|
1d0e81b00a | ||
|
|
5910249d02 | ||
|
|
b464c4fd4a | ||
|
|
56754fe40c | ||
|
|
6a06efc9d9 | ||
|
|
5c4c33e0a8 | ||
|
|
0b5506f6f2 | ||
|
|
a2549b1f60 | ||
|
|
9b9e60ec7f | ||
|
|
e557eea79c | ||
|
|
d7e785cd36 | ||
|
|
5cda3086c4 | ||
|
|
362f5d00f6 | ||
|
|
6adf3c117f | ||
|
|
9f0b10425c | ||
|
|
5f2384f123 | ||
|
|
eab7d3adc5 | ||
|
|
d2d0223c37 | ||
|
|
56268d724d | ||
|
|
cffb6c3d7e | ||
|
|
bd1726f305 | ||
|
|
7ddb122a22 | ||
|
|
98c951bbdb | ||
|
|
374c117b79 | ||
|
|
d6a5cf3c78 | ||
|
|
2aeabda455 | ||
|
|
c048c097ed | ||
|
|
ed148628fb | ||
|
|
d48080c772 | ||
|
|
3f64338929 | ||
|
|
0ab838bc25 | ||
|
|
b7382ceeaf | ||
|
|
69367fbc6b | ||
|
|
2515bd705c | ||
|
|
579e169be2 | ||
|
|
d6f5da92be | ||
|
|
9ccfe20b10 | ||
|
|
e871d347d5 | ||
|
|
b8af719bb7 | ||
|
|
f48987aa03 | ||
|
|
ef051eca1b | ||
|
|
173d44ee0a | ||
|
|
4e811372f8 | ||
|
|
ec4aa45852 | ||
|
|
5e11a64181 | ||
|
|
57d9ce177a | ||
|
|
b12de87388 | ||
|
|
8a98949634 | ||
|
|
0eb9f48082 | ||
|
|
9db6fc0704 | ||
|
|
27fede3caa | ||
|
|
29c723f7ec | ||
|
|
2755582c47 | ||
|
|
fa73d722ea | ||
|
|
c535e4e44f | ||
|
|
0ea07b1760 | ||
|
|
03b531f179 | ||
|
|
d8570d1c2c | ||
|
|
3fe70b458d | ||
|
|
ade8b43599 | ||
|
|
d24fc3ca4e | ||
|
|
7c244bbb98 | ||
|
|
1369c46a83 | ||
|
|
deccf1cfaf | ||
|
|
1573cebadf | ||
|
|
85a2926cde | ||
|
|
58a51b079e | ||
|
|
f1f3fc566f | ||
|
|
44726ff65a | ||
|
|
fd0d4af21f | ||
|
|
13409ada5a | ||
|
|
9a1379be6c | ||
|
|
31d6c198da | ||
|
|
22afe4de77 | ||
|
|
b63982c3a7 | ||
|
|
9e12d32cc3 | ||
|
|
ff03891d47 | ||
|
|
f21dc30875 | ||
|
|
69fa7286e2 | ||
|
|
e160852e4d |
13
.github/workflows/ci.yml
vendored
13
.github/workflows/ci.yml
vendored
@@ -24,25 +24,12 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
ruby-version:
|
ruby-version:
|
||||||
- "2.7"
|
|
||||||
- "3.1"
|
- "3.1"
|
||||||
- "3.2"
|
- "3.2"
|
||||||
- "3.3"
|
- "3.3"
|
||||||
gemfile:
|
gemfile:
|
||||||
- Gemfile
|
- Gemfile
|
||||||
- gemfiles/ruby_2.7.gemfile
|
|
||||||
- gemfiles/rails_edge.gemfile
|
- gemfiles/rails_edge.gemfile
|
||||||
exclude:
|
|
||||||
- ruby-version: "2.7"
|
|
||||||
gemfile: Gemfile
|
|
||||||
- ruby-version: "2.7"
|
|
||||||
gemfile: gemfiles/rails_edge.gemfile
|
|
||||||
- ruby-version: "3.1"
|
|
||||||
gemfile: gemfiles/ruby_2.7.gemfile
|
|
||||||
- ruby-version: "3.2"
|
|
||||||
gemfile: gemfiles/ruby_2.7.gemfile
|
|
||||||
- ruby-version: "3.3"
|
|
||||||
gemfile: gemfiles/ruby_2.7.gemfile
|
|
||||||
name: ${{ format('Tests (Ruby {0})', matrix.ruby-version) }}
|
name: ${{ format('Tests (Ruby {0})', matrix.ruby-version) }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
continue-on-error: true
|
continue-on-error: true
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# Use the official Ruby 3.2.0 Alpine image as the base image
|
# Use the official Ruby 3.2.0 Alpine image as the base image
|
||||||
FROM ruby:3.2.0-alpine
|
FROM ruby:3.2.0-alpine
|
||||||
|
|
||||||
# Install docker/buildx-bin
|
# Install docker/buildx-bin
|
||||||
COPY --from=docker/buildx-bin /buildx /usr/libexec/docker/cli-plugins/docker-buildx
|
COPY --from=docker/buildx-bin /buildx /usr/libexec/docker/cli-plugins/docker-buildx
|
||||||
|
|
||||||
# Set the working directory to /kamal
|
# Set the working directory to /kamal
|
||||||
@@ -14,7 +14,7 @@ COPY Gemfile Gemfile.lock kamal.gemspec ./
|
|||||||
COPY lib/kamal/version.rb /kamal/lib/kamal/version.rb
|
COPY lib/kamal/version.rb /kamal/lib/kamal/version.rb
|
||||||
|
|
||||||
# Install system dependencies
|
# Install system dependencies
|
||||||
RUN apk add --no-cache --update build-base git docker openrc openssh-client-default \
|
RUN apk add --no-cache build-base git docker openrc openssh-client-default \
|
||||||
&& rc-update add docker boot \
|
&& rc-update add docker boot \
|
||||||
&& gem install bundler --version=2.4.3 \
|
&& gem install bundler --version=2.4.3 \
|
||||||
&& bundle install
|
&& bundle install
|
||||||
|
|||||||
116
Gemfile.lock
116
Gemfile.lock
@@ -1,25 +1,24 @@
|
|||||||
PATH
|
PATH
|
||||||
remote: .
|
remote: .
|
||||||
specs:
|
specs:
|
||||||
kamal (1.7.1)
|
kamal (2.0.0.rc2)
|
||||||
activesupport (>= 7.0)
|
activesupport (>= 7.0)
|
||||||
base64 (~> 0.2)
|
base64 (~> 0.2)
|
||||||
bcrypt_pbkdf (~> 1.0)
|
bcrypt_pbkdf (~> 1.0)
|
||||||
concurrent-ruby (~> 1.2)
|
concurrent-ruby (~> 1.2)
|
||||||
dotenv (~> 2.8)
|
dotenv (~> 3.1)
|
||||||
ed25519 (~> 1.2)
|
ed25519 (~> 1.2)
|
||||||
net-ssh (~> 7.0)
|
net-ssh (~> 7.0)
|
||||||
sshkit (>= 1.22.2, < 2.0)
|
sshkit (>= 1.23.0, < 2.0)
|
||||||
thor (~> 1.2)
|
thor (~> 1.3)
|
||||||
x25519 (~> 1.0, >= 1.0.10)
|
|
||||||
zeitwerk (~> 2.5)
|
zeitwerk (~> 2.5)
|
||||||
|
|
||||||
GEM
|
GEM
|
||||||
remote: https://rubygems.org/
|
remote: https://rubygems.org/
|
||||||
specs:
|
specs:
|
||||||
actionpack (7.1.2)
|
actionpack (7.1.3.4)
|
||||||
actionview (= 7.1.2)
|
actionview (= 7.1.3.4)
|
||||||
activesupport (= 7.1.2)
|
activesupport (= 7.1.3.4)
|
||||||
nokogiri (>= 1.8.5)
|
nokogiri (>= 1.8.5)
|
||||||
racc
|
racc
|
||||||
rack (>= 2.2.4)
|
rack (>= 2.2.4)
|
||||||
@@ -27,13 +26,13 @@ GEM
|
|||||||
rack-test (>= 0.6.3)
|
rack-test (>= 0.6.3)
|
||||||
rails-dom-testing (~> 2.2)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.6)
|
rails-html-sanitizer (~> 1.6)
|
||||||
actionview (7.1.2)
|
actionview (7.1.3.4)
|
||||||
activesupport (= 7.1.2)
|
activesupport (= 7.1.3.4)
|
||||||
builder (~> 3.1)
|
builder (~> 3.1)
|
||||||
erubi (~> 1.11)
|
erubi (~> 1.11)
|
||||||
rails-dom-testing (~> 2.2)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.6)
|
rails-html-sanitizer (~> 1.6)
|
||||||
activesupport (7.1.2)
|
activesupport (7.1.3.4)
|
||||||
base64
|
base64
|
||||||
bigdecimal
|
bigdecimal
|
||||||
concurrent-ruby (~> 1.0, >= 1.0.2)
|
concurrent-ruby (~> 1.0, >= 1.0.2)
|
||||||
@@ -45,54 +44,55 @@ GEM
|
|||||||
tzinfo (~> 2.0)
|
tzinfo (~> 2.0)
|
||||||
ast (2.4.2)
|
ast (2.4.2)
|
||||||
base64 (0.2.0)
|
base64 (0.2.0)
|
||||||
bcrypt_pbkdf (1.1.0)
|
bcrypt_pbkdf (1.1.1)
|
||||||
bigdecimal (3.1.5)
|
bcrypt_pbkdf (1.1.1-arm64-darwin)
|
||||||
builder (3.2.4)
|
bcrypt_pbkdf (1.1.1-x86_64-darwin)
|
||||||
concurrent-ruby (1.2.2)
|
bigdecimal (3.1.8)
|
||||||
|
builder (3.3.0)
|
||||||
|
concurrent-ruby (1.3.3)
|
||||||
connection_pool (2.4.1)
|
connection_pool (2.4.1)
|
||||||
crass (1.0.6)
|
crass (1.0.6)
|
||||||
debug (1.9.1)
|
debug (1.9.2)
|
||||||
irb (~> 1.10)
|
irb (~> 1.10)
|
||||||
reline (>= 0.3.8)
|
reline (>= 0.3.8)
|
||||||
dotenv (2.8.1)
|
dotenv (3.1.2)
|
||||||
drb (2.2.0)
|
drb (2.2.1)
|
||||||
ruby2_keywords
|
|
||||||
ed25519 (1.3.0)
|
ed25519 (1.3.0)
|
||||||
erubi (1.12.0)
|
erubi (1.13.0)
|
||||||
i18n (1.14.1)
|
i18n (1.14.5)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
io-console (0.7.1)
|
io-console (0.7.2)
|
||||||
irb (1.11.0)
|
irb (1.14.0)
|
||||||
rdoc
|
rdoc (>= 4.0.0)
|
||||||
reline (>= 0.3.8)
|
reline (>= 0.4.2)
|
||||||
json (2.7.1)
|
json (2.7.2)
|
||||||
language_server-protocol (3.17.0.3)
|
language_server-protocol (3.17.0.3)
|
||||||
loofah (2.22.0)
|
loofah (2.22.0)
|
||||||
crass (~> 1.0.2)
|
crass (~> 1.0.2)
|
||||||
nokogiri (>= 1.12.0)
|
nokogiri (>= 1.12.0)
|
||||||
minitest (5.20.0)
|
minitest (5.24.1)
|
||||||
mocha (2.1.0)
|
mocha (2.4.5)
|
||||||
ruby2_keywords (>= 0.0.5)
|
ruby2_keywords (>= 0.0.5)
|
||||||
mutex_m (0.2.0)
|
mutex_m (0.2.0)
|
||||||
net-scp (4.0.0)
|
net-scp (4.0.0)
|
||||||
net-ssh (>= 2.6.5, < 8.0.0)
|
net-ssh (>= 2.6.5, < 8.0.0)
|
||||||
net-sftp (4.0.0)
|
net-sftp (4.0.0)
|
||||||
net-ssh (>= 5.0.0, < 8.0.0)
|
net-ssh (>= 5.0.0, < 8.0.0)
|
||||||
net-ssh (7.2.1)
|
net-ssh (7.2.3)
|
||||||
nokogiri (1.16.0-arm64-darwin)
|
nokogiri (1.16.7-arm64-darwin)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
nokogiri (1.16.0-x86_64-darwin)
|
nokogiri (1.16.7-x86_64-darwin)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
nokogiri (1.16.0-x86_64-linux)
|
nokogiri (1.16.7-x86_64-linux)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
parallel (1.24.0)
|
parallel (1.25.1)
|
||||||
parser (3.3.0.5)
|
parser (3.3.4.0)
|
||||||
ast (~> 2.4.1)
|
ast (~> 2.4.1)
|
||||||
racc
|
racc
|
||||||
psych (5.1.2)
|
psych (5.1.2)
|
||||||
stringio
|
stringio
|
||||||
racc (1.7.3)
|
racc (1.8.1)
|
||||||
rack (3.0.8)
|
rack (3.1.7)
|
||||||
rack-session (2.0.0)
|
rack-session (2.0.0)
|
||||||
rack (>= 3.0.0)
|
rack (>= 3.0.0)
|
||||||
rack-test (2.1.0)
|
rack-test (2.1.0)
|
||||||
@@ -107,42 +107,43 @@ GEM
|
|||||||
rails-html-sanitizer (1.6.0)
|
rails-html-sanitizer (1.6.0)
|
||||||
loofah (~> 2.21)
|
loofah (~> 2.21)
|
||||||
nokogiri (~> 1.14)
|
nokogiri (~> 1.14)
|
||||||
railties (7.1.2)
|
railties (7.1.3.4)
|
||||||
actionpack (= 7.1.2)
|
actionpack (= 7.1.3.4)
|
||||||
activesupport (= 7.1.2)
|
activesupport (= 7.1.3.4)
|
||||||
irb
|
irb
|
||||||
rackup (>= 1.0.0)
|
rackup (>= 1.0.0)
|
||||||
rake (>= 12.2)
|
rake (>= 12.2)
|
||||||
thor (~> 1.0, >= 1.2.2)
|
thor (~> 1.0, >= 1.2.2)
|
||||||
zeitwerk (~> 2.6)
|
zeitwerk (~> 2.6)
|
||||||
rainbow (3.1.1)
|
rainbow (3.1.1)
|
||||||
rake (13.1.0)
|
rake (13.2.1)
|
||||||
rdoc (6.6.2)
|
rdoc (6.7.0)
|
||||||
psych (>= 4.0.0)
|
psych (>= 4.0.0)
|
||||||
regexp_parser (2.9.0)
|
regexp_parser (2.9.2)
|
||||||
reline (0.4.2)
|
reline (0.5.9)
|
||||||
io-console (~> 0.5)
|
io-console (~> 0.5)
|
||||||
rexml (3.2.6)
|
rexml (3.3.4)
|
||||||
rubocop (1.62.1)
|
strscan
|
||||||
|
rubocop (1.65.1)
|
||||||
json (~> 2.3)
|
json (~> 2.3)
|
||||||
language_server-protocol (>= 3.17.0)
|
language_server-protocol (>= 3.17.0)
|
||||||
parallel (~> 1.10)
|
parallel (~> 1.10)
|
||||||
parser (>= 3.3.0.2)
|
parser (>= 3.3.0.2)
|
||||||
rainbow (>= 2.2.2, < 4.0)
|
rainbow (>= 2.2.2, < 4.0)
|
||||||
regexp_parser (>= 1.8, < 3.0)
|
regexp_parser (>= 2.4, < 3.0)
|
||||||
rexml (>= 3.2.5, < 4.0)
|
rexml (>= 3.2.5, < 4.0)
|
||||||
rubocop-ast (>= 1.31.1, < 2.0)
|
rubocop-ast (>= 1.31.1, < 2.0)
|
||||||
ruby-progressbar (~> 1.7)
|
ruby-progressbar (~> 1.7)
|
||||||
unicode-display_width (>= 2.4.0, < 3.0)
|
unicode-display_width (>= 2.4.0, < 3.0)
|
||||||
rubocop-ast (1.31.2)
|
rubocop-ast (1.32.0)
|
||||||
parser (>= 3.3.0.4)
|
parser (>= 3.3.1.0)
|
||||||
rubocop-minitest (0.35.0)
|
rubocop-minitest (0.35.1)
|
||||||
rubocop (>= 1.61, < 2.0)
|
rubocop (>= 1.61, < 2.0)
|
||||||
rubocop-ast (>= 1.31.1, < 2.0)
|
rubocop-ast (>= 1.31.1, < 2.0)
|
||||||
rubocop-performance (1.20.2)
|
rubocop-performance (1.21.1)
|
||||||
rubocop (>= 1.48.1, < 2.0)
|
rubocop (>= 1.48.1, < 2.0)
|
||||||
rubocop-ast (>= 1.30.0, < 2.0)
|
rubocop-ast (>= 1.31.1, < 2.0)
|
||||||
rubocop-rails (2.24.0)
|
rubocop-rails (2.25.1)
|
||||||
activesupport (>= 4.2.0)
|
activesupport (>= 4.2.0)
|
||||||
rack (>= 1.1)
|
rack (>= 1.1)
|
||||||
rubocop (>= 1.33.0, < 2.0)
|
rubocop (>= 1.33.0, < 2.0)
|
||||||
@@ -154,20 +155,19 @@ GEM
|
|||||||
rubocop-rails
|
rubocop-rails
|
||||||
ruby-progressbar (1.13.0)
|
ruby-progressbar (1.13.0)
|
||||||
ruby2_keywords (0.0.5)
|
ruby2_keywords (0.0.5)
|
||||||
sshkit (1.22.2)
|
sshkit (1.23.0)
|
||||||
base64
|
base64
|
||||||
mutex_m
|
|
||||||
net-scp (>= 1.1.2)
|
net-scp (>= 1.1.2)
|
||||||
net-sftp (>= 2.1.2)
|
net-sftp (>= 2.1.2)
|
||||||
net-ssh (>= 2.8.0)
|
net-ssh (>= 2.8.0)
|
||||||
stringio (3.1.0)
|
stringio (3.1.1)
|
||||||
thor (1.3.0)
|
strscan (3.1.0)
|
||||||
|
thor (1.3.1)
|
||||||
tzinfo (2.0.6)
|
tzinfo (2.0.6)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
unicode-display_width (2.5.0)
|
unicode-display_width (2.5.0)
|
||||||
webrick (1.8.1)
|
webrick (1.8.1)
|
||||||
x25519 (1.0.10)
|
zeitwerk (2.6.17)
|
||||||
zeitwerk (2.6.12)
|
|
||||||
|
|
||||||
PLATFORMS
|
PLATFORMS
|
||||||
arm64-darwin
|
arm64-darwin
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# Kamal: Deploy web apps anywhere
|
# Kamal: Deploy web apps anywhere
|
||||||
|
|
||||||
From bare metal to cloud VMs, deploy web apps anywhere with zero downtime. Kamal has the dynamic reverse-proxy Traefik hold requests while a new app container is started and the old one is stopped. Works seamlessly across multiple hosts, using SSHKit to execute commands. Originally built for Rails apps, Kamal will work with any type of web app that can be containerized with Docker.
|
From bare metal to cloud VMs, deploy web apps anywhere with zero downtime. Kamal uses [kamal-proxy](https://github.com/basecamp/kamal-proxy) to seamlessly switch requests between containers. Works seamlessly across multiple servers, using SSHKit to execute commands. Originally built for Rails apps, Kamal will work with any type of web app that can be containerized with Docker.
|
||||||
|
|
||||||
➡️ See [kamal-deploy.org](https://kamal-deploy.org) for documentation on [installation](https://kamal-deploy.org/docs/installation), [configuration](https://kamal-deploy.org/docs/configuration), and [commands](https://kamal-deploy.org/docs/commands).
|
➡️ See [kamal-deploy.org](https://kamal-deploy.org) for documentation on [installation](https://kamal-deploy.org/docs/installation), [configuration](https://kamal-deploy.org/docs/configuration), and [commands](https://kamal-deploy.org/docs/commands).
|
||||||
|
|
||||||
|
|||||||
25
bin/docs
25
bin/docs
@@ -17,18 +17,18 @@ end
|
|||||||
|
|
||||||
DOCS = {
|
DOCS = {
|
||||||
"accessory" => "Accessories",
|
"accessory" => "Accessories",
|
||||||
|
"alias" => "Aliases",
|
||||||
"boot" => "Booting",
|
"boot" => "Booting",
|
||||||
"builder" => "Builders",
|
"builder" => "Builders",
|
||||||
"configuration" => "Configuration overview",
|
"configuration" => "Configuration overview",
|
||||||
"env" => "Environment variables",
|
"env" => "Environment variables",
|
||||||
"healthcheck" => "Healthchecks",
|
|
||||||
"logging" => "Logging",
|
"logging" => "Logging",
|
||||||
|
"proxy" => "Proxy",
|
||||||
"registry" => "Docker Registry",
|
"registry" => "Docker Registry",
|
||||||
"role" => "Roles",
|
"role" => "Roles",
|
||||||
"servers" => "Servers",
|
"servers" => "Servers",
|
||||||
"ssh" => "SSH",
|
"ssh" => "SSH",
|
||||||
"sshkit" => "SSHKit",
|
"sshkit" => "SSHKit"
|
||||||
"traefik" => "Traefik"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class DocWriter
|
class DocWriter
|
||||||
@@ -67,26 +67,27 @@ class DocWriter
|
|||||||
output.puts
|
output.puts
|
||||||
place = :new_section
|
place = :new_section
|
||||||
elsif line =~ /^ *#/
|
elsif line =~ /^ *#/
|
||||||
generate_line(line, place: place)
|
generate_line(line, heading: place == :new_section)
|
||||||
place = :in_section
|
place = :in_section
|
||||||
else
|
else
|
||||||
output.puts "```yaml"
|
output.puts "```yaml"
|
||||||
output.print line
|
output.puts line
|
||||||
place = :in_yaml
|
place = :in_yaml
|
||||||
end
|
end
|
||||||
when :in_yaml
|
when :in_yaml, :in_empty_line_yaml
|
||||||
if line =~ /^ *#/
|
if line =~ /^ *#/
|
||||||
output.puts "```"
|
output.puts "```"
|
||||||
generate_line(line, place: :new_section)
|
generate_line(line, heading: place == :in_empty_line_yaml)
|
||||||
place = :in_section
|
place = :in_section
|
||||||
|
elsif line.empty?
|
||||||
|
place = :in_empty_line_yaml
|
||||||
else
|
else
|
||||||
output.puts
|
output.puts line
|
||||||
output.print line
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
output.puts "\n```" if place == :in_yaml
|
output.puts "```" if place == :in_yaml
|
||||||
end
|
end
|
||||||
|
|
||||||
def generate_header
|
def generate_header
|
||||||
@@ -98,7 +99,7 @@ class DocWriter
|
|||||||
output.puts
|
output.puts
|
||||||
end
|
end
|
||||||
|
|
||||||
def generate_line(line, place: :in_section)
|
def generate_line(line, heading: false)
|
||||||
line = line.gsub(/^ *#\s?/, "")
|
line = line.gsub(/^ *#\s?/, "")
|
||||||
|
|
||||||
if line =~ /(.*)kamal docs ([a-z]*)(.*)/
|
if line =~ /(.*)kamal docs ([a-z]*)(.*)/
|
||||||
@@ -109,7 +110,7 @@ class DocWriter
|
|||||||
line = "#{$1}[#{titlify($2.split("/").last)}](#{$2})#{$3}"
|
line = "#{$1}[#{titlify($2.split("/").last)}](#{$2})#{$3}"
|
||||||
end
|
end
|
||||||
|
|
||||||
if place == :new_section
|
if heading
|
||||||
output.puts "## [#{line}](##{linkify(line)})"
|
output.puts "## [#{line}](##{linkify(line)})"
|
||||||
else
|
else
|
||||||
output.puts line
|
output.puts line
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
source 'https://rubygems.org'
|
|
||||||
git_source(:github) { |repo| "https://github.com/#{repo}.git" }
|
|
||||||
|
|
||||||
gemspec path: "../"
|
|
||||||
|
|
||||||
gem "nokogiri", "~> 1.15.0"
|
|
||||||
@@ -12,13 +12,12 @@ Gem::Specification.new do |spec|
|
|||||||
spec.executables = %w[ kamal ]
|
spec.executables = %w[ kamal ]
|
||||||
|
|
||||||
spec.add_dependency "activesupport", ">= 7.0"
|
spec.add_dependency "activesupport", ">= 7.0"
|
||||||
spec.add_dependency "sshkit", ">= 1.22.2", "< 2.0"
|
spec.add_dependency "sshkit", ">= 1.23.0", "< 2.0"
|
||||||
spec.add_dependency "net-ssh", "~> 7.0"
|
spec.add_dependency "net-ssh", "~> 7.0"
|
||||||
spec.add_dependency "thor", "~> 1.2"
|
spec.add_dependency "thor", "~> 1.3"
|
||||||
spec.add_dependency "dotenv", "~> 2.8"
|
spec.add_dependency "dotenv", "~> 3.1"
|
||||||
spec.add_dependency "zeitwerk", "~> 2.5"
|
spec.add_dependency "zeitwerk", "~> 2.5"
|
||||||
spec.add_dependency "ed25519", "~> 1.2"
|
spec.add_dependency "ed25519", "~> 1.2"
|
||||||
spec.add_dependency "x25519", "~> 1.0", ">= 1.0.10"
|
|
||||||
spec.add_dependency "bcrypt_pbkdf", "~> 1.0"
|
spec.add_dependency "bcrypt_pbkdf", "~> 1.0"
|
||||||
spec.add_dependency "concurrent-ruby", "~> 1.2"
|
spec.add_dependency "concurrent-ruby", "~> 1.2"
|
||||||
spec.add_dependency "base64", "~> 0.2"
|
spec.add_dependency "base64", "~> 0.2"
|
||||||
|
|||||||
@@ -5,8 +5,10 @@ end
|
|||||||
require "active_support"
|
require "active_support"
|
||||||
require "zeitwerk"
|
require "zeitwerk"
|
||||||
require "yaml"
|
require "yaml"
|
||||||
|
require "tmpdir"
|
||||||
|
require "pathname"
|
||||||
|
|
||||||
loader = Zeitwerk::Loader.for_gem
|
loader = Zeitwerk::Loader.for_gem
|
||||||
loader.ignore(File.join(__dir__, "kamal", "sshkit_with_ext.rb"))
|
loader.ignore(File.join(__dir__, "kamal", "sshkit_with_ext.rb"))
|
||||||
loader.setup
|
loader.setup
|
||||||
loader.eager_load # We need all commands loaded.
|
loader.eager_load_namespace(Kamal::Cli) # We need all commands loaded.
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
module Kamal::Cli
|
module Kamal::Cli
|
||||||
|
class BootError < StandardError; end
|
||||||
class HookError < StandardError; end
|
class HookError < StandardError; end
|
||||||
class LockError < StandardError; end
|
class LockError < StandardError; end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,17 +1,20 @@
|
|||||||
class Kamal::Cli::Accessory < Kamal::Cli::Base
|
class Kamal::Cli::Accessory < Kamal::Cli::Base
|
||||||
desc "boot [NAME]", "Boot new accessory service on host (use NAME=all to boot all accessories)"
|
desc "boot [NAME]", "Boot new accessory service on host (use NAME=all to boot all accessories)"
|
||||||
def boot(name, login: true)
|
def boot(name, prepare: true)
|
||||||
with_lock do
|
with_lock do
|
||||||
if name == "all"
|
if name == "all"
|
||||||
KAMAL.accessory_names.each { |accessory_name| boot(accessory_name) }
|
KAMAL.accessory_names.each { |accessory_name| boot(accessory_name) }
|
||||||
else
|
else
|
||||||
|
prepare(name) if prepare
|
||||||
|
|
||||||
with_accessory(name) do |accessory, hosts|
|
with_accessory(name) do |accessory, hosts|
|
||||||
directories(name)
|
directories(name)
|
||||||
upload(name)
|
upload(name)
|
||||||
|
|
||||||
on(hosts) do
|
on(hosts) do
|
||||||
execute *KAMAL.registry.login if login
|
|
||||||
execute *KAMAL.auditor.record("Booted #{name} accessory"), verbosity: :debug
|
execute *KAMAL.auditor.record("Booted #{name} accessory"), verbosity: :debug
|
||||||
|
execute *accessory.ensure_env_directory
|
||||||
|
upload! accessory.secrets_io, accessory.secrets_path, mode: "0600"
|
||||||
execute *accessory.run
|
execute *accessory.run
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -55,15 +58,10 @@ class Kamal::Cli::Accessory < Kamal::Cli::Base
|
|||||||
if name == "all"
|
if name == "all"
|
||||||
KAMAL.accessory_names.each { |accessory_name| reboot(accessory_name) }
|
KAMAL.accessory_names.each { |accessory_name| reboot(accessory_name) }
|
||||||
else
|
else
|
||||||
with_accessory(name) do |accessory, hosts|
|
prepare(name)
|
||||||
on(hosts) do
|
stop(name)
|
||||||
execute *KAMAL.registry.login
|
remove_container(name)
|
||||||
end
|
boot(name, prepare: false)
|
||||||
|
|
||||||
stop(name)
|
|
||||||
remove_container(name)
|
|
||||||
boot(name, login: false)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -95,10 +93,8 @@ class Kamal::Cli::Accessory < Kamal::Cli::Base
|
|||||||
desc "restart [NAME]", "Restart existing accessory container on host"
|
desc "restart [NAME]", "Restart existing accessory container on host"
|
||||||
def restart(name)
|
def restart(name)
|
||||||
with_lock do
|
with_lock do
|
||||||
with_accessory(name) do
|
stop(name)
|
||||||
stop(name)
|
start(name)
|
||||||
start(name)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -151,23 +147,25 @@ class Kamal::Cli::Accessory < Kamal::Cli::Base
|
|||||||
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
||||||
option :grep_options, aliases: "-o", desc: "Additional options supplied to grep"
|
option :grep_options, aliases: "-o", desc: "Additional options supplied to grep"
|
||||||
option :follow, aliases: "-f", desc: "Follow logs on primary server (or specific host set by --hosts)"
|
option :follow, aliases: "-f", desc: "Follow logs on primary server (or specific host set by --hosts)"
|
||||||
|
option :skip_timestamps, type: :boolean, aliases: "-T", desc: "Skip appending timestamps to logging output"
|
||||||
def logs(name)
|
def logs(name)
|
||||||
with_accessory(name) do |accessory, hosts|
|
with_accessory(name) do |accessory, hosts|
|
||||||
grep = options[:grep]
|
grep = options[:grep]
|
||||||
grep_options = options[:grep_options]
|
grep_options = options[:grep_options]
|
||||||
|
timestamps = !options[:skip_timestamps]
|
||||||
|
|
||||||
if options[:follow]
|
if options[:follow]
|
||||||
run_locally do
|
run_locally do
|
||||||
info "Following logs on #{hosts}..."
|
info "Following logs on #{hosts}..."
|
||||||
info accessory.follow_logs(grep: grep, grep_options: grep_options)
|
info accessory.follow_logs(timestamps: timestamps, grep: grep, grep_options: grep_options)
|
||||||
exec accessory.follow_logs(grep: grep, grep_options: grep_options)
|
exec accessory.follow_logs(timestamps: timestamps, grep: grep, grep_options: grep_options)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
since = options[:since]
|
since = options[:since]
|
||||||
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
||||||
|
|
||||||
on(hosts) do
|
on(hosts) do
|
||||||
puts capture_with_info(*accessory.logs(since: since, lines: lines, grep: grep, grep_options: grep_options))
|
puts capture_with_info(*accessory.logs(timestamps: timestamps, since: since, lines: lines, grep: grep, grep_options: grep_options))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -222,6 +220,25 @@ class Kamal::Cli::Accessory < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
desc "upgrade", "Upgrade accessories from Kamal 1.x to 2.0 (restart them in 'kamal' network)"
|
||||||
|
option :rolling, type: :boolean, default: false, desc: "Upgrade one host at a time"
|
||||||
|
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
||||||
|
def upgrade(name)
|
||||||
|
confirming "This will restart all accessories" do
|
||||||
|
with_lock do
|
||||||
|
host_groups = options[:rolling] ? KAMAL.accessory_hosts : [ KAMAL.accessory_hosts ]
|
||||||
|
host_groups.each do |hosts|
|
||||||
|
host_list = Array(hosts).join(",")
|
||||||
|
KAMAL.with_specific_hosts(hosts) do
|
||||||
|
say "Upgrading #{name} accessories on #{host_list}...", :magenta
|
||||||
|
reboot name
|
||||||
|
say "Upgraded #{name} accessories on #{host_list}...", :magenta
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def with_accessory(name)
|
def with_accessory(name)
|
||||||
if KAMAL.config.accessory(name)
|
if KAMAL.config.accessory(name)
|
||||||
@@ -249,11 +266,20 @@ class Kamal::Cli::Accessory < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def remove_accessory(name)
|
def remove_accessory(name)
|
||||||
with_accessory(name) do
|
stop(name)
|
||||||
stop(name)
|
remove_container(name)
|
||||||
remove_container(name)
|
remove_image(name)
|
||||||
remove_image(name)
|
remove_service_directory(name)
|
||||||
remove_service_directory(name)
|
end
|
||||||
|
|
||||||
|
def prepare(name)
|
||||||
|
with_accessory(name) do |accessory, hosts|
|
||||||
|
on(hosts) do
|
||||||
|
execute *KAMAL.registry.login
|
||||||
|
execute *KAMAL.docker.create_network
|
||||||
|
rescue SSHKit::Command::Failed => e
|
||||||
|
raise unless e.message.include?("already exists")
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
9
lib/kamal/cli/alias/command.rb
Normal file
9
lib/kamal/cli/alias/command.rb
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
class Kamal::Cli::Alias::Command < Thor::DynamicCommand
|
||||||
|
def run(instance, args = [])
|
||||||
|
if (_alias = KAMAL.config.aliases[name])
|
||||||
|
Kamal::Cli::Main.start(Shellwords.split(_alias.command) + ARGV[1..-1])
|
||||||
|
else
|
||||||
|
super
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -4,7 +4,7 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
with_lock do
|
with_lock do
|
||||||
say "Get most recent version available as an image...", :magenta unless options[:version]
|
say "Get most recent version available as an image...", :magenta unless options[:version]
|
||||||
using_version(version_or_latest) do |version|
|
using_version(version_or_latest) do |version|
|
||||||
say "Start container with version #{version} using a #{KAMAL.config.readiness_delay}s readiness delay (or reboot if already running)...", :magenta
|
say "Start container with version #{version} (or reboot if already running)...", :magenta
|
||||||
|
|
||||||
# Assets are prepared in a separate step to ensure they are on all hosts before booting
|
# Assets are prepared in a separate step to ensure they are on all hosts before booting
|
||||||
on(KAMAL.hosts) do
|
on(KAMAL.hosts) do
|
||||||
@@ -38,8 +38,17 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
roles = KAMAL.roles_on(host)
|
roles = KAMAL.roles_on(host)
|
||||||
|
|
||||||
roles.each do |role|
|
roles.each do |role|
|
||||||
|
app = KAMAL.app(role: role, host: host)
|
||||||
execute *KAMAL.auditor.record("Started app version #{KAMAL.config.version}"), verbosity: :debug
|
execute *KAMAL.auditor.record("Started app version #{KAMAL.config.version}"), verbosity: :debug
|
||||||
execute *KAMAL.app(role: role, host: host).start, raise_on_non_zero_exit: false
|
execute *app.start, raise_on_non_zero_exit: false
|
||||||
|
|
||||||
|
if role.running_proxy?
|
||||||
|
version = capture_with_info(*app.current_running_version, raise_on_non_zero_exit: false).strip
|
||||||
|
endpoint = capture_with_info(*app.container_id_for_version(version)).strip
|
||||||
|
raise Kamal::Cli::BootError, "Failed to get endpoint for #{role} on #{host}, did the container boot?" if endpoint.empty?
|
||||||
|
|
||||||
|
execute *app.deploy(target: endpoint)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -52,8 +61,18 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
roles = KAMAL.roles_on(host)
|
roles = KAMAL.roles_on(host)
|
||||||
|
|
||||||
roles.each do |role|
|
roles.each do |role|
|
||||||
|
app = KAMAL.app(role: role, host: host)
|
||||||
execute *KAMAL.auditor.record("Stopped app", role: role), verbosity: :debug
|
execute *KAMAL.auditor.record("Stopped app", role: role), verbosity: :debug
|
||||||
execute *KAMAL.app(role: role, host: host).stop, raise_on_non_zero_exit: false
|
|
||||||
|
if role.running_proxy?
|
||||||
|
version = capture_with_info(*app.current_running_version, raise_on_non_zero_exit: false).strip
|
||||||
|
endpoint = capture_with_info(*app.container_id_for_version(version)).strip
|
||||||
|
if endpoint.present?
|
||||||
|
execute *app.remove(target: endpoint), raise_on_non_zero_exit: false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
execute *app.stop, raise_on_non_zero_exit: false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -71,11 +90,12 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "exec [CMD]", "Execute a custom command on servers within the app container (use --help to show options)"
|
desc "exec [CMD...]", "Execute a custom command on servers within the app container (use --help to show options)"
|
||||||
option :interactive, aliases: "-i", type: :boolean, default: false, desc: "Execute command over ssh for an interactive shell (use for console/bash)"
|
option :interactive, aliases: "-i", type: :boolean, default: false, desc: "Execute command over ssh for an interactive shell (use for console/bash)"
|
||||||
option :reuse, type: :boolean, default: false, desc: "Reuse currently running container instead of starting a new one"
|
option :reuse, type: :boolean, default: false, desc: "Reuse currently running container instead of starting a new one"
|
||||||
option :env, aliases: "-e", type: :hash, desc: "Set environment variables for the command"
|
option :env, aliases: "-e", type: :hash, desc: "Set environment variables for the command"
|
||||||
def exec(cmd)
|
def exec(*cmd)
|
||||||
|
cmd = Kamal::Utils.join_commands(cmd)
|
||||||
env = options[:env]
|
env = options[:env]
|
||||||
case
|
case
|
||||||
when options[:interactive] && options[:reuse]
|
when options[:interactive] && options[:reuse]
|
||||||
@@ -168,12 +188,14 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
||||||
option :grep_options, aliases: "-o", desc: "Additional options supplied to grep"
|
option :grep_options, aliases: "-o", desc: "Additional options supplied to grep"
|
||||||
option :follow, aliases: "-f", desc: "Follow log on primary server (or specific host set by --hosts)"
|
option :follow, aliases: "-f", desc: "Follow log on primary server (or specific host set by --hosts)"
|
||||||
|
option :skip_timestamps, type: :boolean, aliases: "-T", desc: "Skip appending timestamps to logging output"
|
||||||
def logs
|
def logs
|
||||||
# FIXME: Catch when app containers aren't running
|
# FIXME: Catch when app containers aren't running
|
||||||
|
|
||||||
grep = options[:grep]
|
grep = options[:grep]
|
||||||
grep_options = options[:grep_options]
|
grep_options = options[:grep_options]
|
||||||
since = options[:since]
|
since = options[:since]
|
||||||
|
timestamps = !options[:skip_timestamps]
|
||||||
|
|
||||||
if options[:follow]
|
if options[:follow]
|
||||||
lines = options[:lines].presence || ((since || grep) ? nil : 10) # Default to 10 lines if since or grep isn't set
|
lines = options[:lines].presence || ((since || grep) ? nil : 10) # Default to 10 lines if since or grep isn't set
|
||||||
@@ -185,8 +207,8 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
role = KAMAL.roles_on(KAMAL.primary_host).first
|
role = KAMAL.roles_on(KAMAL.primary_host).first
|
||||||
|
|
||||||
app = KAMAL.app(role: role, host: host)
|
app = KAMAL.app(role: role, host: host)
|
||||||
info app.follow_logs(host: KAMAL.primary_host, lines: lines, grep: grep, grep_options: grep_options)
|
info app.follow_logs(host: KAMAL.primary_host, timestamps: timestamps, lines: lines, grep: grep, grep_options: grep_options)
|
||||||
exec app.follow_logs(host: KAMAL.primary_host, lines: lines, grep: grep, grep_options: grep_options)
|
exec app.follow_logs(host: KAMAL.primary_host, timestamps: timestamps, lines: lines, grep: grep, grep_options: grep_options)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
||||||
@@ -196,7 +218,7 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
|
|
||||||
roles.each do |role|
|
roles.each do |role|
|
||||||
begin
|
begin
|
||||||
puts_by_host host, capture_with_info(*KAMAL.app(role: role, host: host).logs(since: since, lines: lines, grep: grep, grep_options: grep_options))
|
puts_by_host host, capture_with_info(*KAMAL.app(role: role, host: host).logs(timestamps: timestamps, since: since, lines: lines, grep: grep, grep_options: grep_options))
|
||||||
rescue SSHKit::Command::Failed
|
rescue SSHKit::Command::Failed
|
||||||
puts_by_host host, "Nothing found"
|
puts_by_host host, "Nothing found"
|
||||||
end
|
end
|
||||||
@@ -211,6 +233,7 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
stop
|
stop
|
||||||
remove_containers
|
remove_containers
|
||||||
remove_images
|
remove_images
|
||||||
|
remove_app_directory
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -252,6 +275,20 @@ class Kamal::Cli::App < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
desc "remove_app_directory", "Remove the service directory from servers", hide: true
|
||||||
|
def remove_app_directory
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.hosts) do |host|
|
||||||
|
roles = KAMAL.roles_on(host)
|
||||||
|
|
||||||
|
roles.each do |role|
|
||||||
|
execute *KAMAL.auditor.record("Removed #{KAMAL.config.app_directory} on all servers", role: role), verbosity: :debug
|
||||||
|
execute *KAMAL.server.remove_app_directory, raise_on_non_zero_exit: false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
desc "version", "Show app version currently running on servers"
|
desc "version", "Show app version currently running on servers"
|
||||||
def version
|
def version
|
||||||
on(KAMAL.hosts) do |host|
|
on(KAMAL.hosts) do |host|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
class Kamal::Cli::App::Boot
|
class Kamal::Cli::App::Boot
|
||||||
attr_reader :host, :role, :version, :barrier, :sshkit
|
attr_reader :host, :role, :version, :barrier, :sshkit
|
||||||
delegate :execute, :capture_with_info, :capture_with_pretty_json, :info, :error, to: :sshkit
|
delegate :execute, :capture_with_info, :capture_with_pretty_json, :info, :error, :upload!, to: :sshkit
|
||||||
delegate :uses_cord?, :assets?, :running_traefik?, to: :role
|
delegate :assets?, :running_proxy?, to: :role
|
||||||
|
|
||||||
def initialize(host, role, sshkit, version, barrier)
|
def initialize(host, role, sshkit, version, barrier)
|
||||||
@host = host
|
@host = host
|
||||||
@@ -45,11 +45,22 @@ class Kamal::Cli::App::Boot
|
|||||||
|
|
||||||
def start_new_version
|
def start_new_version
|
||||||
audit "Booted app version #{version}"
|
audit "Booted app version #{version}"
|
||||||
|
|
||||||
execute *app.tie_cord(role.cord_host_file) if uses_cord?
|
|
||||||
hostname = "#{host.to_s[0...51].gsub(/\.+$/, '')}-#{SecureRandom.hex(6)}"
|
hostname = "#{host.to_s[0...51].gsub(/\.+$/, '')}-#{SecureRandom.hex(6)}"
|
||||||
|
|
||||||
|
execute *app.ensure_env_directory
|
||||||
|
upload! role.secrets_io(host), role.secrets_path, mode: "0600"
|
||||||
|
|
||||||
execute *app.run(hostname: hostname)
|
execute *app.run(hostname: hostname)
|
||||||
Kamal::Cli::Healthcheck::Poller.wait_for_healthy(pause_after_ready: true) { capture_with_info(*app.status(version: version)) }
|
if running_proxy?
|
||||||
|
endpoint = capture_with_info(*app.container_id_for_version(version)).strip
|
||||||
|
raise Kamal::Cli::BootError, "Failed to get endpoint for #{role} on #{host}, did the container boot?" if endpoint.empty?
|
||||||
|
execute *app.deploy(target: endpoint)
|
||||||
|
else
|
||||||
|
Kamal::Cli::Healthcheck::Poller.wait_for_healthy(pause_after_ready: true) { capture_with_info(*app.status(version: version)) }
|
||||||
|
end
|
||||||
|
rescue => e
|
||||||
|
error "Failed to boot #{role} on #{host}"
|
||||||
|
raise e
|
||||||
end
|
end
|
||||||
|
|
||||||
def stop_new_version
|
def stop_new_version
|
||||||
@@ -57,16 +68,7 @@ class Kamal::Cli::App::Boot
|
|||||||
end
|
end
|
||||||
|
|
||||||
def stop_old_version(version)
|
def stop_old_version(version)
|
||||||
if uses_cord?
|
|
||||||
cord = capture_with_info(*app.cord(version: version), raise_on_non_zero_exit: false).strip
|
|
||||||
if cord.present?
|
|
||||||
execute *app.cut_cord(cord)
|
|
||||||
Kamal::Cli::Healthcheck::Poller.wait_for_unhealthy(pause_after_ready: true) { capture_with_info(*app.status(version: version)) }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
execute *app.stop(version: version), raise_on_non_zero_exit: false
|
execute *app.stop(version: version), raise_on_non_zero_exit: false
|
||||||
|
|
||||||
execute *app.clean_up_assets if assets?
|
execute *app.clean_up_assets if assets?
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -88,8 +90,12 @@ class Kamal::Cli::App::Boot
|
|||||||
def close_barrier
|
def close_barrier
|
||||||
if barrier.close
|
if barrier.close
|
||||||
info "First #{KAMAL.primary_role} container is unhealthy on #{host}, not booting any other roles"
|
info "First #{KAMAL.primary_role} container is unhealthy on #{host}, not booting any other roles"
|
||||||
error capture_with_info(*app.logs(version: version))
|
begin
|
||||||
error capture_with_info(*app.container_health_log(version: version))
|
error capture_with_info(*app.logs(version: version))
|
||||||
|
error capture_with_info(*app.container_health_log(version: version))
|
||||||
|
rescue SSHKit::Command::Failed
|
||||||
|
error "Could not fetch logs for #{version}"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
require "thor"
|
require "thor"
|
||||||
require "dotenv"
|
|
||||||
require "kamal/sshkit_with_ext"
|
require "kamal/sshkit_with_ext"
|
||||||
|
|
||||||
module Kamal::Cli
|
module Kamal::Cli
|
||||||
class Base < Thor
|
class Base < Thor
|
||||||
include SSHKit::DSL
|
include SSHKit::DSL
|
||||||
|
|
||||||
def self.exit_on_failure?() true end
|
def self.exit_on_failure?() false end
|
||||||
|
def self.dynamic_command_class() Kamal::Cli::Alias::Command end
|
||||||
|
|
||||||
class_option :verbose, type: :boolean, aliases: "-v", desc: "Detailed logging"
|
class_option :verbose, type: :boolean, aliases: "-v", desc: "Detailed logging"
|
||||||
class_option :quiet, type: :boolean, aliases: "-q", desc: "Minimal logging"
|
class_option :quiet, type: :boolean, aliases: "-q", desc: "Minimal logging"
|
||||||
@@ -22,33 +22,23 @@ module Kamal::Cli
|
|||||||
|
|
||||||
class_option :skip_hooks, aliases: "-H", type: :boolean, default: false, desc: "Don't run hooks"
|
class_option :skip_hooks, aliases: "-H", type: :boolean, default: false, desc: "Don't run hooks"
|
||||||
|
|
||||||
def initialize(*)
|
def initialize(args = [], local_options = {}, config = {})
|
||||||
super
|
if config[:current_command].is_a?(Kamal::Cli::Alias::Command)
|
||||||
@original_env = ENV.to_h.dup
|
# When Thor generates a dynamic command, it doesn't attempt to parse the arguments.
|
||||||
load_envs
|
# For our purposes, it means the arguments are passed in args rather than local_options.
|
||||||
initialize_commander(options_with_subcommand_class_options)
|
super([], args, config)
|
||||||
|
else
|
||||||
|
super
|
||||||
|
end
|
||||||
|
initialize_commander unless KAMAL.configured?
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def load_envs
|
|
||||||
if destination = options[:destination]
|
|
||||||
Dotenv.load(".env.#{destination}", ".env")
|
|
||||||
else
|
|
||||||
Dotenv.load(".env")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def reload_envs
|
|
||||||
ENV.clear
|
|
||||||
ENV.update(@original_env)
|
|
||||||
load_envs
|
|
||||||
end
|
|
||||||
|
|
||||||
def options_with_subcommand_class_options
|
def options_with_subcommand_class_options
|
||||||
options.merge(@_initializer.last[:class_options] || {})
|
options.merge(@_initializer.last[:class_options] || {})
|
||||||
end
|
end
|
||||||
|
|
||||||
def initialize_commander(options)
|
def initialize_commander
|
||||||
KAMAL.tap do |commander|
|
KAMAL.tap do |commander|
|
||||||
if options[:verbose]
|
if options[:verbose]
|
||||||
ENV["VERBOSE"] = "1" # For backtraces via cli/start
|
ENV["VERBOSE"] = "1" # For backtraces via cli/start
|
||||||
@@ -83,8 +73,6 @@ module Kamal::Cli
|
|||||||
if KAMAL.holding_lock?
|
if KAMAL.holding_lock?
|
||||||
yield
|
yield
|
||||||
else
|
else
|
||||||
ensure_run_and_locks_directory
|
|
||||||
|
|
||||||
acquire_lock
|
acquire_lock
|
||||||
|
|
||||||
begin
|
begin
|
||||||
@@ -113,6 +101,8 @@ module Kamal::Cli
|
|||||||
end
|
end
|
||||||
|
|
||||||
def acquire_lock
|
def acquire_lock
|
||||||
|
ensure_run_directory
|
||||||
|
|
||||||
raise_if_locked do
|
raise_if_locked do
|
||||||
say "Acquiring the deploy lock...", :magenta
|
say "Acquiring the deploy lock...", :magenta
|
||||||
on(KAMAL.primary_host) { execute *KAMAL.lock.acquire("Automatic deploy lock", KAMAL.config.version), verbosity: :debug }
|
on(KAMAL.primary_host) { execute *KAMAL.lock.acquire("Automatic deploy lock", KAMAL.config.version), verbosity: :debug }
|
||||||
@@ -145,8 +135,10 @@ module Kamal::Cli
|
|||||||
details = { hosts: KAMAL.hosts.join(","), command: command, subcommand: subcommand }
|
details = { hosts: KAMAL.hosts.join(","), command: command, subcommand: subcommand }
|
||||||
|
|
||||||
say "Running the #{hook} hook...", :magenta
|
say "Running the #{hook} hook...", :magenta
|
||||||
run_locally do
|
with_env KAMAL.hook.env(**details, **extra_details) do
|
||||||
execute *KAMAL.hook.run(hook, **details, **extra_details)
|
run_locally do
|
||||||
|
execute *KAMAL.hook.run(hook)
|
||||||
|
end
|
||||||
rescue SSHKit::Command::Failed => e
|
rescue SSHKit::Command::Failed => e
|
||||||
raise HookError.new("Hook `#{hook}` failed:\n#{e.message}")
|
raise HookError.new("Hook `#{hook}` failed:\n#{e.message}")
|
||||||
end
|
end
|
||||||
@@ -184,14 +176,23 @@ module Kamal::Cli
|
|||||||
instance_variable_get("@_invocations").first
|
instance_variable_get("@_invocations").first
|
||||||
end
|
end
|
||||||
|
|
||||||
def ensure_run_and_locks_directory
|
def reset_invocation(cli_class)
|
||||||
|
instance_variable_get("@_invocations")[cli_class].pop
|
||||||
|
end
|
||||||
|
|
||||||
|
def ensure_run_directory
|
||||||
on(KAMAL.hosts) do
|
on(KAMAL.hosts) do
|
||||||
execute(*KAMAL.server.ensure_run_directory)
|
execute(*KAMAL.server.ensure_run_directory)
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
on(KAMAL.primary_host) do
|
def with_env(env)
|
||||||
execute(*KAMAL.lock.ensure_locks_directory)
|
current_env = ENV.to_h.dup
|
||||||
end
|
ENV.update(env)
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
ENV.clear
|
||||||
|
ENV.update(current_env)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -30,46 +30,50 @@ class Kamal::Cli::Build < Kamal::Cli::Base
|
|||||||
say "Building with uncommitted changes:\n #{uncommitted_changes}", :yellow
|
say "Building with uncommitted changes:\n #{uncommitted_changes}", :yellow
|
||||||
end
|
end
|
||||||
|
|
||||||
# Get the command here to ensure the Dir.chdir doesn't interfere with it
|
with_env(KAMAL.config.builder.secrets) do
|
||||||
push = KAMAL.builder.push
|
run_locally do
|
||||||
|
begin
|
||||||
run_locally do
|
execute *KAMAL.builder.inspect_builder
|
||||||
begin
|
rescue SSHKit::Command::Failed => e
|
||||||
context_hosts = capture_with_info(*KAMAL.builder.context_hosts).split("\n")
|
if e.message =~ /(context not found|no builder|no compatible builder|does not exist)/
|
||||||
|
warn "Missing compatible builder, so creating a new one first"
|
||||||
if context_hosts != KAMAL.builder.config_context_hosts
|
begin
|
||||||
warn "Context hosts have changed, so re-creating builder, was: #{context_hosts.join(", ")}], now: #{KAMAL.builder.config_context_hosts.join(", ")}"
|
cli.remove
|
||||||
cli.remove
|
rescue SSHKit::Command::Failed
|
||||||
cli.create
|
raise unless e.message =~ /(context not found|no builder|does not exist)/
|
||||||
|
end
|
||||||
|
cli.create
|
||||||
|
else
|
||||||
|
raise
|
||||||
|
end
|
||||||
end
|
end
|
||||||
rescue SSHKit::Command::Failed => e
|
|
||||||
warn "Missing compatible builder, so creating a new one first"
|
|
||||||
if e.message =~ /(context not found|no builder)/
|
|
||||||
cli.create
|
|
||||||
else
|
|
||||||
raise
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
KAMAL.with_verbosity(:debug) do
|
# Get the command here to ensure the Dir.chdir doesn't interfere with it
|
||||||
Dir.chdir(KAMAL.config.builder.build_directory) { execute *push }
|
push = KAMAL.builder.push
|
||||||
|
|
||||||
|
KAMAL.with_verbosity(:debug) do
|
||||||
|
Dir.chdir(KAMAL.config.builder.build_directory) { execute *push }
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "pull", "Pull app image from registry onto servers"
|
desc "pull", "Pull app image from registry onto servers"
|
||||||
def pull
|
def pull
|
||||||
on(KAMAL.hosts) do
|
if (first_hosts = mirror_hosts).any?
|
||||||
execute *KAMAL.auditor.record("Pulled image with version #{KAMAL.config.version}"), verbosity: :debug
|
# Pull on a single host per mirror first to seed them
|
||||||
execute *KAMAL.builder.clean, raise_on_non_zero_exit: false
|
say "Pulling image on #{first_hosts.join(", ")} to seed the #{"mirror".pluralize(first_hosts.count)}...", :magenta
|
||||||
execute *KAMAL.builder.pull
|
pull_on_hosts(first_hosts)
|
||||||
execute *KAMAL.builder.validate_image
|
say "Pulling image on remaining hosts...", :magenta
|
||||||
|
pull_on_hosts(KAMAL.hosts - first_hosts)
|
||||||
|
else
|
||||||
|
pull_on_hosts(KAMAL.hosts)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "create", "Create a build setup"
|
desc "create", "Create a build setup"
|
||||||
def create
|
def create
|
||||||
if (remote_host = KAMAL.config.builder.remote_host)
|
if (remote_host = KAMAL.config.builder.remote)
|
||||||
connect_to_remote_host(remote_host)
|
connect_to_remote_host(remote_host)
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -131,4 +135,28 @@ class Kamal::Cli::Build < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def mirror_hosts
|
||||||
|
if KAMAL.hosts.many?
|
||||||
|
mirror_hosts = Concurrent::Hash.new
|
||||||
|
on(KAMAL.hosts) do |host|
|
||||||
|
first_mirror = capture_with_info(*KAMAL.builder.first_mirror).strip.presence
|
||||||
|
mirror_hosts[first_mirror] ||= host.to_s if first_mirror
|
||||||
|
rescue SSHKit::Command::Failed => e
|
||||||
|
raise unless e.message =~ /error calling index: reflect: slice index out of range/
|
||||||
|
end
|
||||||
|
mirror_hosts.values
|
||||||
|
else
|
||||||
|
[]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def pull_on_hosts(hosts)
|
||||||
|
on(hosts) do
|
||||||
|
execute *KAMAL.auditor.record("Pulled image with version #{KAMAL.config.version}"), verbosity: :debug
|
||||||
|
execute *KAMAL.builder.clean, raise_on_non_zero_exit: false
|
||||||
|
execute *KAMAL.builder.pull
|
||||||
|
execute *KAMAL.builder.validate_image
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,54 +0,0 @@
|
|||||||
require "tempfile"
|
|
||||||
|
|
||||||
class Kamal::Cli::Env < Kamal::Cli::Base
|
|
||||||
desc "push", "Push the env file to the remote hosts"
|
|
||||||
def push
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Pushed env files"), verbosity: :debug
|
|
||||||
|
|
||||||
KAMAL.roles_on(host).each do |role|
|
|
||||||
execute *KAMAL.app(role: role, host: host).make_env_directory
|
|
||||||
upload! role.env(host).secrets_io, role.env(host).secrets_file, mode: 400
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.traefik.make_env_directory
|
|
||||||
upload! KAMAL.traefik.env.secrets_io, KAMAL.traefik.env.secrets_file, mode: 400
|
|
||||||
end
|
|
||||||
|
|
||||||
on(KAMAL.accessory_hosts) do
|
|
||||||
KAMAL.accessories_on(host).each do |accessory|
|
|
||||||
accessory_config = KAMAL.config.accessory(accessory)
|
|
||||||
execute *KAMAL.accessory(accessory).make_env_directory
|
|
||||||
upload! accessory_config.env.secrets_io, accessory_config.env.secrets_file, mode: 400
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "delete", "Delete the env file from the remote hosts"
|
|
||||||
def delete
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Deleted env files"), verbosity: :debug
|
|
||||||
|
|
||||||
KAMAL.roles_on(host).each do |role|
|
|
||||||
execute *KAMAL.app(role: role, host: host).remove_env_file
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.traefik.remove_env_file
|
|
||||||
end
|
|
||||||
|
|
||||||
on(KAMAL.accessory_hosts) do
|
|
||||||
KAMAL.accessories_on(host).each do |accessory|
|
|
||||||
accessory_config = KAMAL.config.accessory(accessory)
|
|
||||||
execute *KAMAL.accessory(accessory).remove_env_file
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
require "concurrent/ivar"
|
||||||
|
|
||||||
class Kamal::Cli::Healthcheck::Barrier
|
class Kamal::Cli::Healthcheck::Barrier
|
||||||
def initialize
|
def initialize
|
||||||
@ivar = Concurrent::IVar.new
|
@ivar = Concurrent::IVar.new
|
||||||
|
|||||||
@@ -1,26 +1,30 @@
|
|||||||
module Kamal::Cli::Healthcheck::Poller
|
module Kamal::Cli::Healthcheck::Poller
|
||||||
extend self
|
extend self
|
||||||
|
|
||||||
TRAEFIK_UPDATE_DELAY = 5
|
def wait_for_healthy(role, &block)
|
||||||
|
|
||||||
|
|
||||||
def wait_for_healthy(pause_after_ready: false, &block)
|
|
||||||
attempt = 1
|
attempt = 1
|
||||||
max_attempts = KAMAL.config.healthcheck.max_attempts
|
timeout_at = Time.now + KAMAL.config.deploy_timeout
|
||||||
|
readiness_delay = KAMAL.config.readiness_delay
|
||||||
|
|
||||||
begin
|
begin
|
||||||
case status = block.call
|
status = block.call
|
||||||
when "healthy"
|
|
||||||
sleep TRAEFIK_UPDATE_DELAY if pause_after_ready
|
if status == "running"
|
||||||
when "running" # No health check configured
|
# Wait for the readiness delay and confirm it is still running
|
||||||
sleep KAMAL.config.readiness_delay if pause_after_ready
|
if readiness_delay > 0
|
||||||
else
|
info "Container is running, waiting for readiness delay of #{readiness_delay} seconds"
|
||||||
raise Kamal::Cli::Healthcheck::Error, "container not ready (#{status})"
|
sleep readiness_delay
|
||||||
|
status = block.call
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
unless %w[ running healthy ].include?(status)
|
||||||
|
raise Kamal::Cli::Healthcheck::Error, "container not ready after #{KAMAL.config.deploy_timeout} seconds (#{status})"
|
||||||
end
|
end
|
||||||
rescue Kamal::Cli::Healthcheck::Error => e
|
rescue Kamal::Cli::Healthcheck::Error => e
|
||||||
if attempt <= max_attempts
|
time_left = timeout_at - Time.now
|
||||||
info "#{e.message}, retrying in #{attempt}s (attempt #{attempt}/#{max_attempts})..."
|
if time_left > 0
|
||||||
sleep attempt
|
sleep [ attempt, time_left ].min
|
||||||
attempt += 1
|
attempt += 1
|
||||||
retry
|
retry
|
||||||
else
|
else
|
||||||
@@ -31,31 +35,6 @@ module Kamal::Cli::Healthcheck::Poller
|
|||||||
info "Container is healthy!"
|
info "Container is healthy!"
|
||||||
end
|
end
|
||||||
|
|
||||||
def wait_for_unhealthy(pause_after_ready: false, &block)
|
|
||||||
attempt = 1
|
|
||||||
max_attempts = KAMAL.config.healthcheck.max_attempts
|
|
||||||
|
|
||||||
begin
|
|
||||||
case status = block.call
|
|
||||||
when "unhealthy"
|
|
||||||
sleep TRAEFIK_UPDATE_DELAY if pause_after_ready
|
|
||||||
else
|
|
||||||
raise Kamal::Cli::Healthcheck::Error, "container not unhealthy (#{status})"
|
|
||||||
end
|
|
||||||
rescue Kamal::Cli::Healthcheck::Error => e
|
|
||||||
if attempt <= max_attempts
|
|
||||||
info "#{e.message}, retrying in #{attempt}s (attempt #{attempt}/#{max_attempts})..."
|
|
||||||
sleep attempt
|
|
||||||
attempt += 1
|
|
||||||
retry
|
|
||||||
else
|
|
||||||
raise
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
info "Container is unhealthy!"
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
def info(message)
|
def info(message)
|
||||||
SSHKit.config.output.info(message)
|
SSHKit.config.output.info(message)
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ class Kamal::Cli::Lock < Kamal::Cli::Base
|
|||||||
def status
|
def status
|
||||||
handle_missing_lock do
|
handle_missing_lock do
|
||||||
on(KAMAL.primary_host) do
|
on(KAMAL.primary_host) do
|
||||||
execute *KAMAL.server.ensure_run_directory
|
|
||||||
puts capture_with_debug(*KAMAL.lock.status)
|
puts capture_with_debug(*KAMAL.lock.status)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -13,9 +12,10 @@ class Kamal::Cli::Lock < Kamal::Cli::Base
|
|||||||
option :message, aliases: "-m", type: :string, desc: "A lock message", required: true
|
option :message, aliases: "-m", type: :string, desc: "A lock message", required: true
|
||||||
def acquire
|
def acquire
|
||||||
message = options[:message]
|
message = options[:message]
|
||||||
|
ensure_run_directory
|
||||||
|
|
||||||
raise_if_locked do
|
raise_if_locked do
|
||||||
on(KAMAL.primary_host) do
|
on(KAMAL.primary_host) do
|
||||||
execute *KAMAL.server.ensure_run_directory
|
|
||||||
execute *KAMAL.lock.acquire(message, KAMAL.config.version), verbosity: :debug
|
execute *KAMAL.lock.acquire(message, KAMAL.config.version), verbosity: :debug
|
||||||
end
|
end
|
||||||
say "Acquired the deploy lock"
|
say "Acquired the deploy lock"
|
||||||
@@ -26,7 +26,6 @@ class Kamal::Cli::Lock < Kamal::Cli::Base
|
|||||||
def release
|
def release
|
||||||
handle_missing_lock do
|
handle_missing_lock do
|
||||||
on(KAMAL.primary_host) do
|
on(KAMAL.primary_host) do
|
||||||
execute *KAMAL.server.ensure_run_directory
|
|
||||||
execute *KAMAL.lock.release, verbosity: :debug
|
execute *KAMAL.lock.release, verbosity: :debug
|
||||||
end
|
end
|
||||||
say "Released the deploy lock"
|
say "Released the deploy lock"
|
||||||
|
|||||||
@@ -9,10 +9,6 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
say "Ensure Docker is installed...", :magenta
|
say "Ensure Docker is installed...", :magenta
|
||||||
invoke "kamal:cli:server:bootstrap", [], invoke_options
|
invoke "kamal:cli:server:bootstrap", [], invoke_options
|
||||||
|
|
||||||
say "Evaluate and push env files...", :magenta
|
|
||||||
invoke "kamal:cli:main:envify", [], invoke_options
|
|
||||||
invoke "kamal:cli:env:push", [], invoke_options
|
|
||||||
|
|
||||||
invoke "kamal:cli:accessory:boot", [ "all" ], invoke_options
|
invoke "kamal:cli:accessory:boot", [ "all" ], invoke_options
|
||||||
deploy
|
deploy
|
||||||
end
|
end
|
||||||
@@ -37,10 +33,10 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
with_lock do
|
with_lock do
|
||||||
run_hook "pre-deploy"
|
run_hook "pre-deploy", secrets: true
|
||||||
|
|
||||||
say "Ensure Traefik is running...", :magenta
|
say "Ensure kamal-proxy is running...", :magenta
|
||||||
invoke "kamal:cli:traefik:boot", [], invoke_options
|
invoke "kamal:cli:proxy:boot", [], invoke_options
|
||||||
|
|
||||||
say "Detect stale containers...", :magenta
|
say "Detect stale containers...", :magenta
|
||||||
invoke "kamal:cli:app:stale_containers", [], invoke_options.merge(stop: true)
|
invoke "kamal:cli:app:stale_containers", [], invoke_options.merge(stop: true)
|
||||||
@@ -52,10 +48,10 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
run_hook "post-deploy", runtime: runtime.round
|
run_hook "post-deploy", secrets: true, runtime: runtime.round.to_s
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "redeploy", "Deploy app to servers without bootstrapping servers, starting Traefik, pruning, and registry login"
|
desc "redeploy", "Deploy app to servers without bootstrapping servers, starting kamal-proxy, pruning, and registry login"
|
||||||
option :skip_push, aliases: "-P", type: :boolean, default: false, desc: "Skip image build and push"
|
option :skip_push, aliases: "-P", type: :boolean, default: false, desc: "Skip image build and push"
|
||||||
def redeploy
|
def redeploy
|
||||||
runtime = print_runtime do
|
runtime = print_runtime do
|
||||||
@@ -70,7 +66,7 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
with_lock do
|
with_lock do
|
||||||
run_hook "pre-deploy"
|
run_hook "pre-deploy", secrets: true
|
||||||
|
|
||||||
say "Detect stale containers...", :magenta
|
say "Detect stale containers...", :magenta
|
||||||
invoke "kamal:cli:app:stale_containers", [], invoke_options.merge(stop: true)
|
invoke "kamal:cli:app:stale_containers", [], invoke_options.merge(stop: true)
|
||||||
@@ -79,7 +75,7 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
run_hook "post-deploy", runtime: runtime.round
|
run_hook "post-deploy", secrets: true, runtime: runtime.round.to_s
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "rollback [VERSION]", "Rollback app to VERSION"
|
desc "rollback [VERSION]", "Rollback app to VERSION"
|
||||||
@@ -93,7 +89,7 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
old_version = nil
|
old_version = nil
|
||||||
|
|
||||||
if container_available?(version)
|
if container_available?(version)
|
||||||
run_hook "pre-deploy"
|
run_hook "pre-deploy", secrets: true
|
||||||
|
|
||||||
invoke "kamal:cli:app:boot", [], invoke_options.merge(version: version)
|
invoke "kamal:cli:app:boot", [], invoke_options.merge(version: version)
|
||||||
rolled_back = true
|
rolled_back = true
|
||||||
@@ -103,12 +99,12 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
run_hook "post-deploy", runtime: runtime.round if rolled_back
|
run_hook "post-deploy", secrets: true, runtime: runtime.round.to_s if rolled_back
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "details", "Show details about all containers"
|
desc "details", "Show details about all containers"
|
||||||
def details
|
def details
|
||||||
invoke "kamal:cli:traefik:details"
|
invoke "kamal:cli:proxy:details"
|
||||||
invoke "kamal:cli:app:details"
|
invoke "kamal:cli:app:details"
|
||||||
invoke "kamal:cli:accessory:details", [ "all" ]
|
invoke "kamal:cli:accessory:details", [ "all" ]
|
||||||
end
|
end
|
||||||
@@ -127,7 +123,7 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "docs", "Show Kamal documentation for configuration setting"
|
desc "docs [SECTION]", "Show Kamal configuration documentation"
|
||||||
def docs(section = nil)
|
def docs(section = nil)
|
||||||
case section
|
case section
|
||||||
when NilClass
|
when NilClass
|
||||||
@@ -152,9 +148,10 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
puts "Created configuration file in config/deploy.yml"
|
puts "Created configuration file in config/deploy.yml"
|
||||||
end
|
end
|
||||||
|
|
||||||
unless (deploy_file = Pathname.new(File.expand_path(".env"))).exist?
|
unless (secrets_file = Pathname.new(File.expand_path(".kamal/secrets"))).exist?
|
||||||
FileUtils.cp_r Pathname.new(File.expand_path("templates/template.env", __dir__)), deploy_file
|
FileUtils.mkdir_p secrets_file.dirname
|
||||||
puts "Created .env file"
|
FileUtils.cp_r Pathname.new(File.expand_path("templates/secrets", __dir__)), secrets_file
|
||||||
|
puts "Created .kamal/secrets file"
|
||||||
end
|
end
|
||||||
|
|
||||||
unless (hooks_dir = Pathname.new(File.expand_path(".kamal/hooks"))).exist?
|
unless (hooks_dir = Pathname.new(File.expand_path(".kamal/hooks"))).exist?
|
||||||
@@ -179,42 +176,50 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "envify", "Create .env by evaluating .env.erb (or .env.staging.erb -> .env.staging when using -d staging)"
|
desc "remove", "Remove kamal-proxy, app, accessories, and registry session from servers"
|
||||||
option :skip_push, aliases: "-P", type: :boolean, default: false, desc: "Skip .env file push"
|
|
||||||
def envify
|
|
||||||
if destination = options[:destination]
|
|
||||||
env_template_path = ".env.#{destination}.erb"
|
|
||||||
env_path = ".env.#{destination}"
|
|
||||||
else
|
|
||||||
env_template_path = ".env.erb"
|
|
||||||
env_path = ".env"
|
|
||||||
end
|
|
||||||
|
|
||||||
if Pathname.new(File.expand_path(env_template_path)).exist?
|
|
||||||
File.write(env_path, ERB.new(File.read(env_template_path), trim_mode: "-").result, perm: 0600)
|
|
||||||
|
|
||||||
unless options[:skip_push]
|
|
||||||
reload_envs
|
|
||||||
invoke "kamal:cli:env:push", options
|
|
||||||
end
|
|
||||||
else
|
|
||||||
puts "Skipping envify (no #{env_template_path} exist)"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "remove", "Remove Traefik, app, accessories, and registry session from servers"
|
|
||||||
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
||||||
def remove
|
def remove
|
||||||
confirming "This will remove all containers and images. Are you sure?" do
|
confirming "This will remove all containers and images. Are you sure?" do
|
||||||
with_lock do
|
with_lock do
|
||||||
invoke "kamal:cli:traefik:remove", [], options.without(:confirmed)
|
|
||||||
invoke "kamal:cli:app:remove", [], options.without(:confirmed)
|
invoke "kamal:cli:app:remove", [], options.without(:confirmed)
|
||||||
|
invoke "kamal:cli:proxy:remove", [], options.without(:confirmed)
|
||||||
invoke "kamal:cli:accessory:remove", [ "all" ], options
|
invoke "kamal:cli:accessory:remove", [ "all" ], options
|
||||||
invoke "kamal:cli:registry:logout", [], options.without(:confirmed).merge(skip_local: true)
|
invoke "kamal:cli:registry:logout", [], options.without(:confirmed).merge(skip_local: true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
desc "upgrade", "Upgrade from Kamal 1.x to 2.0"
|
||||||
|
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
||||||
|
option :rolling, type: :boolean, default: false, desc: "Upgrade one host at a time"
|
||||||
|
def upgrade
|
||||||
|
confirming "This will replace Traefik with kamal-proxy and restart all accessories" do
|
||||||
|
with_lock do
|
||||||
|
if options[:rolling]
|
||||||
|
(KAMAL.hosts | KAMAL.accessory_hosts).each do |host|
|
||||||
|
KAMAL.with_specific_hosts(host) do
|
||||||
|
say "Upgrading #{host}...", :magenta
|
||||||
|
if KAMAL.hosts.include?(host)
|
||||||
|
invoke "kamal:cli:proxy:upgrade", [], options.merge(confirmed: true, rolling: false)
|
||||||
|
reset_invocation(Kamal::Cli::Proxy)
|
||||||
|
end
|
||||||
|
if KAMAL.accessory_hosts.include?(host)
|
||||||
|
invoke "kamal:cli:accessory:upgrade", [ "all" ], options.merge(confirmed: true, rolling: false)
|
||||||
|
reset_invocation(Kamal::Cli::Accessory)
|
||||||
|
end
|
||||||
|
say "Upgraded #{host}", :magenta
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
say "Upgrading all hosts...", :magenta
|
||||||
|
invoke "kamal:cli:proxy:upgrade", [], options.merge(confirmed: true)
|
||||||
|
invoke "kamal:cli:accessory:upgrade", [ "all" ], options.merge(confirmed: true)
|
||||||
|
say "Upgraded all hosts", :magenta
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
desc "version", "Show Kamal version"
|
desc "version", "Show Kamal version"
|
||||||
def version
|
def version
|
||||||
puts Kamal::VERSION
|
puts Kamal::VERSION
|
||||||
@@ -229,24 +234,24 @@ class Kamal::Cli::Main < Kamal::Cli::Base
|
|||||||
desc "build", "Build application image"
|
desc "build", "Build application image"
|
||||||
subcommand "build", Kamal::Cli::Build
|
subcommand "build", Kamal::Cli::Build
|
||||||
|
|
||||||
desc "env", "Manage environment files"
|
|
||||||
subcommand "env", Kamal::Cli::Env
|
|
||||||
|
|
||||||
desc "lock", "Manage the deploy lock"
|
desc "lock", "Manage the deploy lock"
|
||||||
subcommand "lock", Kamal::Cli::Lock
|
subcommand "lock", Kamal::Cli::Lock
|
||||||
|
|
||||||
|
desc "proxy", "Manage kamal-proxy"
|
||||||
|
subcommand "proxy", Kamal::Cli::Proxy
|
||||||
|
|
||||||
desc "prune", "Prune old application images and containers"
|
desc "prune", "Prune old application images and containers"
|
||||||
subcommand "prune", Kamal::Cli::Prune
|
subcommand "prune", Kamal::Cli::Prune
|
||||||
|
|
||||||
desc "registry", "Login and -out of the image registry"
|
desc "registry", "Login and -out of the image registry"
|
||||||
subcommand "registry", Kamal::Cli::Registry
|
subcommand "registry", Kamal::Cli::Registry
|
||||||
|
|
||||||
|
desc "secrets", "Helpers for extracting secrets"
|
||||||
|
subcommand "secrets", Kamal::Cli::Secrets
|
||||||
|
|
||||||
desc "server", "Bootstrap servers with curl and Docker"
|
desc "server", "Bootstrap servers with curl and Docker"
|
||||||
subcommand "server", Kamal::Cli::Server
|
subcommand "server", Kamal::Cli::Server
|
||||||
|
|
||||||
desc "traefik", "Manage Traefik load balancer"
|
|
||||||
subcommand "traefik", Kamal::Cli::Traefik
|
|
||||||
|
|
||||||
private
|
private
|
||||||
def container_available?(version)
|
def container_available?(version)
|
||||||
begin
|
begin
|
||||||
|
|||||||
215
lib/kamal/cli/proxy.rb
Normal file
215
lib/kamal/cli/proxy.rb
Normal file
@@ -0,0 +1,215 @@
|
|||||||
|
class Kamal::Cli::Proxy < Kamal::Cli::Base
|
||||||
|
desc "boot", "Boot proxy on servers"
|
||||||
|
def boot
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.hosts) do |host|
|
||||||
|
execute *KAMAL.docker.create_network
|
||||||
|
rescue SSHKit::Command::Failed => e
|
||||||
|
raise unless e.message.include?("already exists")
|
||||||
|
end
|
||||||
|
|
||||||
|
on(KAMAL.proxy_hosts) do |host|
|
||||||
|
execute *KAMAL.registry.login
|
||||||
|
|
||||||
|
version = capture_with_info(*KAMAL.proxy.version).strip.presence
|
||||||
|
|
||||||
|
if version && Kamal::Utils.older_version?(version, Kamal::Configuration::PROXY_MINIMUM_VERSION)
|
||||||
|
raise "kamal-proxy version #{version} is too old, please run `kamal proxy reboot` to update to at least #{Kamal::Configuration::PROXY_MINIMUM_VERSION}"
|
||||||
|
end
|
||||||
|
execute *KAMAL.proxy.start_or_run
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "reboot", "Reboot proxy on servers (stop container, remove container, start new container)"
|
||||||
|
option :rolling, type: :boolean, default: false, desc: "Reboot proxy on hosts in sequence, rather than in parallel"
|
||||||
|
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
||||||
|
def reboot
|
||||||
|
confirming "This will cause a brief outage on each host. Are you sure?" do
|
||||||
|
with_lock do
|
||||||
|
host_groups = options[:rolling] ? KAMAL.proxy_hosts : [ KAMAL.proxy_hosts ]
|
||||||
|
host_groups.each do |hosts|
|
||||||
|
host_list = Array(hosts).join(",")
|
||||||
|
run_hook "pre-proxy-reboot", hosts: host_list
|
||||||
|
on(hosts) do |host|
|
||||||
|
execute *KAMAL.auditor.record("Rebooted proxy"), verbosity: :debug
|
||||||
|
execute *KAMAL.registry.login
|
||||||
|
|
||||||
|
"Stopping and removing Traefik on #{host}, if running..."
|
||||||
|
execute *KAMAL.proxy.cleanup_traefik
|
||||||
|
|
||||||
|
"Stopping and removing kamal-proxy on #{host}, if running..."
|
||||||
|
execute *KAMAL.proxy.stop, raise_on_non_zero_exit: false
|
||||||
|
execute *KAMAL.proxy.remove_container
|
||||||
|
|
||||||
|
execute *KAMAL.proxy.run
|
||||||
|
|
||||||
|
KAMAL.roles_on(host).select(&:running_proxy?).each do |role|
|
||||||
|
app = KAMAL.app(role: role, host: host)
|
||||||
|
|
||||||
|
version = capture_with_info(*app.current_running_version, raise_on_non_zero_exit: false).strip
|
||||||
|
endpoint = capture_with_info(*app.container_id_for_version(version)).strip
|
||||||
|
|
||||||
|
if endpoint.present?
|
||||||
|
info "Deploying #{endpoint} for role `#{role}` on #{host}..."
|
||||||
|
execute *app.deploy(target: endpoint)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
run_hook "post-proxy-reboot", hosts: host_list
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "upgrade", "Upgrade to kamal-proxy on servers (stop container, remove container, start new container, reboot app)", hide: true
|
||||||
|
option :rolling, type: :boolean, default: false, desc: "Reboot proxy on hosts in sequence, rather than in parallel"
|
||||||
|
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
||||||
|
def upgrade
|
||||||
|
invoke_options = { "version" => KAMAL.config.latest_tag }.merge(options)
|
||||||
|
|
||||||
|
confirming "This will cause a brief outage on each host. Are you sure?" do
|
||||||
|
host_groups = options[:rolling] ? KAMAL.hosts : [ KAMAL.hosts ]
|
||||||
|
host_groups.each do |hosts|
|
||||||
|
host_list = Array(hosts).join(",")
|
||||||
|
say "Upgrading proxy on #{host_list}...", :magenta
|
||||||
|
run_hook "pre-proxy-reboot", hosts: host_list
|
||||||
|
on(hosts) do |host|
|
||||||
|
execute *KAMAL.auditor.record("Rebooted proxy"), verbosity: :debug
|
||||||
|
execute *KAMAL.registry.login
|
||||||
|
|
||||||
|
"Stopping and removing Traefik on #{host}, if running..."
|
||||||
|
execute *KAMAL.proxy.cleanup_traefik
|
||||||
|
|
||||||
|
"Stopping and removing kamal-proxy on #{host}, if running..."
|
||||||
|
execute *KAMAL.proxy.stop, raise_on_non_zero_exit: false
|
||||||
|
execute *KAMAL.proxy.remove_container
|
||||||
|
execute *KAMAL.proxy.remove_image
|
||||||
|
end
|
||||||
|
|
||||||
|
KAMAL.with_specific_hosts(hosts) do
|
||||||
|
invoke "kamal:cli:proxy:boot", [], invoke_options
|
||||||
|
reset_invocation(Kamal::Cli::Proxy)
|
||||||
|
invoke "kamal:cli:app:boot", [], invoke_options
|
||||||
|
reset_invocation(Kamal::Cli::App)
|
||||||
|
invoke "kamal:cli:prune:all", [], invoke_options
|
||||||
|
reset_invocation(Kamal::Cli::Prune)
|
||||||
|
end
|
||||||
|
|
||||||
|
run_hook "post-proxy-reboot", hosts: host_list
|
||||||
|
say "Upgraded proxy on #{host_list}", :magenta
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "start", "Start existing proxy container on servers"
|
||||||
|
def start
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.proxy_hosts) do |host|
|
||||||
|
execute *KAMAL.auditor.record("Started proxy"), verbosity: :debug
|
||||||
|
execute *KAMAL.proxy.start
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "stop", "Stop existing proxy container on servers"
|
||||||
|
def stop
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.proxy_hosts) do |host|
|
||||||
|
execute *KAMAL.auditor.record("Stopped proxy"), verbosity: :debug
|
||||||
|
execute *KAMAL.proxy.stop, raise_on_non_zero_exit: false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "restart", "Restart existing proxy container on servers"
|
||||||
|
def restart
|
||||||
|
with_lock do
|
||||||
|
stop
|
||||||
|
start
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "details", "Show details about proxy container from servers"
|
||||||
|
def details
|
||||||
|
on(KAMAL.proxy_hosts) { |host| puts_by_host host, capture_with_info(*KAMAL.proxy.info), type: "Proxy" }
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "logs", "Show log lines from proxy on servers"
|
||||||
|
option :since, aliases: "-s", desc: "Show logs since timestamp (e.g. 2013-01-02T13:23:37Z) or relative (e.g. 42m for 42 minutes)"
|
||||||
|
option :lines, type: :numeric, aliases: "-n", desc: "Number of log lines to pull from each server"
|
||||||
|
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
||||||
|
option :follow, aliases: "-f", desc: "Follow logs on primary server (or specific host set by --hosts)"
|
||||||
|
option :skip_timestamps, type: :boolean, aliases: "-T", desc: "Skip appending timestamps to logging output"
|
||||||
|
def logs
|
||||||
|
grep = options[:grep]
|
||||||
|
timestamps = !options[:skip_timestamps]
|
||||||
|
|
||||||
|
if options[:follow]
|
||||||
|
run_locally do
|
||||||
|
info "Following logs on #{KAMAL.primary_host}..."
|
||||||
|
info KAMAL.proxy.follow_logs(host: KAMAL.primary_host, timestamps: timestamps, grep: grep)
|
||||||
|
exec KAMAL.proxy.follow_logs(host: KAMAL.primary_host, timestamps: timestamps, grep: grep)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
since = options[:since]
|
||||||
|
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
||||||
|
|
||||||
|
on(KAMAL.proxy_hosts) do |host|
|
||||||
|
puts_by_host host, capture(*KAMAL.proxy.logs(timestamps: timestamps, since: since, lines: lines, grep: grep)), type: "Proxy"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "remove", "Remove proxy container and image from servers"
|
||||||
|
option :force, type: :boolean, default: false, desc: "Force removing proxy when apps are still installed"
|
||||||
|
def remove
|
||||||
|
with_lock do
|
||||||
|
if removal_allowed?(options[:force])
|
||||||
|
stop
|
||||||
|
remove_container
|
||||||
|
remove_image
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "remove_container", "Remove proxy container from servers", hide: true
|
||||||
|
def remove_container
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.proxy_hosts) do
|
||||||
|
execute *KAMAL.auditor.record("Removed proxy container"), verbosity: :debug
|
||||||
|
execute *KAMAL.proxy.remove_container
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "remove_image", "Remove proxy image from servers", hide: true
|
||||||
|
def remove_image
|
||||||
|
with_lock do
|
||||||
|
on(KAMAL.proxy_hosts) do
|
||||||
|
execute *KAMAL.auditor.record("Removed proxy image"), verbosity: :debug
|
||||||
|
execute *KAMAL.proxy.remove_image
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def removal_allowed?(force)
|
||||||
|
on(KAMAL.proxy_hosts) do |host|
|
||||||
|
app_count = capture_with_info(*KAMAL.server.app_directory_count).chomp.to_i
|
||||||
|
raise "The are other applications installed on #{host}" if app_count > 0
|
||||||
|
end
|
||||||
|
|
||||||
|
true
|
||||||
|
rescue SSHKit::Runner::ExecuteError => e
|
||||||
|
raise unless e.message.include?("The are other applications installed on")
|
||||||
|
|
||||||
|
if force
|
||||||
|
say "Forcing, so removing the proxy, even though other apps are installed", :magenta
|
||||||
|
else
|
||||||
|
say "Not removing the proxy, as other apps are installed, ignore this check with kamal proxy remove --force", :magenta
|
||||||
|
end
|
||||||
|
|
||||||
|
force
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -28,7 +28,6 @@ class Kamal::Cli::Prune < Kamal::Cli::Base
|
|||||||
on(KAMAL.hosts) do
|
on(KAMAL.hosts) do
|
||||||
execute *KAMAL.auditor.record("Pruned containers"), verbosity: :debug
|
execute *KAMAL.auditor.record("Pruned containers"), verbosity: :debug
|
||||||
execute *KAMAL.prune.app_containers(retain: retain)
|
execute *KAMAL.prune.app_containers(retain: retain)
|
||||||
execute *KAMAL.prune.healthcheck_containers
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
36
lib/kamal/cli/secrets.rb
Normal file
36
lib/kamal/cli/secrets.rb
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
class Kamal::Cli::Secrets < Kamal::Cli::Base
|
||||||
|
desc "fetch [SECRETS...]", "Fetch secrets from a vault"
|
||||||
|
option :adapter, type: :string, aliases: "-a", required: true, desc: "Which vault adapter to use"
|
||||||
|
option :account, type: :string, required: true, desc: "The account identifier or username"
|
||||||
|
option :from, type: :string, required: false, desc: "A vault or folder to fetch the secrets from"
|
||||||
|
option :inline, type: :boolean, required: false, hidden: true
|
||||||
|
def fetch(*secrets)
|
||||||
|
results = adapter(options[:adapter]).fetch(secrets, **options.slice(:account, :from).symbolize_keys)
|
||||||
|
|
||||||
|
return_or_puts JSON.dump(results).shellescape, inline: options[:inline]
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "extract", "Extract a single secret from the results of a fetch call"
|
||||||
|
option :inline, type: :boolean, required: false, hidden: true
|
||||||
|
def extract(name, secrets)
|
||||||
|
parsed_secrets = JSON.parse(secrets)
|
||||||
|
value = parsed_secrets[name] || parsed_secrets.find { |k, v| k.end_with?("/#{name}") }&.last
|
||||||
|
|
||||||
|
raise "Could not find secret #{name}" if value.nil?
|
||||||
|
|
||||||
|
return_or_puts value, inline: options[:inline]
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def adapter(adapter)
|
||||||
|
Kamal::Secrets::Adapters.lookup(adapter)
|
||||||
|
end
|
||||||
|
|
||||||
|
def return_or_puts(value, inline: nil)
|
||||||
|
if inline
|
||||||
|
value
|
||||||
|
else
|
||||||
|
puts value
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,7 +1,8 @@
|
|||||||
class Kamal::Cli::Server < Kamal::Cli::Base
|
class Kamal::Cli::Server < Kamal::Cli::Base
|
||||||
desc "exec", "Run a custom command on the server (use --help to show options)"
|
desc "exec", "Run a custom command on the server (use --help to show options)"
|
||||||
option :interactive, type: :boolean, aliases: "-i", default: false, desc: "Run the command interactively (use for console/bash)"
|
option :interactive, type: :boolean, aliases: "-i", default: false, desc: "Run the command interactively (use for console/bash)"
|
||||||
def exec(cmd)
|
def exec(*cmd)
|
||||||
|
cmd = Kamal::Utils.join_commands(cmd)
|
||||||
hosts = KAMAL.hosts | KAMAL.accessory_hosts
|
hosts = KAMAL.hosts | KAMAL.accessory_hosts
|
||||||
|
|
||||||
case
|
case
|
||||||
@@ -35,8 +36,6 @@ class Kamal::Cli::Server < Kamal::Cli::Base
|
|||||||
missing << host
|
missing << host
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
execute(*KAMAL.server.ensure_run_directory)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
if missing.any?
|
if missing.any?
|
||||||
|
|||||||
@@ -18,6 +18,10 @@ registry:
|
|||||||
password:
|
password:
|
||||||
- KAMAL_REGISTRY_PASSWORD
|
- KAMAL_REGISTRY_PASSWORD
|
||||||
|
|
||||||
|
# Configure builder setup.
|
||||||
|
builder:
|
||||||
|
arch: amd64
|
||||||
|
|
||||||
# Inject ENV variables into containers (secrets come from .env).
|
# Inject ENV variables into containers (secrets come from .env).
|
||||||
# Remember to run `kamal env push` after making changes!
|
# Remember to run `kamal env push` after making changes!
|
||||||
# env:
|
# env:
|
||||||
@@ -30,16 +34,6 @@ registry:
|
|||||||
# ssh:
|
# ssh:
|
||||||
# user: app
|
# user: app
|
||||||
|
|
||||||
# Configure builder setup.
|
|
||||||
# builder:
|
|
||||||
# args:
|
|
||||||
# RUBY_VERSION: 3.2.0
|
|
||||||
# secrets:
|
|
||||||
# - GITHUB_TOKEN
|
|
||||||
# remote:
|
|
||||||
# arch: amd64
|
|
||||||
# host: ssh://app@192.168.0.1
|
|
||||||
|
|
||||||
# Use accessory services (secrets come from .env).
|
# Use accessory services (secrets come from .env).
|
||||||
# accessories:
|
# accessories:
|
||||||
# db:
|
# db:
|
||||||
@@ -63,17 +57,6 @@ registry:
|
|||||||
# directories:
|
# directories:
|
||||||
# - data:/data
|
# - data:/data
|
||||||
|
|
||||||
# Configure custom arguments for Traefik. Be sure to reboot traefik when you modify it.
|
|
||||||
# traefik:
|
|
||||||
# args:
|
|
||||||
# accesslog: true
|
|
||||||
# accesslog.format: json
|
|
||||||
|
|
||||||
# Configure a custom healthcheck (default is /up on port 3000)
|
|
||||||
# healthcheck:
|
|
||||||
# path: /healthz
|
|
||||||
# port: 4000
|
|
||||||
|
|
||||||
# Bridge fingerprinted assets, like JS and CSS, between versions to avoid
|
# Bridge fingerprinted assets, like JS and CSS, between versions to avoid
|
||||||
# hitting 404 on in-flight requests. Combines all files from new and old
|
# hitting 404 on in-flight requests. Combines all files from new and old
|
||||||
# version inside the asset_path.
|
# version inside the asset_path.
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
#!/bin/sh
|
#!/usr/bin/env ruby
|
||||||
|
|
||||||
# A sample docker-setup hook
|
# A sample docker-setup hook
|
||||||
#
|
#
|
||||||
# Sets up a Docker network which can then be used by the application’s containers
|
# Sets up a Docker network on defined hosts which can then be used by the application’s containers
|
||||||
|
|
||||||
ssh user@example.com docker network create kamal
|
hosts = ENV["KAMAL_HOSTS"].split(",")
|
||||||
|
|
||||||
|
hosts.each do |ip|
|
||||||
|
destination = "root@#{ip}"
|
||||||
|
puts "Creating a Docker network \"kamal\" on #{destination}"
|
||||||
|
`ssh #{destination} docker network create kamal`
|
||||||
|
end
|
||||||
|
|||||||
3
lib/kamal/cli/templates/sample_hooks/post-proxy-reboot.sample
Executable file
3
lib/kamal/cli/templates/sample_hooks/post-proxy-reboot.sample
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Rebooted kamal-proxy on $KAMAL_HOSTS"
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
echo "Rebooted Traefik on $KAMAL_HOSTS"
|
|
||||||
16
lib/kamal/cli/templates/secrets
Normal file
16
lib/kamal/cli/templates/secrets
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
# WARNING: Avoid adding secrets directly to this file
|
||||||
|
# If you must, then add `.kamal/secrets*` to your .gitignore file
|
||||||
|
|
||||||
|
# Option 1: Read secrets from the environment
|
||||||
|
KAMAL_REGISTRY_PASSWORD=$KAMAL_REGISTRY_PASSWORD
|
||||||
|
|
||||||
|
# Option 2: Read secrets via a command
|
||||||
|
# RAILS_MASTER_KEY=$(cat config/master.key)
|
||||||
|
|
||||||
|
# Option 3: Read secrets via kamal secrets helpers
|
||||||
|
# These will handle logging in and fetching the secrets in as few calls as possible
|
||||||
|
# There are adapters for 1Password, LastPass + Bitwarden
|
||||||
|
#
|
||||||
|
# SECRETS=$(kamal secrets fetch --adapter 1password --account my-account --from MyVault/MyItem KAMAL_REGISTRY_PASSWORD RAILS_MASTER_KEY)
|
||||||
|
# KAMAL_REGISTRY_PASSWORD=$(kamal secrets extract KAMAL_REGISTRY_PASSWORD $SECRETS)
|
||||||
|
# RAILS_MASTER_KEY=$(kamal secrets extract RAILS_MASTER_KEY $SECRETS)
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
KAMAL_REGISTRY_PASSWORD=change-this
|
|
||||||
RAILS_MASTER_KEY=another-env
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
class Kamal::Cli::Traefik < Kamal::Cli::Base
|
|
||||||
desc "boot", "Boot Traefik on servers"
|
|
||||||
def boot
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.registry.login
|
|
||||||
execute *KAMAL.traefik.start_or_run
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "reboot", "Reboot Traefik on servers (stop container, remove container, start new container)"
|
|
||||||
option :rolling, type: :boolean, default: false, desc: "Reboot traefik on hosts in sequence, rather than in parallel"
|
|
||||||
option :confirmed, aliases: "-y", type: :boolean, default: false, desc: "Proceed without confirmation question"
|
|
||||||
def reboot
|
|
||||||
confirming "This will cause a brief outage on each host. Are you sure?" do
|
|
||||||
with_lock do
|
|
||||||
host_groups = options[:rolling] ? KAMAL.traefik_hosts : [ KAMAL.traefik_hosts ]
|
|
||||||
host_groups.each do |hosts|
|
|
||||||
host_list = Array(hosts).join(",")
|
|
||||||
run_hook "pre-traefik-reboot", hosts: host_list
|
|
||||||
on(hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Rebooted traefik"), verbosity: :debug
|
|
||||||
execute *KAMAL.registry.login
|
|
||||||
execute *KAMAL.traefik.stop, raise_on_non_zero_exit: false
|
|
||||||
execute *KAMAL.traefik.remove_container
|
|
||||||
execute *KAMAL.traefik.run
|
|
||||||
end
|
|
||||||
run_hook "post-traefik-reboot", hosts: host_list
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "start", "Start existing Traefik container on servers"
|
|
||||||
def start
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Started traefik"), verbosity: :debug
|
|
||||||
execute *KAMAL.traefik.start
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "stop", "Stop existing Traefik container on servers"
|
|
||||||
def stop
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Stopped traefik"), verbosity: :debug
|
|
||||||
execute *KAMAL.traefik.stop, raise_on_non_zero_exit: false
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "restart", "Restart existing Traefik container on servers"
|
|
||||||
def restart
|
|
||||||
with_lock do
|
|
||||||
stop
|
|
||||||
start
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "details", "Show details about Traefik container from servers"
|
|
||||||
def details
|
|
||||||
on(KAMAL.traefik_hosts) { |host| puts_by_host host, capture_with_info(*KAMAL.traefik.info), type: "Traefik" }
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "logs", "Show log lines from Traefik on servers"
|
|
||||||
option :since, aliases: "-s", desc: "Show logs since timestamp (e.g. 2013-01-02T13:23:37Z) or relative (e.g. 42m for 42 minutes)"
|
|
||||||
option :lines, type: :numeric, aliases: "-n", desc: "Number of log lines to pull from each server"
|
|
||||||
option :grep, aliases: "-g", desc: "Show lines with grep match only (use this to fetch specific requests by id)"
|
|
||||||
option :grep_options, aliases: "-o", desc: "Additional options supplied to grep"
|
|
||||||
option :follow, aliases: "-f", desc: "Follow logs on primary server (or specific host set by --hosts)"
|
|
||||||
def logs
|
|
||||||
grep = options[:grep]
|
|
||||||
grep_options = options[:grep_options]
|
|
||||||
|
|
||||||
if options[:follow]
|
|
||||||
run_locally do
|
|
||||||
info "Following logs on #{KAMAL.primary_host}..."
|
|
||||||
info KAMAL.traefik.follow_logs(host: KAMAL.primary_host, grep: grep, grep_options: grep_options)
|
|
||||||
exec KAMAL.traefik.follow_logs(host: KAMAL.primary_host, grep: grep, grep_options: grep_options)
|
|
||||||
end
|
|
||||||
else
|
|
||||||
since = options[:since]
|
|
||||||
lines = options[:lines].presence || ((since || grep) ? nil : 100) # Default to 100 lines if since or grep isn't set
|
|
||||||
|
|
||||||
on(KAMAL.traefik_hosts) do |host|
|
|
||||||
puts_by_host host, capture(*KAMAL.traefik.logs(since: since, lines: lines, grep: grep, grep_options: grep_options)), type: "Traefik"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "remove", "Remove Traefik container and image from servers"
|
|
||||||
def remove
|
|
||||||
with_lock do
|
|
||||||
stop
|
|
||||||
remove_container
|
|
||||||
remove_image
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "remove_container", "Remove Traefik container from servers", hide: true
|
|
||||||
def remove_container
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Removed traefik container"), verbosity: :debug
|
|
||||||
execute *KAMAL.traefik.remove_container
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
desc "remove_image", "Remove Traefik image from servers", hide: true
|
|
||||||
def remove_image
|
|
||||||
with_lock do
|
|
||||||
on(KAMAL.traefik_hosts) do
|
|
||||||
execute *KAMAL.auditor.record("Removed traefik image"), verbosity: :debug
|
|
||||||
execute *KAMAL.traefik.remove_image
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,9 +1,10 @@
|
|||||||
require "active_support/core_ext/enumerable"
|
require "active_support/core_ext/enumerable"
|
||||||
require "active_support/core_ext/module/delegation"
|
require "active_support/core_ext/module/delegation"
|
||||||
|
require "active_support/core_ext/object/blank"
|
||||||
|
|
||||||
class Kamal::Commander
|
class Kamal::Commander
|
||||||
attr_accessor :verbosity, :holding_lock, :connected
|
attr_accessor :verbosity, :holding_lock, :connected
|
||||||
delegate :hosts, :roles, :primary_host, :primary_role, :roles_on, :traefik_hosts, :accessory_hosts, to: :specifics
|
delegate :hosts, :roles, :primary_host, :primary_role, :roles_on, :proxy_hosts, :accessory_hosts, to: :specifics
|
||||||
|
|
||||||
def initialize
|
def initialize
|
||||||
self.verbosity = :info
|
self.verbosity = :info
|
||||||
@@ -23,11 +24,19 @@ class Kamal::Commander
|
|||||||
@config, @config_kwargs = nil, kwargs
|
@config, @config_kwargs = nil, kwargs
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def configured?
|
||||||
|
@config || @config_kwargs
|
||||||
|
end
|
||||||
|
|
||||||
attr_reader :specific_roles, :specific_hosts
|
attr_reader :specific_roles, :specific_hosts
|
||||||
|
|
||||||
def specific_primary!
|
def specific_primary!
|
||||||
@specifics = nil
|
@specifics = nil
|
||||||
self.specific_hosts = [ config.primary_host ]
|
if specific_roles.present?
|
||||||
|
self.specific_hosts = [ specific_roles.first.primary_host ]
|
||||||
|
else
|
||||||
|
self.specific_hosts = [ config.primary_host ]
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def specific_roles=(role_names)
|
def specific_roles=(role_names)
|
||||||
@@ -56,6 +65,13 @@ class Kamal::Commander
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def with_specific_hosts(hosts)
|
||||||
|
original_hosts, self.specific_hosts = specific_hosts, hosts
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
self.specific_hosts = original_hosts
|
||||||
|
end
|
||||||
|
|
||||||
def accessory_names
|
def accessory_names
|
||||||
config.accessories&.collect(&:name) || []
|
config.accessories&.collect(&:name) || []
|
||||||
end
|
end
|
||||||
@@ -85,10 +101,6 @@ class Kamal::Commander
|
|||||||
@docker ||= Kamal::Commands::Docker.new(config)
|
@docker ||= Kamal::Commands::Docker.new(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
def healthcheck
|
|
||||||
@healthcheck ||= Kamal::Commands::Healthcheck.new(config)
|
|
||||||
end
|
|
||||||
|
|
||||||
def hook
|
def hook
|
||||||
@hook ||= Kamal::Commands::Hook.new(config)
|
@hook ||= Kamal::Commands::Hook.new(config)
|
||||||
end
|
end
|
||||||
@@ -97,6 +109,10 @@ class Kamal::Commander
|
|||||||
@lock ||= Kamal::Commands::Lock.new(config)
|
@lock ||= Kamal::Commands::Lock.new(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def proxy
|
||||||
|
@proxy ||= Kamal::Commands::Proxy.new(config)
|
||||||
|
end
|
||||||
|
|
||||||
def prune
|
def prune
|
||||||
@prune ||= Kamal::Commands::Prune.new(config)
|
@prune ||= Kamal::Commands::Prune.new(config)
|
||||||
end
|
end
|
||||||
@@ -109,8 +125,8 @@ class Kamal::Commander
|
|||||||
@server ||= Kamal::Commands::Server.new(config)
|
@server ||= Kamal::Commands::Server.new(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik
|
def alias(name)
|
||||||
@traefik ||= Kamal::Commands::Traefik.new(config)
|
config.aliases[name]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -18,12 +18,12 @@ class Kamal::Commander::Specifics
|
|||||||
roles.select { |role| role.hosts.include?(host.to_s) }
|
roles.select { |role| role.hosts.include?(host.to_s) }
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik_hosts
|
def proxy_hosts
|
||||||
config.traefik_hosts & specified_hosts
|
config.proxy_hosts & specified_hosts
|
||||||
end
|
end
|
||||||
|
|
||||||
def accessory_hosts
|
def accessory_hosts
|
||||||
specific_hosts || config.accessories.flat_map(&:hosts)
|
config.accessories.flat_map(&:hosts) & specified_hosts
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
class Kamal::Commands::Accessory < Kamal::Commands::Base
|
class Kamal::Commands::Accessory < Kamal::Commands::Base
|
||||||
attr_reader :accessory_config
|
attr_reader :accessory_config
|
||||||
delegate :service_name, :image, :hosts, :port, :files, :directories, :cmd,
|
delegate :service_name, :image, :hosts, :port, :files, :directories, :cmd,
|
||||||
:publish_args, :env_args, :volume_args, :label_args, :option_args, to: :accessory_config
|
:publish_args, :env_args, :volume_args, :label_args, :option_args,
|
||||||
|
:secrets_io, :secrets_path, :env_directory,
|
||||||
|
to: :accessory_config
|
||||||
|
|
||||||
def initialize(config, name:)
|
def initialize(config, name:)
|
||||||
super(config)
|
super(config)
|
||||||
@@ -13,6 +15,7 @@ class Kamal::Commands::Accessory < Kamal::Commands::Base
|
|||||||
"--name", service_name,
|
"--name", service_name,
|
||||||
"--detach",
|
"--detach",
|
||||||
"--restart", "unless-stopped",
|
"--restart", "unless-stopped",
|
||||||
|
"--network", "kamal",
|
||||||
*config.logging_args,
|
*config.logging_args,
|
||||||
*publish_args,
|
*publish_args,
|
||||||
*env_args,
|
*env_args,
|
||||||
@@ -36,16 +39,16 @@ class Kamal::Commands::Accessory < Kamal::Commands::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def logs(since: nil, lines: nil, grep: nil, grep_options: nil)
|
def logs(timestamps: true, since: nil, lines: nil, grep: nil, grep_options: nil)
|
||||||
pipe \
|
pipe \
|
||||||
docker(:logs, service_name, (" --since #{since}" if since), (" --tail #{lines}" if lines), "--timestamps", "2>&1"),
|
docker(:logs, service_name, (" --since #{since}" if since), (" --tail #{lines}" if lines), ("--timestamps" if timestamps), "2>&1"),
|
||||||
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
||||||
end
|
end
|
||||||
|
|
||||||
def follow_logs(grep: nil, grep_options: nil)
|
def follow_logs(timestamps: true, grep: nil, grep_options: nil)
|
||||||
run_over_ssh \
|
run_over_ssh \
|
||||||
pipe \
|
pipe \
|
||||||
docker(:logs, service_name, "--timestamps", "--tail", "10", "--follow", "2>&1"),
|
docker(:logs, service_name, ("--timestamps" if timestamps), "--tail", "10", "--follow", "2>&1"),
|
||||||
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -61,6 +64,7 @@ class Kamal::Commands::Accessory < Kamal::Commands::Base
|
|||||||
docker :run,
|
docker :run,
|
||||||
("-it" if interactive),
|
("-it" if interactive),
|
||||||
"--rm",
|
"--rm",
|
||||||
|
"--network", "kamal",
|
||||||
*env_args,
|
*env_args,
|
||||||
*volume_args,
|
*volume_args,
|
||||||
image,
|
image,
|
||||||
@@ -98,12 +102,8 @@ class Kamal::Commands::Accessory < Kamal::Commands::Base
|
|||||||
docker :image, :rm, "--force", image
|
docker :image, :rm, "--force", image
|
||||||
end
|
end
|
||||||
|
|
||||||
def make_env_directory
|
def ensure_env_directory
|
||||||
make_directory accessory_config.env.secrets_directory
|
make_directory env_directory
|
||||||
end
|
|
||||||
|
|
||||||
def remove_env_file
|
|
||||||
[ :rm, "-f", accessory_config.env.secrets_file ]
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
class Kamal::Commands::App < Kamal::Commands::Base
|
class Kamal::Commands::App < Kamal::Commands::Base
|
||||||
include Assets, Containers, Cord, Execution, Images, Logging
|
include Assets, Containers, Execution, Images, Logging, Proxy
|
||||||
|
|
||||||
ACTIVE_DOCKER_STATUSES = [ :running, :restarting ]
|
ACTIVE_DOCKER_STATUSES = [ :running, :restarting ]
|
||||||
|
|
||||||
attr_reader :role, :host
|
attr_reader :role, :host
|
||||||
|
|
||||||
|
delegate :container_name, to: :role
|
||||||
|
|
||||||
def initialize(config, role: nil, host: nil)
|
def initialize(config, role: nil, host: nil)
|
||||||
super(config)
|
super(config)
|
||||||
@role = role
|
@role = role
|
||||||
@@ -16,11 +18,11 @@ class Kamal::Commands::App < Kamal::Commands::Base
|
|||||||
"--detach",
|
"--detach",
|
||||||
"--restart unless-stopped",
|
"--restart unless-stopped",
|
||||||
"--name", container_name,
|
"--name", container_name,
|
||||||
|
"--network", "kamal",
|
||||||
*([ "--hostname", hostname ] if hostname),
|
*([ "--hostname", hostname ] if hostname),
|
||||||
"-e", "KAMAL_CONTAINER_NAME=\"#{container_name}\"",
|
"-e", "KAMAL_CONTAINER_NAME=\"#{container_name}\"",
|
||||||
"-e", "KAMAL_VERSION=\"#{config.version}\"",
|
"-e", "KAMAL_VERSION=\"#{config.version}\"",
|
||||||
*role.env_args(host),
|
*role.env_args(host),
|
||||||
*role.health_check_args,
|
|
||||||
*role.logging_args,
|
*role.logging_args,
|
||||||
*config.volume_args,
|
*config.volume_args,
|
||||||
*role.asset_volume_args,
|
*role.asset_volume_args,
|
||||||
@@ -41,7 +43,7 @@ class Kamal::Commands::App < Kamal::Commands::Base
|
|||||||
def stop(version: nil)
|
def stop(version: nil)
|
||||||
pipe \
|
pipe \
|
||||||
version ? container_id_for_version(version) : current_running_container_id,
|
version ? container_id_for_version(version) : current_running_container_id,
|
||||||
xargs(config.stop_wait_time ? docker(:stop, "-t", config.stop_wait_time) : docker(:stop))
|
xargs(docker(:stop, *role.stop_args))
|
||||||
end
|
end
|
||||||
|
|
||||||
def info
|
def info
|
||||||
@@ -69,21 +71,11 @@ class Kamal::Commands::App < Kamal::Commands::Base
|
|||||||
extract_version_from_name
|
extract_version_from_name
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def ensure_env_directory
|
||||||
def make_env_directory
|
make_directory role.env_directory
|
||||||
make_directory role.env(host).secrets_directory
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def remove_env_file
|
|
||||||
[ :rm, "-f", role.env(host).secrets_file ]
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
private
|
private
|
||||||
def container_name(version = nil)
|
|
||||||
[ role.container_prefix, version || config.version ].compact.join("-")
|
|
||||||
end
|
|
||||||
|
|
||||||
def latest_image_id
|
def latest_image_id
|
||||||
docker :image, :ls, *argumentize("--filter", "reference=#{config.latest_image}"), "--format", "'{{.ID}}'"
|
docker :image, :ls, *argumentize("--filter", "reference=#{config.latest_image}"), "--format", "'{{.ID}}'"
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -3,18 +3,18 @@ module Kamal::Commands::App::Assets
|
|||||||
asset_container = "#{role.container_prefix}-assets"
|
asset_container = "#{role.container_prefix}-assets"
|
||||||
|
|
||||||
combine \
|
combine \
|
||||||
make_directory(role.asset_extracted_path),
|
make_directory(role.asset_extracted_directory),
|
||||||
[ *docker(:stop, "-t 1", asset_container, "2> /dev/null"), "|| true" ],
|
[ *docker(:stop, "-t 1", asset_container, "2> /dev/null"), "|| true" ],
|
||||||
docker(:run, "--name", asset_container, "--detach", "--rm", config.absolute_image, "sleep 1000000"),
|
docker(:run, "--name", asset_container, "--detach", "--rm", "--entrypoint", "sleep", config.absolute_image, "1000000"),
|
||||||
docker(:cp, "-L", "#{asset_container}:#{role.asset_path}/.", role.asset_extracted_path),
|
docker(:cp, "-L", "#{asset_container}:#{role.asset_path}/.", role.asset_extracted_directory),
|
||||||
docker(:stop, "-t 1", asset_container),
|
docker(:stop, "-t 1", asset_container),
|
||||||
by: "&&"
|
by: "&&"
|
||||||
end
|
end
|
||||||
|
|
||||||
def sync_asset_volumes(old_version: nil)
|
def sync_asset_volumes(old_version: nil)
|
||||||
new_extracted_path, new_volume_path = role.asset_extracted_path(config.version), role.asset_volume.host_path
|
new_extracted_path, new_volume_path = role.asset_extracted_directory(config.version), role.asset_volume.host_path
|
||||||
if old_version.present?
|
if old_version.present?
|
||||||
old_extracted_path, old_volume_path = role.asset_extracted_path(old_version), role.asset_volume(old_version).host_path
|
old_extracted_path, old_volume_path = role.asset_extracted_directory(old_version), role.asset_volume(old_version).host_path
|
||||||
end
|
end
|
||||||
|
|
||||||
commands = [ make_directory(new_volume_path), copy_contents(new_extracted_path, new_volume_path) ]
|
commands = [ make_directory(new_volume_path), copy_contents(new_extracted_path, new_volume_path) ]
|
||||||
@@ -29,8 +29,8 @@ module Kamal::Commands::App::Assets
|
|||||||
|
|
||||||
def clean_up_assets
|
def clean_up_assets
|
||||||
chain \
|
chain \
|
||||||
find_and_remove_older_siblings(role.asset_extracted_path),
|
find_and_remove_older_siblings(role.asset_extracted_directory),
|
||||||
find_and_remove_older_siblings(role.asset_volume_path)
|
find_and_remove_older_siblings(role.asset_volume_directory)
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -39,7 +39,7 @@ module Kamal::Commands::App::Assets
|
|||||||
:find,
|
:find,
|
||||||
Pathname.new(path).dirname.to_s,
|
Pathname.new(path).dirname.to_s,
|
||||||
"-maxdepth 1",
|
"-maxdepth 1",
|
||||||
"-name", "'#{role.container_prefix}-*'",
|
"-name", "'#{role.name}-*'",
|
||||||
"!", "-name", Pathname.new(path).basename.to_s,
|
"!", "-name", Pathname.new(path).basename.to_s,
|
||||||
"-exec rm -rf \"{}\" +"
|
"-exec rm -rf \"{}\" +"
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1,22 +0,0 @@
|
|||||||
module Kamal::Commands::App::Cord
|
|
||||||
def cord(version:)
|
|
||||||
pipe \
|
|
||||||
docker(:inspect, "-f '{{ range .Mounts }}{{printf \"%s %s\\n\" .Source .Destination}}{{ end }}'", container_name(version)),
|
|
||||||
[ :awk, "'$2 == \"#{role.cord_volume.container_path}\" {print $1}'" ]
|
|
||||||
end
|
|
||||||
|
|
||||||
def tie_cord(cord)
|
|
||||||
create_empty_file(cord)
|
|
||||||
end
|
|
||||||
|
|
||||||
def cut_cord(cord)
|
|
||||||
remove_directory(cord)
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def create_empty_file(file)
|
|
||||||
chain \
|
|
||||||
make_directory_for(file),
|
|
||||||
[ :touch, file ]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -11,6 +11,7 @@ module Kamal::Commands::App::Execution
|
|||||||
docker :run,
|
docker :run,
|
||||||
("-it" if interactive),
|
("-it" if interactive),
|
||||||
"--rm",
|
"--rm",
|
||||||
|
"--network", "kamal",
|
||||||
*role&.env_args(host),
|
*role&.env_args(host),
|
||||||
*argumentize("--env", env),
|
*argumentize("--env", env),
|
||||||
*config.volume_args,
|
*config.volume_args,
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
module Kamal::Commands::App::Logging
|
module Kamal::Commands::App::Logging
|
||||||
def logs(version: nil, since: nil, lines: nil, grep: nil, grep_options: nil)
|
def logs(version: nil, timestamps: true, since: nil, lines: nil, grep: nil, grep_options: nil)
|
||||||
pipe \
|
pipe \
|
||||||
version ? container_id_for_version(version) : current_running_container_id,
|
version ? container_id_for_version(version) : current_running_container_id,
|
||||||
"xargs docker logs#{" --since #{since}" if since}#{" --tail #{lines}" if lines} 2>&1",
|
"xargs docker logs#{" --timestamps" if timestamps}#{" --since #{since}" if since}#{" --tail #{lines}" if lines} 2>&1",
|
||||||
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
||||||
end
|
end
|
||||||
|
|
||||||
def follow_logs(host:, lines: nil, grep: nil, grep_options: nil)
|
def follow_logs(host:, timestamps: true, lines: nil, grep: nil, grep_options: nil)
|
||||||
run_over_ssh \
|
run_over_ssh \
|
||||||
pipe(
|
pipe(
|
||||||
current_running_container_id,
|
current_running_container_id,
|
||||||
"xargs docker logs --timestamps#{" --tail #{lines}" if lines} --follow 2>&1",
|
"xargs docker logs#{" --timestamps" if timestamps}#{" --tail #{lines}" if lines} --follow 2>&1",
|
||||||
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
||||||
),
|
),
|
||||||
host: host
|
host: host
|
||||||
|
|||||||
16
lib/kamal/commands/app/proxy.rb
Normal file
16
lib/kamal/commands/app/proxy.rb
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
module Kamal::Commands::App::Proxy
|
||||||
|
delegate :proxy_container_name, to: :config
|
||||||
|
|
||||||
|
def deploy(target:)
|
||||||
|
proxy_exec :deploy, role.container_prefix, *role.proxy.deploy_command_args(target: target)
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove(target:)
|
||||||
|
proxy_exec :remove, role.container_prefix, *role.proxy.remove_command_args(target: target)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def proxy_exec(*command)
|
||||||
|
docker :exec, proxy_container_name, "kamal-proxy", *command
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -8,9 +8,12 @@ class Kamal::Commands::Auditor < Kamal::Commands::Base
|
|||||||
|
|
||||||
# Runs remotely
|
# Runs remotely
|
||||||
def record(line, **details)
|
def record(line, **details)
|
||||||
append \
|
combine \
|
||||||
[ :echo, audit_tags(**details).except(:version, :service_version).to_s, line ],
|
[ :mkdir, "-p", config.run_directory ],
|
||||||
audit_log_file
|
append(
|
||||||
|
[ :echo, audit_tags(**details).except(:version, :service_version, :service).to_s, line ],
|
||||||
|
audit_log_file
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def reveal
|
def reveal
|
||||||
|
|||||||
@@ -37,6 +37,10 @@ module Kamal::Commands
|
|||||||
[ :rm, "-r", path ]
|
[ :rm, "-r", path ]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def remove_file(path)
|
||||||
|
[ :rm, path ]
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def combine(*commands, by: "&&")
|
def combine(*commands, by: "&&")
|
||||||
commands
|
commands
|
||||||
@@ -81,6 +85,10 @@ module Kamal::Commands
|
|||||||
[ :git, *([ "-C", path ] if path), *args.compact ]
|
[ :git, *([ "-C", path ] if path), *args.compact ]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def grep(*args)
|
||||||
|
args.compact.unshift :grep
|
||||||
|
end
|
||||||
|
|
||||||
def tags(**details)
|
def tags(**details)
|
||||||
Kamal::Tags.from_config(config, **details)
|
Kamal::Tags.from_config(config, **details)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
require "active_support/core_ext/string/filters"
|
require "active_support/core_ext/string/filters"
|
||||||
|
|
||||||
class Kamal::Commands::Builder < Kamal::Commands::Base
|
class Kamal::Commands::Builder < Kamal::Commands::Base
|
||||||
delegate :create, :remove, :push, :clean, :pull, :info, :context_hosts, :config_context_hosts, :validate_image,
|
delegate :create, :remove, :push, :clean, :pull, :info, :inspect_builder, :validate_image, :first_mirror, to: :target
|
||||||
to: :target
|
delegate :local?, :remote?, to: "config.builder"
|
||||||
|
|
||||||
include Clone
|
include Clone
|
||||||
|
|
||||||
@@ -11,43 +11,27 @@ class Kamal::Commands::Builder < Kamal::Commands::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def target
|
def target
|
||||||
if config.builder.multiarch?
|
if remote?
|
||||||
if config.builder.remote?
|
if local?
|
||||||
if config.builder.local?
|
hybrid
|
||||||
multiarch_remote
|
|
||||||
else
|
|
||||||
native_remote
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
multiarch
|
remote
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
if config.builder.cached?
|
local
|
||||||
native_cached
|
|
||||||
else
|
|
||||||
native
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def native
|
def remote
|
||||||
@native ||= Kamal::Commands::Builder::Native.new(config)
|
@remote ||= Kamal::Commands::Builder::Remote.new(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
def native_cached
|
def local
|
||||||
@native ||= Kamal::Commands::Builder::Native::Cached.new(config)
|
@local ||= Kamal::Commands::Builder::Local.new(config)
|
||||||
end
|
end
|
||||||
|
|
||||||
def native_remote
|
def hybrid
|
||||||
@native ||= Kamal::Commands::Builder::Native::Remote.new(config)
|
@hybrid ||= Kamal::Commands::Builder::Hybrid.new(config)
|
||||||
end
|
|
||||||
|
|
||||||
def multiarch
|
|
||||||
@multiarch ||= Kamal::Commands::Builder::Multiarch.new(config)
|
|
||||||
end
|
|
||||||
|
|
||||||
def multiarch_remote
|
|
||||||
@multiarch_remote ||= Kamal::Commands::Builder::Multiarch::Remote.new(config)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,41 @@
|
|||||||
|
|
||||||
class Kamal::Commands::Builder::Base < Kamal::Commands::Base
|
class Kamal::Commands::Builder::Base < Kamal::Commands::Base
|
||||||
class BuilderError < StandardError; end
|
class BuilderError < StandardError; end
|
||||||
|
|
||||||
ENDPOINT_DOCKER_HOST_INSPECT = "'{{.Endpoints.docker.Host}}'"
|
ENDPOINT_DOCKER_HOST_INSPECT = "'{{.Endpoints.docker.Host}}'"
|
||||||
|
|
||||||
delegate :argumentize, to: Kamal::Utils
|
delegate :argumentize, to: Kamal::Utils
|
||||||
delegate :args, :secrets, :dockerfile, :target, :local_arch, :local_host, :remote_arch, :remote_host, :cache_from, :cache_to, :ssh, to: :builder_config
|
delegate \
|
||||||
|
:args, :secrets, :dockerfile, :target, :arches, :local_arches, :remote_arches, :remote,
|
||||||
|
:cache_from, :cache_to, :ssh, :driver, :docker_driver?,
|
||||||
|
to: :builder_config
|
||||||
|
|
||||||
def clean
|
def clean
|
||||||
docker :image, :rm, "--force", config.absolute_image
|
docker :image, :rm, "--force", config.absolute_image
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def push
|
||||||
|
docker :buildx, :build,
|
||||||
|
"--push",
|
||||||
|
*platform_options(arches),
|
||||||
|
*([ "--builder", builder_name ] unless docker_driver?),
|
||||||
|
*build_options,
|
||||||
|
build_context
|
||||||
|
end
|
||||||
|
|
||||||
def pull
|
def pull
|
||||||
docker :pull, config.absolute_image
|
docker :pull, config.absolute_image
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def info
|
||||||
|
combine \
|
||||||
|
docker(:context, :ls),
|
||||||
|
docker(:buildx, :ls)
|
||||||
|
end
|
||||||
|
|
||||||
|
def inspect_builder
|
||||||
|
docker :buildx, :inspect, builder_name unless docker_driver?
|
||||||
|
end
|
||||||
|
|
||||||
def build_options
|
def build_options
|
||||||
[ *build_tags, *build_cache, *build_labels, *build_args, *build_secrets, *build_dockerfile, *build_target, *build_ssh ]
|
[ *build_tags, *build_cache, *build_labels, *build_args, *build_secrets, *build_dockerfile, *build_target, *build_ssh ]
|
||||||
end
|
end
|
||||||
@@ -32,12 +53,8 @@ class Kamal::Commands::Builder::Base < Kamal::Commands::Base
|
|||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def context_hosts
|
def first_mirror
|
||||||
:true
|
docker(:info, "--format '{{index .RegistryConfig.Mirrors 0}}'")
|
||||||
end
|
|
||||||
|
|
||||||
def config_context_hosts
|
|
||||||
[]
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -61,7 +78,7 @@ class Kamal::Commands::Builder::Base < Kamal::Commands::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def build_secrets
|
def build_secrets
|
||||||
argumentize "--secret", secrets.collect { |secret| [ "id", secret ] }
|
argumentize "--secret", secrets.keys.collect { |secret| [ "id", secret ] }
|
||||||
end
|
end
|
||||||
|
|
||||||
def build_dockerfile
|
def build_dockerfile
|
||||||
@@ -84,7 +101,7 @@ class Kamal::Commands::Builder::Base < Kamal::Commands::Base
|
|||||||
config.builder
|
config.builder
|
||||||
end
|
end
|
||||||
|
|
||||||
def context_host(builder_name)
|
def platform_options(arches)
|
||||||
docker :context, :inspect, builder_name, "--format", ENDPOINT_DOCKER_HOST_INSPECT
|
argumentize "--platform", arches.map { |arch| "linux/#{arch}" }.join(",") if arches.any?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ module Kamal::Commands::Builder::Clone
|
|||||||
end
|
end
|
||||||
|
|
||||||
def clone
|
def clone
|
||||||
git :clone, Kamal::Git.root, path: clone_directory
|
git :clone, Kamal::Git.root, "--recurse-submodules", path: clone_directory
|
||||||
end
|
end
|
||||||
|
|
||||||
def clone_reset_steps
|
def clone_reset_steps
|
||||||
@@ -14,7 +14,8 @@ module Kamal::Commands::Builder::Clone
|
|||||||
git(:remote, "set-url", :origin, Kamal::Git.root, path: build_directory),
|
git(:remote, "set-url", :origin, Kamal::Git.root, path: build_directory),
|
||||||
git(:fetch, :origin, path: build_directory),
|
git(:fetch, :origin, path: build_directory),
|
||||||
git(:reset, "--hard", Kamal::Git.revision, path: build_directory),
|
git(:reset, "--hard", Kamal::Git.revision, path: build_directory),
|
||||||
git(:clean, "-fdx", path: build_directory)
|
git(:clean, "-fdx", path: build_directory),
|
||||||
|
git(:submodule, :update, "--init", path: build_directory)
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
21
lib/kamal/commands/builder/hybrid.rb
Normal file
21
lib/kamal/commands/builder/hybrid.rb
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
class Kamal::Commands::Builder::Hybrid < Kamal::Commands::Builder::Remote
|
||||||
|
def create
|
||||||
|
combine \
|
||||||
|
create_local_buildx,
|
||||||
|
create_remote_context,
|
||||||
|
append_remote_buildx
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def builder_name
|
||||||
|
"kamal-hybrid-#{driver}-#{remote.gsub(/[^a-z0-9_-]/, "-")}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def create_local_buildx
|
||||||
|
docker :buildx, :create, *platform_options(local_arches), "--name", builder_name, "--driver=#{driver}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def append_remote_buildx
|
||||||
|
docker :buildx, :create, *platform_options(remote_arches), "--append", "--name", builder_name, remote_context_name
|
||||||
|
end
|
||||||
|
end
|
||||||
14
lib/kamal/commands/builder/local.rb
Normal file
14
lib/kamal/commands/builder/local.rb
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
class Kamal::Commands::Builder::Local < Kamal::Commands::Builder::Base
|
||||||
|
def create
|
||||||
|
docker :buildx, :create, "--name", builder_name, "--driver=#{driver}" unless docker_driver?
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove
|
||||||
|
docker :buildx, :rm, builder_name unless docker_driver?
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def builder_name
|
||||||
|
"kamal-local-#{driver}"
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
class Kamal::Commands::Builder::Multiarch < Kamal::Commands::Builder::Base
|
|
||||||
def create
|
|
||||||
docker :buildx, :create, "--use", "--name", builder_name
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove
|
|
||||||
docker :buildx, :rm, builder_name
|
|
||||||
end
|
|
||||||
|
|
||||||
def info
|
|
||||||
combine \
|
|
||||||
docker(:context, :ls),
|
|
||||||
docker(:buildx, :ls)
|
|
||||||
end
|
|
||||||
|
|
||||||
def push
|
|
||||||
docker :buildx, :build,
|
|
||||||
"--push",
|
|
||||||
"--platform", platform_names,
|
|
||||||
"--builder", builder_name,
|
|
||||||
*build_options,
|
|
||||||
build_context
|
|
||||||
end
|
|
||||||
|
|
||||||
def context_hosts
|
|
||||||
docker :buildx, :inspect, builder_name, "> /dev/null"
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def builder_name
|
|
||||||
"kamal-#{config.service}-multiarch"
|
|
||||||
end
|
|
||||||
|
|
||||||
def platform_names
|
|
||||||
if local_arch
|
|
||||||
"linux/#{local_arch}"
|
|
||||||
else
|
|
||||||
"linux/amd64,linux/arm64"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
class Kamal::Commands::Builder::Multiarch::Remote < Kamal::Commands::Builder::Multiarch
|
|
||||||
def create
|
|
||||||
combine \
|
|
||||||
create_contexts,
|
|
||||||
create_local_buildx,
|
|
||||||
append_remote_buildx
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove
|
|
||||||
combine \
|
|
||||||
remove_contexts,
|
|
||||||
super
|
|
||||||
end
|
|
||||||
|
|
||||||
def context_hosts
|
|
||||||
chain \
|
|
||||||
context_host(builder_name_with_arch(local_arch)),
|
|
||||||
context_host(builder_name_with_arch(remote_arch))
|
|
||||||
end
|
|
||||||
|
|
||||||
def config_context_hosts
|
|
||||||
[ local_host, remote_host ].compact
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def builder_name
|
|
||||||
super + "-remote"
|
|
||||||
end
|
|
||||||
|
|
||||||
def builder_name_with_arch(arch)
|
|
||||||
"#{builder_name}-#{arch}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_local_buildx
|
|
||||||
docker :buildx, :create, "--name", builder_name, builder_name_with_arch(local_arch), "--platform", "linux/#{local_arch}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def append_remote_buildx
|
|
||||||
docker :buildx, :create, "--append", "--name", builder_name, builder_name_with_arch(remote_arch), "--platform", "linux/#{remote_arch}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_contexts
|
|
||||||
combine \
|
|
||||||
create_context(local_arch, local_host),
|
|
||||||
create_context(remote_arch, remote_host)
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_context(arch, host)
|
|
||||||
docker :context, :create, builder_name_with_arch(arch), "--description", "'#{builder_name} #{arch} native host'", "--docker", "'host=#{host}'"
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_contexts
|
|
||||||
combine \
|
|
||||||
remove_context(local_arch),
|
|
||||||
remove_context(remote_arch)
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_context(arch)
|
|
||||||
docker :context, :rm, builder_name_with_arch(arch)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
class Kamal::Commands::Builder::Native < Kamal::Commands::Builder::Base
|
|
||||||
def create
|
|
||||||
# No-op on native without cache
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove
|
|
||||||
# No-op on native without cache
|
|
||||||
end
|
|
||||||
|
|
||||||
def info
|
|
||||||
# No-op on native
|
|
||||||
end
|
|
||||||
|
|
||||||
def push
|
|
||||||
combine \
|
|
||||||
docker(:build, *build_options, build_context),
|
|
||||||
docker(:push, config.absolute_image),
|
|
||||||
docker(:push, config.latest_image)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
class Kamal::Commands::Builder::Native::Cached < Kamal::Commands::Builder::Native
|
|
||||||
def create
|
|
||||||
docker :buildx, :create, "--name", builder_name, "--use", "--driver=docker-container"
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove
|
|
||||||
docker :buildx, :rm, builder_name
|
|
||||||
end
|
|
||||||
|
|
||||||
def push
|
|
||||||
docker :buildx, :build,
|
|
||||||
"--push",
|
|
||||||
*build_options,
|
|
||||||
build_context
|
|
||||||
end
|
|
||||||
|
|
||||||
def context_hosts
|
|
||||||
docker :buildx, :inspect, builder_name, "> /dev/null"
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def builder_name
|
|
||||||
"kamal-#{config.service}-native-cached"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
class Kamal::Commands::Builder::Native::Remote < Kamal::Commands::Builder::Native
|
|
||||||
def create
|
|
||||||
chain \
|
|
||||||
create_context,
|
|
||||||
create_buildx
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove
|
|
||||||
chain \
|
|
||||||
remove_context,
|
|
||||||
remove_buildx
|
|
||||||
end
|
|
||||||
|
|
||||||
def info
|
|
||||||
chain \
|
|
||||||
docker(:context, :ls),
|
|
||||||
docker(:buildx, :ls)
|
|
||||||
end
|
|
||||||
|
|
||||||
def push
|
|
||||||
docker :buildx, :build,
|
|
||||||
"--push",
|
|
||||||
"--platform", platform,
|
|
||||||
"--builder", builder_name,
|
|
||||||
*build_options,
|
|
||||||
build_context
|
|
||||||
end
|
|
||||||
|
|
||||||
def context_hosts
|
|
||||||
context_host(builder_name_with_arch)
|
|
||||||
end
|
|
||||||
|
|
||||||
def config_context_hosts
|
|
||||||
[ remote_host ]
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
private
|
|
||||||
def builder_name
|
|
||||||
"kamal-#{config.service}-native-remote"
|
|
||||||
end
|
|
||||||
|
|
||||||
def builder_name_with_arch
|
|
||||||
"#{builder_name}-#{remote_arch}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def platform
|
|
||||||
"linux/#{remote_arch}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_context
|
|
||||||
docker :context, :create,
|
|
||||||
builder_name_with_arch, "--description", "'#{builder_name} #{remote_arch} native host'", "--docker", "'host=#{remote_host}'"
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_context
|
|
||||||
docker :context, :rm, builder_name_with_arch
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_buildx
|
|
||||||
docker :buildx, :create, "--name", builder_name, builder_name_with_arch, "--platform", platform
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_buildx
|
|
||||||
docker :buildx, :rm, builder_name
|
|
||||||
end
|
|
||||||
end
|
|
||||||
63
lib/kamal/commands/builder/remote.rb
Normal file
63
lib/kamal/commands/builder/remote.rb
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
class Kamal::Commands::Builder::Remote < Kamal::Commands::Builder::Base
|
||||||
|
def create
|
||||||
|
chain \
|
||||||
|
create_remote_context,
|
||||||
|
create_buildx
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove
|
||||||
|
chain \
|
||||||
|
remove_remote_context,
|
||||||
|
remove_buildx
|
||||||
|
end
|
||||||
|
|
||||||
|
def info
|
||||||
|
chain \
|
||||||
|
docker(:context, :ls),
|
||||||
|
docker(:buildx, :ls)
|
||||||
|
end
|
||||||
|
|
||||||
|
def inspect_builder
|
||||||
|
combine \
|
||||||
|
combine inspect_buildx, inspect_remote_context,
|
||||||
|
[ "(echo no compatible builder && exit 1)" ],
|
||||||
|
by: "||"
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def builder_name
|
||||||
|
"kamal-remote-#{remote.gsub(/[^a-z0-9_-]/, "-")}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def remote_context_name
|
||||||
|
"#{builder_name}-context"
|
||||||
|
end
|
||||||
|
|
||||||
|
def inspect_buildx
|
||||||
|
pipe \
|
||||||
|
docker(:buildx, :inspect, builder_name),
|
||||||
|
grep("-q", "Endpoint:.*#{remote_context_name}")
|
||||||
|
end
|
||||||
|
|
||||||
|
def inspect_remote_context
|
||||||
|
pipe \
|
||||||
|
docker(:context, :inspect, remote_context_name, "--format", ENDPOINT_DOCKER_HOST_INSPECT),
|
||||||
|
grep("-xq", remote)
|
||||||
|
end
|
||||||
|
|
||||||
|
def create_remote_context
|
||||||
|
docker :context, :create, remote_context_name, "--description", "'#{builder_name} host'", "--docker", "'host=#{remote}'"
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_remote_context
|
||||||
|
docker :context, :rm, remote_context_name
|
||||||
|
end
|
||||||
|
|
||||||
|
def create_buildx
|
||||||
|
docker :buildx, :create, "--name", builder_name, remote_context_name
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_buildx
|
||||||
|
docker :buildx, :rm, builder_name
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -19,6 +19,10 @@ class Kamal::Commands::Docker < Kamal::Commands::Base
|
|||||||
[ '[ "${EUID:-$(id -u)}" -eq 0 ] || command -v sudo >/dev/null || command -v su >/dev/null' ]
|
[ '[ "${EUID:-$(id -u)}" -eq 0 ] || command -v sudo >/dev/null || command -v su >/dev/null' ]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def create_network
|
||||||
|
docker :network, :create, :kamal
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def get_docker
|
def get_docker
|
||||||
shell \
|
shell \
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
class Kamal::Commands::Hook < Kamal::Commands::Base
|
class Kamal::Commands::Hook < Kamal::Commands::Base
|
||||||
def run(hook, **details)
|
def run(hook)
|
||||||
[ hook_file(hook), env: tags(**details).env ]
|
[ hook_file(hook) ]
|
||||||
|
end
|
||||||
|
|
||||||
|
def env(secrets: false, **details)
|
||||||
|
tags(**details).env.tap do |env|
|
||||||
|
env.merge!(config.secrets.to_h) if secrets
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def hook_exists?(hook)
|
def hook_exists?(hook)
|
||||||
|
|||||||
@@ -44,14 +44,10 @@ class Kamal::Commands::Lock < Kamal::Commands::Base
|
|||||||
"/dev/null"
|
"/dev/null"
|
||||||
end
|
end
|
||||||
|
|
||||||
def locks_dir
|
|
||||||
File.join(config.run_directory, "locks")
|
|
||||||
end
|
|
||||||
|
|
||||||
def lock_dir
|
def lock_dir
|
||||||
dir_name = [ config.service, config.destination ].compact.join("-")
|
dir_name = [ "lock", config.service, config.destination ].compact.join("-")
|
||||||
|
|
||||||
File.join(locks_dir, dir_name)
|
File.join(config.run_directory, dir_name)
|
||||||
end
|
end
|
||||||
|
|
||||||
def lock_details_file
|
def lock_details_file
|
||||||
|
|||||||
72
lib/kamal/commands/proxy.rb
Normal file
72
lib/kamal/commands/proxy.rb
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
class Kamal::Commands::Proxy < Kamal::Commands::Base
|
||||||
|
delegate :argumentize, :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
|
def run
|
||||||
|
docker :run,
|
||||||
|
"--name", container_name,
|
||||||
|
"--network", "kamal",
|
||||||
|
"--detach",
|
||||||
|
"--restart", "unless-stopped",
|
||||||
|
*config.proxy_publish_args,
|
||||||
|
"--volume", "kamal-proxy-config:/home/kamal-proxy/.config/kamal-proxy",
|
||||||
|
*config.logging_args,
|
||||||
|
config.proxy_image
|
||||||
|
end
|
||||||
|
|
||||||
|
def start
|
||||||
|
docker :container, :start, container_name
|
||||||
|
end
|
||||||
|
|
||||||
|
def stop(name: container_name)
|
||||||
|
docker :container, :stop, name
|
||||||
|
end
|
||||||
|
|
||||||
|
def start_or_run
|
||||||
|
combine start, run, by: "||"
|
||||||
|
end
|
||||||
|
|
||||||
|
def info
|
||||||
|
docker :ps, "--filter", "name=^#{container_name}$"
|
||||||
|
end
|
||||||
|
|
||||||
|
def version
|
||||||
|
pipe \
|
||||||
|
docker(:inspect, container_name, "--format '{{.Config.Image}}'"),
|
||||||
|
[ :cut, "-d:", "-f2" ]
|
||||||
|
end
|
||||||
|
|
||||||
|
def logs(timestamps: true, since: nil, lines: nil, grep: nil, grep_options: nil)
|
||||||
|
pipe \
|
||||||
|
docker(:logs, container_name, ("--since #{since}" if since), ("--tail #{lines}" if lines), ("--timestamps" if timestamps), "2>&1"),
|
||||||
|
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
||||||
|
end
|
||||||
|
|
||||||
|
def follow_logs(host:, timestamps: true, grep: nil, grep_options: nil)
|
||||||
|
run_over_ssh pipe(
|
||||||
|
docker(:logs, container_name, ("--timestamps" if timestamps), "--tail", "10", "--follow", "2>&1"),
|
||||||
|
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
||||||
|
).join(" "), host: host
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_container
|
||||||
|
docker :container, :prune, "--force", "--filter", "label=org.opencontainers.image.title=kamal-proxy"
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_image
|
||||||
|
docker :image, :prune, "--all", "--force", "--filter", "label=org.opencontainers.image.title=kamal-proxy"
|
||||||
|
end
|
||||||
|
|
||||||
|
def cleanup_traefik
|
||||||
|
chain \
|
||||||
|
docker(:container, :stop, "traefik"),
|
||||||
|
combine(
|
||||||
|
docker(:container, :prune, "--force", "--filter", "label=org.opencontainers.image.title=Traefik"),
|
||||||
|
docker(:image, :prune, "--all", "--force", "--filter", "label=org.opencontainers.image.title=Traefik")
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def container_name
|
||||||
|
config.proxy_container_name
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -9,7 +9,7 @@ class Kamal::Commands::Prune < Kamal::Commands::Base
|
|||||||
def tagged_images
|
def tagged_images
|
||||||
pipe \
|
pipe \
|
||||||
docker(:image, :ls, *service_filter, "--format", "'{{.ID}} {{.Repository}}:{{.Tag}}'"),
|
docker(:image, :ls, *service_filter, "--format", "'{{.ID}} {{.Repository}}:{{.Tag}}'"),
|
||||||
"grep -v -w \"#{active_image_list}\"",
|
grep("-v -w \"#{active_image_list}\""),
|
||||||
"while read image tag; do docker rmi $tag; done"
|
"while read image tag; do docker rmi $tag; done"
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -20,10 +20,6 @@ class Kamal::Commands::Prune < Kamal::Commands::Base
|
|||||||
"while read container_id; do docker rm $container_id; done"
|
"while read container_id; do docker rm $container_id; done"
|
||||||
end
|
end
|
||||||
|
|
||||||
def healthcheck_containers
|
|
||||||
docker :container, :prune, "--force", *healthcheck_service_filter
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
def stopped_containers_filters
|
def stopped_containers_filters
|
||||||
[ "created", "exited", "dead" ].flat_map { |status| [ "--filter", "status=#{status}" ] }
|
[ "created", "exited", "dead" ].flat_map { |status| [ "--filter", "status=#{status}" ] }
|
||||||
@@ -39,8 +35,4 @@ class Kamal::Commands::Prune < Kamal::Commands::Base
|
|||||||
def service_filter
|
def service_filter
|
||||||
[ "--filter", "label=service=#{config.service}" ]
|
[ "--filter", "label=service=#{config.service}" ]
|
||||||
end
|
end
|
||||||
|
|
||||||
def healthcheck_service_filter
|
|
||||||
[ "--filter", "label=service=#{config.healthcheck_service}" ]
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,5 +1,15 @@
|
|||||||
class Kamal::Commands::Server < Kamal::Commands::Base
|
class Kamal::Commands::Server < Kamal::Commands::Base
|
||||||
def ensure_run_directory
|
def ensure_run_directory
|
||||||
[ :mkdir, "-p", config.run_directory ]
|
make_directory config.run_directory
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_app_directory
|
||||||
|
remove_directory config.app_directory
|
||||||
|
end
|
||||||
|
|
||||||
|
def app_directory_count
|
||||||
|
pipe \
|
||||||
|
[ :ls, config.apps_directory ],
|
||||||
|
[ :wc, "-l" ]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,85 +0,0 @@
|
|||||||
class Kamal::Commands::Traefik < Kamal::Commands::Base
|
|
||||||
delegate :argumentize, :optionize, to: Kamal::Utils
|
|
||||||
delegate :port, :publish?, :labels, :env, :image, :options, :args, to: :"config.traefik"
|
|
||||||
|
|
||||||
def run
|
|
||||||
docker :run, "--name traefik",
|
|
||||||
"--detach",
|
|
||||||
"--restart", "unless-stopped",
|
|
||||||
*publish_args,
|
|
||||||
"--volume", "/var/run/docker.sock:/var/run/docker.sock",
|
|
||||||
*env_args,
|
|
||||||
*config.logging_args,
|
|
||||||
*label_args,
|
|
||||||
*docker_options_args,
|
|
||||||
image,
|
|
||||||
"--providers.docker",
|
|
||||||
*cmd_option_args
|
|
||||||
end
|
|
||||||
|
|
||||||
def start
|
|
||||||
docker :container, :start, "traefik"
|
|
||||||
end
|
|
||||||
|
|
||||||
def stop
|
|
||||||
docker :container, :stop, "traefik"
|
|
||||||
end
|
|
||||||
|
|
||||||
def start_or_run
|
|
||||||
any start, run
|
|
||||||
end
|
|
||||||
|
|
||||||
def info
|
|
||||||
docker :ps, "--filter", "name=^traefik$"
|
|
||||||
end
|
|
||||||
|
|
||||||
def logs(since: nil, lines: nil, grep: nil, grep_options: nil)
|
|
||||||
pipe \
|
|
||||||
docker(:logs, "traefik", (" --since #{since}" if since), (" --tail #{lines}" if lines), "--timestamps", "2>&1"),
|
|
||||||
("grep '#{grep}'#{" #{grep_options}" if grep_options}" if grep)
|
|
||||||
end
|
|
||||||
|
|
||||||
def follow_logs(host:, grep: nil, grep_options: nil)
|
|
||||||
run_over_ssh pipe(
|
|
||||||
docker(:logs, "traefik", "--timestamps", "--tail", "10", "--follow", "2>&1"),
|
|
||||||
(%(grep "#{grep}"#{" #{grep_options}" if grep_options}) if grep)
|
|
||||||
).join(" "), host: host
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_container
|
|
||||||
docker :container, :prune, "--force", "--filter", "label=org.opencontainers.image.title=Traefik"
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_image
|
|
||||||
docker :image, :prune, "--all", "--force", "--filter", "label=org.opencontainers.image.title=Traefik"
|
|
||||||
end
|
|
||||||
|
|
||||||
def make_env_directory
|
|
||||||
make_directory(env.secrets_directory)
|
|
||||||
end
|
|
||||||
|
|
||||||
def remove_env_file
|
|
||||||
[ :rm, "-f", env.secrets_file ]
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def publish_args
|
|
||||||
argumentize "--publish", port if publish?
|
|
||||||
end
|
|
||||||
|
|
||||||
def label_args
|
|
||||||
argumentize "--label", labels
|
|
||||||
end
|
|
||||||
|
|
||||||
def env_args
|
|
||||||
env.args
|
|
||||||
end
|
|
||||||
|
|
||||||
def docker_options_args
|
|
||||||
optionize(options)
|
|
||||||
end
|
|
||||||
|
|
||||||
def cmd_option_args
|
|
||||||
optionize args, with: "="
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -2,19 +2,22 @@ require "active_support/ordered_options"
|
|||||||
require "active_support/core_ext/string/inquiry"
|
require "active_support/core_ext/string/inquiry"
|
||||||
require "active_support/core_ext/module/delegation"
|
require "active_support/core_ext/module/delegation"
|
||||||
require "active_support/core_ext/hash/keys"
|
require "active_support/core_ext/hash/keys"
|
||||||
require "pathname"
|
|
||||||
require "erb"
|
require "erb"
|
||||||
require "net/ssh/proxy/jump"
|
require "net/ssh/proxy/jump"
|
||||||
|
|
||||||
class Kamal::Configuration
|
class Kamal::Configuration
|
||||||
delegate :service, :image, :labels, :stop_wait_time, :hooks_path, to: :raw_config, allow_nil: true
|
delegate :service, :image, :labels, :hooks_path, to: :raw_config, allow_nil: true
|
||||||
delegate :argumentize, :optionize, to: Kamal::Utils
|
delegate :argumentize, :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
attr_reader :destination, :raw_config
|
attr_reader :destination, :raw_config, :secrets
|
||||||
attr_reader :accessories, :boot, :builder, :env, :healthcheck, :logging, :traefik, :servers, :ssh, :sshkit, :registry
|
attr_reader :accessories, :aliases, :boot, :builder, :env, :logging, :proxy, :servers, :ssh, :sshkit, :registry
|
||||||
|
|
||||||
include Validation
|
include Validation
|
||||||
|
|
||||||
|
PROXY_MINIMUM_VERSION = "v0.5.0"
|
||||||
|
PROXY_HTTP_PORT = 80
|
||||||
|
PROXY_HTTPS_PORT = 443
|
||||||
|
|
||||||
class << self
|
class << self
|
||||||
def create_from(config_file:, destination: nil, version: nil)
|
def create_from(config_file:, destination: nil, version: nil)
|
||||||
raw_config = load_config_files(config_file, *destination_config_file(config_file, destination))
|
raw_config = load_config_files(config_file, *destination_config_file(config_file, destination))
|
||||||
@@ -47,20 +50,22 @@ class Kamal::Configuration
|
|||||||
@destination = destination
|
@destination = destination
|
||||||
@declared_version = version
|
@declared_version = version
|
||||||
|
|
||||||
validate! raw_config, example: validation_yml.symbolize_keys, context: ""
|
validate! raw_config, example: validation_yml.symbolize_keys, context: "", with: Kamal::Configuration::Validator::Configuration
|
||||||
|
|
||||||
|
@secrets = Kamal::Secrets.new(destination: destination)
|
||||||
|
|
||||||
# Eager load config to validate it, these are first as they have dependencies later on
|
# Eager load config to validate it, these are first as they have dependencies later on
|
||||||
@servers = Servers.new(config: self)
|
@servers = Servers.new(config: self)
|
||||||
@registry = Registry.new(config: self)
|
@registry = Registry.new(config: self)
|
||||||
|
|
||||||
@accessories = @raw_config.accessories&.keys&.collect { |name| Accessory.new(name, config: self) } || []
|
@accessories = @raw_config.accessories&.keys&.collect { |name| Accessory.new(name, config: self) } || []
|
||||||
|
@aliases = @raw_config.aliases&.keys&.to_h { |name| [ name, Alias.new(name, config: self) ] } || {}
|
||||||
@boot = Boot.new(config: self)
|
@boot = Boot.new(config: self)
|
||||||
@builder = Builder.new(config: self)
|
@builder = Builder.new(config: self)
|
||||||
@env = Env.new(config: @raw_config.env || {})
|
@env = Env.new(config: @raw_config.env || {}, secrets: secrets)
|
||||||
|
|
||||||
@healthcheck = Healthcheck.new(healthcheck_config: @raw_config.healthcheck)
|
|
||||||
@logging = Logging.new(logging_config: @raw_config.logging)
|
@logging = Logging.new(logging_config: @raw_config.logging)
|
||||||
@traefik = Traefik.new(config: self)
|
@proxy = Proxy.new(config: self, proxy_config: @raw_config.proxy || {})
|
||||||
@ssh = Ssh.new(config: self)
|
@ssh = Ssh.new(config: self)
|
||||||
@sshkit = Sshkit.new(config: self)
|
@sshkit = Sshkit.new(config: self)
|
||||||
|
|
||||||
@@ -69,6 +74,9 @@ class Kamal::Configuration
|
|||||||
ensure_valid_kamal_version
|
ensure_valid_kamal_version
|
||||||
ensure_retain_containers_valid
|
ensure_retain_containers_valid
|
||||||
ensure_valid_service_name
|
ensure_valid_service_name
|
||||||
|
ensure_no_traefik_reboot_hooks
|
||||||
|
ensure_one_host_for_ssl_roles
|
||||||
|
ensure_unique_hosts_for_ssl_roles
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
@@ -129,16 +137,16 @@ class Kamal::Configuration
|
|||||||
raw_config.allow_empty_roles
|
raw_config.allow_empty_roles
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik_roles
|
def proxy_roles
|
||||||
roles.select(&:running_traefik?)
|
roles.select(&:running_proxy?)
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik_role_names
|
def proxy_role_names
|
||||||
traefik_roles.flat_map(&:name)
|
proxy_roles.flat_map(&:name)
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik_hosts
|
def proxy_hosts
|
||||||
traefik_roles.flat_map(&:hosts).uniq
|
proxy_roles.flat_map(&:hosts).uniq
|
||||||
end
|
end
|
||||||
|
|
||||||
def repository
|
def repository
|
||||||
@@ -183,31 +191,40 @@ class Kamal::Configuration
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def healthcheck_service
|
|
||||||
[ "healthcheck", service, destination ].compact.join("-")
|
|
||||||
end
|
|
||||||
|
|
||||||
def readiness_delay
|
def readiness_delay
|
||||||
raw_config.readiness_delay || 7
|
raw_config.readiness_delay || 7
|
||||||
end
|
end
|
||||||
|
|
||||||
def run_id
|
def deploy_timeout
|
||||||
@run_id ||= SecureRandom.hex(16)
|
raw_config.deploy_timeout || 30
|
||||||
|
end
|
||||||
|
|
||||||
|
def drain_timeout
|
||||||
|
raw_config.drain_timeout || 30
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def run_directory
|
def run_directory
|
||||||
raw_config.run_directory || ".kamal"
|
".kamal"
|
||||||
end
|
end
|
||||||
|
|
||||||
def run_directory_as_docker_volume
|
def apps_directory
|
||||||
if Pathname.new(run_directory).absolute?
|
File.join run_directory, "apps"
|
||||||
run_directory
|
|
||||||
else
|
|
||||||
File.join "$(pwd)", run_directory
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def app_directory
|
||||||
|
File.join apps_directory, [ service, destination ].compact.join("-")
|
||||||
|
end
|
||||||
|
|
||||||
|
def env_directory
|
||||||
|
File.join app_directory, "env"
|
||||||
|
end
|
||||||
|
|
||||||
|
def assets_directory
|
||||||
|
File.join app_directory, "assets"
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
def hooks_path
|
def hooks_path
|
||||||
raw_config.hooks_path || ".kamal/hooks"
|
raw_config.hooks_path || ".kamal/hooks"
|
||||||
end
|
end
|
||||||
@@ -217,13 +234,9 @@ class Kamal::Configuration
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def host_env_directory
|
|
||||||
File.join(run_directory, "env")
|
|
||||||
end
|
|
||||||
|
|
||||||
def env_tags
|
def env_tags
|
||||||
@env_tags ||= if (tags = raw_config.env["tags"])
|
@env_tags ||= if (tags = raw_config.env["tags"])
|
||||||
tags.collect { |name, config| Env::Tag.new(name, config: config) }
|
tags.collect { |name, config| Env::Tag.new(name, config: config, secrets: secrets) }
|
||||||
else
|
else
|
||||||
[]
|
[]
|
||||||
end
|
end
|
||||||
@@ -233,6 +246,18 @@ class Kamal::Configuration
|
|||||||
env_tags.detect { |t| t.name == name.to_s }
|
env_tags.detect { |t| t.name == name.to_s }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def proxy_publish_args
|
||||||
|
argumentize "--publish", [ "#{PROXY_HTTP_PORT}:#{PROXY_HTTP_PORT}", "#{PROXY_HTTPS_PORT}:#{PROXY_HTTPS_PORT}" ]
|
||||||
|
end
|
||||||
|
|
||||||
|
def proxy_image
|
||||||
|
"basecamp/kamal-proxy:#{PROXY_MINIMUM_VERSION}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def proxy_container_name
|
||||||
|
"kamal-proxy"
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
def to_h
|
def to_h
|
||||||
{
|
{
|
||||||
@@ -248,8 +273,7 @@ class Kamal::Configuration
|
|||||||
sshkit: sshkit.to_h,
|
sshkit: sshkit.to_h,
|
||||||
builder: builder.to_h,
|
builder: builder.to_h,
|
||||||
accessories: raw_config.accessories,
|
accessories: raw_config.accessories,
|
||||||
logging: logging_args,
|
logging: logging_args
|
||||||
healthcheck: healthcheck.to_h
|
|
||||||
}.compact
|
}.compact
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -307,6 +331,30 @@ class Kamal::Configuration
|
|||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def ensure_no_traefik_reboot_hooks
|
||||||
|
hooks = %w[ pre-traefik-reboot post-traefik-reboot ].select { |hook_file| File.exist?(File.join(hooks_path, hook_file)) }
|
||||||
|
|
||||||
|
if hooks.any?
|
||||||
|
raise Kamal::ConfigurationError, "Found #{hooks.join(", ")}, these should be renamed to (pre|post)-proxy-reboot"
|
||||||
|
end
|
||||||
|
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
def ensure_one_host_for_ssl_roles
|
||||||
|
roles.each(&:ensure_one_host_for_ssl)
|
||||||
|
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
def ensure_unique_hosts_for_ssl_roles
|
||||||
|
hosts = roles.select(&:ssl?).map { |role| role.proxy.host }
|
||||||
|
duplicates = hosts.tally.filter_map { |host, count| host if count > 1 }
|
||||||
|
|
||||||
|
raise Kamal::ConfigurationError, "Different roles can't share the same host for SSL: #{duplicates.join(", ")}" if duplicates.any?
|
||||||
|
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
def role_names
|
def role_names
|
||||||
raw_config.servers.is_a?(Array) ? [ "web" ] : raw_config.servers.keys.sort
|
raw_config.servers.is_a?(Array) ? [ "web" ] : raw_config.servers.keys.sort
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ class Kamal::Configuration::Accessory
|
|||||||
|
|
||||||
@env = Kamal::Configuration::Env.new \
|
@env = Kamal::Configuration::Env.new \
|
||||||
config: accessory_config.fetch("env", {}),
|
config: accessory_config.fetch("env", {}),
|
||||||
secrets_file: File.join(config.host_env_directory, "accessories", "#{service_name}.env"),
|
secrets: config.secrets,
|
||||||
context: "accessories/#{name}/env"
|
context: "accessories/#{name}/env"
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -51,7 +51,19 @@ class Kamal::Configuration::Accessory
|
|||||||
end
|
end
|
||||||
|
|
||||||
def env_args
|
def env_args
|
||||||
env.args
|
[ *env.clear_args, *argumentize("--env-file", secrets_path) ]
|
||||||
|
end
|
||||||
|
|
||||||
|
def env_directory
|
||||||
|
File.join(config.env_directory, "accessories")
|
||||||
|
end
|
||||||
|
|
||||||
|
def secrets_io
|
||||||
|
env.secrets_io
|
||||||
|
end
|
||||||
|
|
||||||
|
def secrets_path
|
||||||
|
File.join(config.env_directory, "accessories", "#{name}.env")
|
||||||
end
|
end
|
||||||
|
|
||||||
def files
|
def files
|
||||||
|
|||||||
15
lib/kamal/configuration/alias.rb
Normal file
15
lib/kamal/configuration/alias.rb
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
class Kamal::Configuration::Alias
|
||||||
|
include Kamal::Configuration::Validation
|
||||||
|
|
||||||
|
attr_reader :name, :command
|
||||||
|
|
||||||
|
def initialize(name, config:)
|
||||||
|
@name, @command = name.inquiry, config.raw_config["aliases"][name]
|
||||||
|
|
||||||
|
validate! \
|
||||||
|
command,
|
||||||
|
example: validation_yml["aliases"]["uname"],
|
||||||
|
context: "aliases/#{name}",
|
||||||
|
with: Kamal::Configuration::Validator::Alias
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -19,16 +19,38 @@ class Kamal::Configuration::Builder
|
|||||||
builder_config
|
builder_config
|
||||||
end
|
end
|
||||||
|
|
||||||
def multiarch?
|
def remote
|
||||||
builder_config["multiarch"] != false
|
builder_config["remote"]
|
||||||
end
|
end
|
||||||
|
|
||||||
def local?
|
def arches
|
||||||
!!builder_config["local"]
|
Array(builder_config.fetch("arch", default_arch))
|
||||||
|
end
|
||||||
|
|
||||||
|
def local_arches
|
||||||
|
@local_arches ||= if local_disabled?
|
||||||
|
[]
|
||||||
|
elsif remote
|
||||||
|
arches & [ Kamal::Utils.docker_arch ]
|
||||||
|
else
|
||||||
|
arches
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def remote_arches
|
||||||
|
@remote_arches ||= if remote
|
||||||
|
arches - local_arches
|
||||||
|
else
|
||||||
|
[]
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def remote?
|
def remote?
|
||||||
!!builder_config["remote"]
|
remote_arches.any?
|
||||||
|
end
|
||||||
|
|
||||||
|
def local?
|
||||||
|
!local_disabled? && (arches.empty? || local_arches.any?)
|
||||||
end
|
end
|
||||||
|
|
||||||
def cached?
|
def cached?
|
||||||
@@ -40,7 +62,7 @@ class Kamal::Configuration::Builder
|
|||||||
end
|
end
|
||||||
|
|
||||||
def secrets
|
def secrets
|
||||||
builder_config["secrets"] || []
|
(builder_config["secrets"] || []).to_h { |key| [ key, config.secrets[key] ] }
|
||||||
end
|
end
|
||||||
|
|
||||||
def dockerfile
|
def dockerfile
|
||||||
@@ -55,20 +77,12 @@ class Kamal::Configuration::Builder
|
|||||||
builder_config["context"] || "."
|
builder_config["context"] || "."
|
||||||
end
|
end
|
||||||
|
|
||||||
def local_arch
|
def driver
|
||||||
builder_config["local"]["arch"] if local?
|
builder_config.fetch("driver", "docker-container")
|
||||||
end
|
end
|
||||||
|
|
||||||
def local_host
|
def local_disabled?
|
||||||
builder_config["local"]["host"] if local?
|
builder_config["local"] == false
|
||||||
end
|
|
||||||
|
|
||||||
def remote_arch
|
|
||||||
builder_config["remote"]["arch"] if remote?
|
|
||||||
end
|
|
||||||
|
|
||||||
def remote_host
|
|
||||||
builder_config["remote"]["host"] if remote?
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def cache_from
|
def cache_from
|
||||||
@@ -114,7 +128,23 @@ class Kamal::Configuration::Builder
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def docker_driver?
|
||||||
|
driver == "docker"
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
def valid?
|
||||||
|
if docker_driver?
|
||||||
|
raise ArgumentError, "Invalid builder configuration: the `docker` driver does not not support remote builders" if remote
|
||||||
|
raise ArgumentError, "Invalid builder configuration: the `docker` driver does not not support caching" if cached?
|
||||||
|
raise ArgumentError, "Invalid builder configuration: the `docker` driver does not not support multiple arches" if arches.many?
|
||||||
|
end
|
||||||
|
|
||||||
|
if @options["cache"] && @options["cache"]["type"]
|
||||||
|
raise ArgumentError, "Invalid cache type: #{@options["cache"]["type"]}" unless [ "gha", "registry" ].include?(@options["cache"]["type"])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def cache_image
|
def cache_image
|
||||||
builder_config["cache"]&.fetch("image", nil) || "#{image}-build-cache"
|
builder_config["cache"]&.fetch("image", nil) || "#{image}-build-cache"
|
||||||
end
|
end
|
||||||
@@ -150,4 +180,8 @@ class Kamal::Configuration::Builder
|
|||||||
def pwd_sha
|
def pwd_sha
|
||||||
Digest::SHA256.hexdigest(Dir.pwd)[0..12]
|
Digest::SHA256.hexdigest(Dir.pwd)[0..12]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def default_arch
|
||||||
|
docker_driver? ? [] : [ "amd64", "arm64" ]
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
26
lib/kamal/configuration/docs/alias.yml
Normal file
26
lib/kamal/configuration/docs/alias.yml
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Aliases
|
||||||
|
#
|
||||||
|
# Aliases are shortcuts for Kamal commands.
|
||||||
|
#
|
||||||
|
# For example, for a Rails app, you might open a console with:
|
||||||
|
#
|
||||||
|
# ```shell
|
||||||
|
# kamal app exec -i -r console "rails console"
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# By defining an alias, like this:
|
||||||
|
aliases:
|
||||||
|
console: app exec -r console -i "rails console"
|
||||||
|
# You can now open the console with:
|
||||||
|
# ```shell
|
||||||
|
# kamal console
|
||||||
|
# ```
|
||||||
|
|
||||||
|
# Configuring aliases
|
||||||
|
#
|
||||||
|
# Aliases are defined in the root config under the alias key
|
||||||
|
#
|
||||||
|
# Each alias is named and can only contain lowercase letters, numbers, dashes and underscores.
|
||||||
|
|
||||||
|
aliases:
|
||||||
|
uname: app exec -p -q -r web "uname -a"
|
||||||
@@ -1,10 +1,6 @@
|
|||||||
# Builder
|
# Builder
|
||||||
#
|
#
|
||||||
# The builder configuration controls how the application is built with `docker build` or `docker buildx build`
|
# The builder configuration controls how the application is built with `docker build`
|
||||||
#
|
|
||||||
# If no configuration is specified, Kamal will:
|
|
||||||
# 1. Create a buildx context called `kamal-<service>-multiarch`
|
|
||||||
# 2. Use `docker buildx build` to build a multiarch image for linux/amd64,linux/arm64 with that context
|
|
||||||
#
|
#
|
||||||
# See https://kamal-deploy.org/docs/configuration/builder-examples/ for more information
|
# See https://kamal-deploy.org/docs/configuration/builder-examples/ for more information
|
||||||
|
|
||||||
@@ -13,35 +9,33 @@
|
|||||||
# Options go under the builder key in the root configuration.
|
# Options go under the builder key in the root configuration.
|
||||||
builder:
|
builder:
|
||||||
|
|
||||||
# Multiarch
|
# Arch
|
||||||
#
|
#
|
||||||
# Enables multiarch builds, defaults to `true`
|
# The architectures to build for - you can set an array or just a single value.
|
||||||
multiarch: false
|
#
|
||||||
|
# Allowed values are `amd64` and `arm64`
|
||||||
|
arch:
|
||||||
|
- amd64
|
||||||
|
|
||||||
# Local configuration
|
# Remote
|
||||||
#
|
#
|
||||||
# The build configuration for local builds, only used if multiarch is enabled (the default)
|
# The connection string for a remote builder. If supplied Kamal will use this
|
||||||
#
|
# for builds that do not match the local architecture of the deployment host.
|
||||||
# If there is no remote configuration, by default we build for amd64 and arm64.
|
remote: ssh://docker@docker-builder
|
||||||
# If you only want to build for one architecture, you can specify it here.
|
|
||||||
# The docker socket is optional and uses the default docker host socket when not specified
|
|
||||||
local:
|
|
||||||
arch: amd64
|
|
||||||
host: /var/run/docker.sock
|
|
||||||
|
|
||||||
# Remote configuration
|
# Local
|
||||||
#
|
#
|
||||||
# The build configuration for remote builds, also only used if multiarch is enabled.
|
# If set to false, Kamal will always use the remote builder even when building
|
||||||
# The arch is required and can be either amd64 or arm64.
|
# the local architecture.
|
||||||
remote:
|
#
|
||||||
arch: arm64
|
# Defaults to true
|
||||||
host: ssh://docker@docker-builder
|
local: true
|
||||||
|
|
||||||
# Builder cache
|
# Builder cache
|
||||||
#
|
#
|
||||||
# The type must be either 'gha' or 'registry'
|
# The type must be either 'gha' or 'registry'
|
||||||
#
|
#
|
||||||
# The image is only used for registry cache
|
# The image is only used for registry cache. Not compatible with the docker driver
|
||||||
cache:
|
cache:
|
||||||
type: registry
|
type: registry
|
||||||
options: mode=max
|
options: mode=max
|
||||||
@@ -80,7 +74,7 @@ builder:
|
|||||||
|
|
||||||
# Build secrets
|
# Build secrets
|
||||||
#
|
#
|
||||||
# Values are read from the environment.
|
# Values are read from .kamal/secrets.
|
||||||
#
|
#
|
||||||
secrets:
|
secrets:
|
||||||
- SECRET1
|
- SECRET1
|
||||||
@@ -105,3 +99,8 @@ builder:
|
|||||||
#
|
#
|
||||||
# SSH agent socket or keys to expose to the build
|
# SSH agent socket or keys to expose to the build
|
||||||
ssh: default=$SSH_AUTH_SOCK
|
ssh: default=$SSH_AUTH_SOCK
|
||||||
|
|
||||||
|
# Driver
|
||||||
|
#
|
||||||
|
# The build driver to use, defaults to `docker-container`
|
||||||
|
driver: docker
|
||||||
|
|||||||
@@ -2,13 +2,24 @@
|
|||||||
#
|
#
|
||||||
# Configuration is read from the `config/deploy.yml`
|
# Configuration is read from the `config/deploy.yml`
|
||||||
#
|
#
|
||||||
|
|
||||||
|
# Destinations
|
||||||
|
#
|
||||||
# When running commands, you can specify a destination with the `-d` flag,
|
# When running commands, you can specify a destination with the `-d` flag,
|
||||||
# e.g. `kamal deploy -d staging`
|
# e.g. `kamal deploy -d staging`
|
||||||
#
|
#
|
||||||
# In this case the configuration will also be read from `config/deploy.staging.yml`
|
# In this case the configuration will also be read from `config/deploy.staging.yml`
|
||||||
# and merged with the base configuration.
|
# and merged with the base configuration.
|
||||||
|
|
||||||
|
# Extensions
|
||||||
#
|
#
|
||||||
# The available configuration options are explained below.
|
# Kamal will not accept unrecognized keys in the configuration file.
|
||||||
|
#
|
||||||
|
# However, you might want to declare a configuration block using YAML anchors
|
||||||
|
# and aliases to avoid repetition.
|
||||||
|
#
|
||||||
|
# You can use prefix a configuration section with `x-` to indicate that it is an
|
||||||
|
# extension. Kamal will ignore the extension and not raise an error.
|
||||||
|
|
||||||
# The service name
|
# The service name
|
||||||
# This is a required value. It is used as the container name prefix.
|
# This is a required value. It is used as the container name prefix.
|
||||||
@@ -25,6 +36,8 @@ image: my-image
|
|||||||
labels:
|
labels:
|
||||||
my-label: my-value
|
my-label: my-value
|
||||||
|
|
||||||
|
# Volumes
|
||||||
|
#
|
||||||
# Additional volumes to mount into the container
|
# Additional volumes to mount into the container
|
||||||
volumes:
|
volumes:
|
||||||
- /path/on/host:/path/in/container:ro
|
- /path/on/host:/path/in/container:ro
|
||||||
@@ -47,7 +60,7 @@ servers:
|
|||||||
env:
|
env:
|
||||||
...
|
...
|
||||||
|
|
||||||
# Asset Bridging
|
# Asset Path
|
||||||
#
|
#
|
||||||
# Used for asset bridging across deployments, default to `nil`
|
# Used for asset bridging across deployments, default to `nil`
|
||||||
#
|
#
|
||||||
@@ -59,10 +72,12 @@ env:
|
|||||||
# volume containing both sets of files.
|
# volume containing both sets of files.
|
||||||
# This requires that file names change when the contents change
|
# This requires that file names change when the contents change
|
||||||
# (e.g. by including a hash of the contents in the name).
|
# (e.g. by including a hash of the contents in the name).
|
||||||
|
#
|
||||||
# To configure this, set the path to the assets:
|
# To configure this, set the path to the assets:
|
||||||
asset_path: /path/to/assets
|
asset_path: /path/to/assets
|
||||||
|
|
||||||
|
# Hooks path
|
||||||
|
#
|
||||||
# Path to hooks, defaults to `.kamal/hooks`
|
# Path to hooks, defaults to `.kamal/hooks`
|
||||||
# See https://kamal-deploy.org/docs/hooks for more information
|
# See https://kamal-deploy.org/docs/hooks for more information
|
||||||
hooks_path: /user_home/kamal/hooks
|
hooks_path: /user_home/kamal/hooks
|
||||||
@@ -72,7 +87,7 @@ hooks_path: /user_home/kamal/hooks
|
|||||||
# Whether deployments require a destination to be specified, defaults to `false`
|
# Whether deployments require a destination to be specified, defaults to `false`
|
||||||
require_destination: true
|
require_destination: true
|
||||||
|
|
||||||
# The primary role
|
# Primary role
|
||||||
#
|
#
|
||||||
# This defaults to `web`, but if you have no web role, you can change this
|
# This defaults to `web`, but if you have no web role, you can change this
|
||||||
primary_role: workers
|
primary_role: workers
|
||||||
@@ -82,11 +97,6 @@ primary_role: workers
|
|||||||
# Whether roles with no servers are allowed. Defaults to `false`.
|
# Whether roles with no servers are allowed. Defaults to `false`.
|
||||||
allow_empty_roles: false
|
allow_empty_roles: false
|
||||||
|
|
||||||
# Stop wait time
|
|
||||||
#
|
|
||||||
# How long we wait for a container to stop before killing it, defaults to 30 seconds
|
|
||||||
stop_wait_time: 60
|
|
||||||
|
|
||||||
# Retain containers
|
# Retain containers
|
||||||
#
|
#
|
||||||
# How many old containers and images we retain, defaults to 5
|
# How many old containers and images we retain, defaults to 5
|
||||||
@@ -100,9 +110,20 @@ minimum_version: 1.3.0
|
|||||||
# Readiness delay
|
# Readiness delay
|
||||||
#
|
#
|
||||||
# Seconds to wait for a container to boot after is running, default 7
|
# Seconds to wait for a container to boot after is running, default 7
|
||||||
# This only applies to containers that do not specify a healthcheck
|
#
|
||||||
|
# This only applies to containers that do not run a proxy or specify a healthcheck
|
||||||
readiness_delay: 4
|
readiness_delay: 4
|
||||||
|
|
||||||
|
# Deploy timeout
|
||||||
|
#
|
||||||
|
# How long to wait for a container to become ready, default 30
|
||||||
|
deploy_timeout: 10
|
||||||
|
|
||||||
|
# Drain timeout
|
||||||
|
#
|
||||||
|
# How long to wait for a containers to drain, default 30
|
||||||
|
drain_timeout: 10
|
||||||
|
|
||||||
# Run directory
|
# Run directory
|
||||||
#
|
#
|
||||||
# Directory to store kamal runtime files in on the host, default `.kamal`
|
# Directory to store kamal runtime files in on the host, default `.kamal`
|
||||||
@@ -126,10 +147,10 @@ builder:
|
|||||||
accessories:
|
accessories:
|
||||||
...
|
...
|
||||||
|
|
||||||
# Traefik
|
# Proxy
|
||||||
#
|
#
|
||||||
# The Traefik proxy is used for zero-downtime deployments, see kamal docs traefik
|
# Configuration for kamal-proxy, see kamal docs proxy
|
||||||
traefik:
|
proxy:
|
||||||
...
|
...
|
||||||
|
|
||||||
# SSHKit
|
# SSHKit
|
||||||
@@ -144,14 +165,14 @@ sshkit:
|
|||||||
boot:
|
boot:
|
||||||
...
|
...
|
||||||
|
|
||||||
# Healthcheck
|
|
||||||
#
|
|
||||||
# Configuring healthcheck commands, intervals and timeouts, see kamal docs healthcheck
|
|
||||||
healthcheck:
|
|
||||||
...
|
|
||||||
|
|
||||||
# Logging
|
# Logging
|
||||||
#
|
#
|
||||||
# Docker logging configuration, see kamal docs logging
|
# Docker logging configuration, see kamal docs logging
|
||||||
logging:
|
logging:
|
||||||
...
|
...
|
||||||
|
|
||||||
|
# Aliases
|
||||||
|
#
|
||||||
|
# Alias configuration, see kamal docs alias
|
||||||
|
aliases:
|
||||||
|
...
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# Environment variables
|
# Environment variables
|
||||||
#
|
#
|
||||||
# Environment variables can be set directory in the Kamal configuration or
|
# Environment variables can be set directly in the Kamal configuration or
|
||||||
# for loaded from a .env file, for secrets that should not be checked into Git.
|
# read from .kamal/secrets.
|
||||||
|
|
||||||
# Reading environment variables from the configuration
|
# Reading environment variables from the configuration
|
||||||
#
|
#
|
||||||
@@ -12,26 +12,38 @@ env:
|
|||||||
DATABASE_HOST: mysql-db1
|
DATABASE_HOST: mysql-db1
|
||||||
DATABASE_PORT: 3306
|
DATABASE_PORT: 3306
|
||||||
|
|
||||||
# Using .env file to load required environment variables
|
# Secrets
|
||||||
#
|
#
|
||||||
# Kamal uses dotenv to automatically load environment variables set in the .env file present
|
# Kamal uses dotenv to automatically load environment variables set in the `.kamal/secrets` file.
|
||||||
# in the application root.
|
#
|
||||||
|
# If you are using destinations, secrets will instead be read from `.kamal/secrets-<DESTINATION>` if
|
||||||
|
# it exists.
|
||||||
|
#
|
||||||
|
# Common secrets across all destinations can be set in `.kamal/secrets-common`.
|
||||||
|
#
|
||||||
|
# This file can be used to set variables like `KAMAL_REGISTRY_PASSWORD` or database passwords.
|
||||||
|
# You can use variable or command substitution in the secrets file.
|
||||||
#
|
#
|
||||||
# This file can be used to set variables like KAMAL_REGISTRY_PASSWORD or database passwords.
|
|
||||||
# But for this reason you must ensure that .env files are not checked into Git or included
|
|
||||||
# in your Dockerfile! The format is just key-value like:
|
|
||||||
# ```
|
# ```
|
||||||
# KAMAL_REGISTRY_PASSWORD=pw
|
# KAMAL_REGISTRY_PASSWORD=$KAMAL_REGISTRY_PASSWORD
|
||||||
# DB_PASSWORD=secret123
|
# RAILS_MASTER_KEY=$(cat config/master.key)
|
||||||
# ```
|
# ```
|
||||||
# See https://kamal-deploy.org/docs/commands/envify/ for how to use generated .env files.
|
#
|
||||||
|
# You can also use [secret helpers](../commands/secrets) for some common password managers.
|
||||||
|
# ```
|
||||||
|
# SECRETS=$(kamal secrets fetch ...)
|
||||||
|
#
|
||||||
|
# REGISTRY_PASSWORD=$(kamal secrets extract REGISTRY_PASSWORD $SECRETS)
|
||||||
|
# DB_PASSWORD=$(kamal secrets extract DB_PASSWORD $SECRETS)
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# If you store secrets directly in .kamal/secrets, ensure that it is not checked into version control.
|
||||||
#
|
#
|
||||||
# To pass the secrets you should list them under the `secret` key. When you do this the
|
# To pass the secrets you should list them under the `secret` key. When you do this the
|
||||||
# other variables need to be moved under the `clear` key.
|
# other variables need to be moved under the `clear` key.
|
||||||
#
|
#
|
||||||
# Unlike clear valies, secrets are not passed directly to the container,
|
# Unlike clear values, secrets are not passed directly to the container,
|
||||||
# but are stored in an env file on the host
|
# but are stored in an env file on the host
|
||||||
# The file is not updated when deploying, only when running `kamal envify` or `kamal env push`.
|
|
||||||
env:
|
env:
|
||||||
clear:
|
clear:
|
||||||
DB_USER: app
|
DB_USER: app
|
||||||
|
|||||||
@@ -1,59 +0,0 @@
|
|||||||
# Healthcheck configuration
|
|
||||||
#
|
|
||||||
# On roles that are running Traefik, Kamal will supply a default healthcheck to `docker run`.
|
|
||||||
# For other roles, by default no healthcheck is supplied.
|
|
||||||
#
|
|
||||||
# If no healthcheck is supplied and the image does not define one, they we wait for the container
|
|
||||||
# to reach a running state and then pause for the readiness delay.
|
|
||||||
#
|
|
||||||
# The default healthcheck is `curl -f http://localhost:<port>/<path>`, so it assumes that `curl`
|
|
||||||
# is available within the container.
|
|
||||||
|
|
||||||
# Healthcheck options
|
|
||||||
#
|
|
||||||
# These go under the `healthcheck` key in the root or role configuration.
|
|
||||||
healthcheck:
|
|
||||||
|
|
||||||
# Command
|
|
||||||
#
|
|
||||||
# The command to run, defaults to `curl -f http://localhost:<port>/<path>` on roles running Traefik
|
|
||||||
cmd: "curl -f http://localhost"
|
|
||||||
|
|
||||||
# Interval
|
|
||||||
#
|
|
||||||
# The Docker healthcheck interval, defaults to `1s`
|
|
||||||
interval: 10s
|
|
||||||
|
|
||||||
# Max attempts
|
|
||||||
#
|
|
||||||
# The maximum number of times we poll the container to see if it is healthy, defaults to `7`
|
|
||||||
# Each check is separated by an increasing interval starting with 1 second.
|
|
||||||
max_attempts: 3
|
|
||||||
|
|
||||||
# Port
|
|
||||||
#
|
|
||||||
# The port to use in the healthcheck, defaults to `3000`
|
|
||||||
port: "80"
|
|
||||||
|
|
||||||
# Path
|
|
||||||
#
|
|
||||||
# The path to use in the healthcheck, defaults to `/up`
|
|
||||||
path: /health
|
|
||||||
|
|
||||||
# Cords for zero-downtime deployments
|
|
||||||
#
|
|
||||||
# The cord file is used for zero-downtime deployments. The healthcheck is augmented with a check
|
|
||||||
# for the existance of the file. This allows us to delete the file and force the container to
|
|
||||||
# become unhealthy, causing Traefik to stop routing traffic to it.
|
|
||||||
#
|
|
||||||
# Kamal mounts a volume at this location and creates the file before starting the container.
|
|
||||||
# You can set the value to `false` to disable the cord file, but this loses the zero-downtime
|
|
||||||
# guarantee.
|
|
||||||
#
|
|
||||||
# The default value is `/tmp/kamal-cord`
|
|
||||||
cord: /cord
|
|
||||||
|
|
||||||
# Log lines
|
|
||||||
#
|
|
||||||
# Number of lines to log from the container when the healthcheck fails, defaults to `50`
|
|
||||||
log_lines: 100
|
|
||||||
100
lib/kamal/configuration/docs/proxy.yml
Normal file
100
lib/kamal/configuration/docs/proxy.yml
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
# Proxy
|
||||||
|
#
|
||||||
|
# Kamal uses [kamal-proxy](https://github.com/basecamp/kamal-proxy) to provide
|
||||||
|
# gapless deployments. It runs on ports 80 and 443 and forwards requests to the
|
||||||
|
# application container.
|
||||||
|
#
|
||||||
|
# The proxy is configured in the root configuration under `proxy`. These are
|
||||||
|
# options that are set when deploying the application, not when booting the proxy
|
||||||
|
#
|
||||||
|
# They are application specific, so are not shared when multiple applications
|
||||||
|
# run on the same proxy.
|
||||||
|
#
|
||||||
|
# The proxy is enabled by default on the primary role, but can be disabled by
|
||||||
|
# setting `proxy: false`.
|
||||||
|
#
|
||||||
|
# It is disabled by default on all other roles, but can be enabled by setting
|
||||||
|
# `proxy: true`, or providing a proxy configuration.
|
||||||
|
proxy:
|
||||||
|
|
||||||
|
# Host
|
||||||
|
#
|
||||||
|
# The hosts that will be used to serve the app. The proxy will only route requests
|
||||||
|
# to this host to your app.
|
||||||
|
#
|
||||||
|
# If no hosts are set, then all requests will be forwarded, except for matching
|
||||||
|
# requests for other apps deployed on that server that do have a host set.
|
||||||
|
host: foo.example.com
|
||||||
|
|
||||||
|
# App port
|
||||||
|
#
|
||||||
|
# The port the application container is exposed on
|
||||||
|
#
|
||||||
|
# Defaults to 80
|
||||||
|
app_port: 3000
|
||||||
|
|
||||||
|
# SSL
|
||||||
|
#
|
||||||
|
# kamal-proxy can provide automatic HTTPS for your application via Let's Encrypt.
|
||||||
|
#
|
||||||
|
# This requires that we are deploying to a one server and the host option is set.
|
||||||
|
# The host value must point to the server we are deploying to and port 443 must be
|
||||||
|
# open for the Let's Encrypt challenge to succeed.
|
||||||
|
#
|
||||||
|
# Defaults to false
|
||||||
|
ssl: true
|
||||||
|
|
||||||
|
# Response timeout
|
||||||
|
#
|
||||||
|
# How long to wait for requests to complete before timing out, defaults to 30 seconds
|
||||||
|
response_timeout: 10
|
||||||
|
|
||||||
|
# Healthcheck
|
||||||
|
#
|
||||||
|
# When deploying, the proxy will by default hit /up once every second until we hit
|
||||||
|
# the deploy timeout, with a 5 second timeout for each request.
|
||||||
|
#
|
||||||
|
# Once the app is up, the proxy will stop hitting the healthcheck endpoint.
|
||||||
|
healthcheck:
|
||||||
|
interval: 3
|
||||||
|
path: /health
|
||||||
|
timeout: 3
|
||||||
|
|
||||||
|
# Buffering
|
||||||
|
#
|
||||||
|
# Whether to buffer request and response bodies in the proxy
|
||||||
|
#
|
||||||
|
# By default buffering is enabled with a max request body size of 1GB and no limit
|
||||||
|
# for response size.
|
||||||
|
#
|
||||||
|
# You can also set the memory limit for buffering, which defaults to 1MB, anything
|
||||||
|
# larger than that is written to disk.
|
||||||
|
buffering:
|
||||||
|
requests: true
|
||||||
|
responses: true
|
||||||
|
max_request_body: 40_000_000
|
||||||
|
max_response_body: 0
|
||||||
|
memory: 2_000_000
|
||||||
|
|
||||||
|
# Logging
|
||||||
|
#
|
||||||
|
# Configure request logging for the proxy
|
||||||
|
# You can specify request and response headers to log.
|
||||||
|
# By default, Cache-Control, Last-Modified and User-Agent request headers are logged
|
||||||
|
logging:
|
||||||
|
request_headers:
|
||||||
|
- Cache-Control
|
||||||
|
- X-Forwarded-Proto
|
||||||
|
response_headers:
|
||||||
|
- X-Request-ID
|
||||||
|
- X-Request-Start
|
||||||
|
|
||||||
|
# Forward headers
|
||||||
|
#
|
||||||
|
# Whether to forward the X-Forwarded-For and X-Forwarded-Proto headers.
|
||||||
|
#
|
||||||
|
# If you are behind a trusted proxy, you can set this to true to forward the headers.
|
||||||
|
#
|
||||||
|
# By default kamal-proxy will not forward the headers the ssl option is set to true, and
|
||||||
|
# will forward them if it is set to false.
|
||||||
|
forward_headers: true
|
||||||
@@ -27,11 +27,13 @@ registry:
|
|||||||
# and [set up roles and permissions](https://cloud.google.com/artifact-registry/docs/access-control#permissions).
|
# and [set up roles and permissions](https://cloud.google.com/artifact-registry/docs/access-control#permissions).
|
||||||
# Normally, assigning a roles/artifactregistry.writer role should be sufficient.
|
# Normally, assigning a roles/artifactregistry.writer role should be sufficient.
|
||||||
#
|
#
|
||||||
# Once the service account is ready, you need to generate and download a JSON key, base64 encode it and add to .env:
|
# Once the service account is ready, you need to generate and download a JSON key and base64 encode it:
|
||||||
#
|
#
|
||||||
# ```shell
|
# ```shell
|
||||||
# echo "KAMAL_REGISTRY_PASSWORD=$(base64 -i /path/to/key.json)" | tr -d "\\n" >> .env
|
# base64 -i /path/to/key.json | tr -d "\\n")
|
||||||
# ```
|
# ```
|
||||||
|
# You'll then need to set the KAMAL_REGISTRY_PASSWORD secret to that value.
|
||||||
|
#
|
||||||
# Use the env variable as password along with _json_key_base64 as username.
|
# Use the env variable as password along with _json_key_base64 as username.
|
||||||
# Here’s the final configuration:
|
# Here’s the final configuration:
|
||||||
|
|
||||||
|
|||||||
@@ -26,8 +26,12 @@ servers:
|
|||||||
#
|
#
|
||||||
# When there are other options to set, the list of hosts goes under the `hosts` key
|
# When there are other options to set, the list of hosts goes under the `hosts` key
|
||||||
#
|
#
|
||||||
# By default only the primary role uses Traefik, but you can set `traefik` to change
|
# By default only the primary role uses a proxy.
|
||||||
# it.
|
#
|
||||||
|
# For other roles, you can set it to `proxy: true` enable it and inherit the root proxy
|
||||||
|
# configuration or provide a map of options to override the root configuration.
|
||||||
|
#
|
||||||
|
# For the primary role, you can set `proxy: false` to disable the proxy.
|
||||||
#
|
#
|
||||||
# You can also set a custom cmd to run in the container, and overwrite other settings
|
# You can also set a custom cmd to run in the container, and overwrite other settings
|
||||||
# from the root configuration.
|
# from the root configuration.
|
||||||
@@ -35,18 +39,16 @@ servers:
|
|||||||
hosts:
|
hosts:
|
||||||
- 172.1.0.3
|
- 172.1.0.3
|
||||||
- 172.1.0.4: experiment1
|
- 172.1.0.4: experiment1
|
||||||
traefik: true
|
|
||||||
cmd: "bin/jobs"
|
cmd: "bin/jobs"
|
||||||
options:
|
options:
|
||||||
memory: 2g
|
memory: 2g
|
||||||
cpus: 4
|
cpus: 4
|
||||||
healthcheck:
|
|
||||||
...
|
|
||||||
logging:
|
logging:
|
||||||
...
|
...
|
||||||
|
proxy:
|
||||||
|
...
|
||||||
labels:
|
labels:
|
||||||
my-label: workers
|
my-label: workers
|
||||||
env:
|
env:
|
||||||
...
|
...
|
||||||
asset_path: /public
|
asset_path: /public
|
||||||
|
|
||||||
|
|||||||
@@ -44,3 +44,23 @@ ssh:
|
|||||||
# Defaults to `fatal`. Set this to debug if you are having
|
# Defaults to `fatal`. Set this to debug if you are having
|
||||||
# SSH connection issues.
|
# SSH connection issues.
|
||||||
log_level: debug
|
log_level: debug
|
||||||
|
|
||||||
|
# Keys Only
|
||||||
|
#
|
||||||
|
# Set to true to use only private keys from keys and key_data parameters,
|
||||||
|
# even if ssh-agent offers more identities. This option is intended for
|
||||||
|
# situations where ssh-agent offers many different identites or you have
|
||||||
|
# a need to overwrite all identites and force a single one.
|
||||||
|
keys_only: false
|
||||||
|
|
||||||
|
# Keys
|
||||||
|
#
|
||||||
|
# An array of file names of private keys to use for publickey
|
||||||
|
# and hostbased authentication
|
||||||
|
keys: [ "~/.ssh/id.pem" ]
|
||||||
|
|
||||||
|
# Key Data
|
||||||
|
#
|
||||||
|
# An array of strings, with each element of the array being
|
||||||
|
# a raw private key in PEM format.
|
||||||
|
key_data: [ "-----BEGIN OPENSSH PRIVATE KEY-----" ]
|
||||||
|
|||||||
@@ -1,62 +0,0 @@
|
|||||||
# Traefik
|
|
||||||
#
|
|
||||||
# Traefik is a reverse proxy, used by Kamal for zero-downtime deployments.
|
|
||||||
#
|
|
||||||
# We start an instance on the hosts in it's own container.
|
|
||||||
#
|
|
||||||
# During a deployment:
|
|
||||||
# 1. We start a new container which Traefik automatically detects due to the labels we have applied
|
|
||||||
# 2. Traefik starts routing traffic to the new container
|
|
||||||
# 3. We force the old container to fail it's healthcheck, causing Traefik to stop routing traffic to it
|
|
||||||
# 4. We stop the old container
|
|
||||||
|
|
||||||
# Traefik settings
|
|
||||||
#
|
|
||||||
# Traekik is configured in the root configuration under `traefik`.
|
|
||||||
traefik:
|
|
||||||
|
|
||||||
# Image
|
|
||||||
#
|
|
||||||
# The Traefik image to use, defaults to `traefik:v2.10`
|
|
||||||
image: traefik:v2.9
|
|
||||||
|
|
||||||
# Host port
|
|
||||||
#
|
|
||||||
# The host port to publish the Traefik container on, defaults to `80`
|
|
||||||
host_port: "8080"
|
|
||||||
|
|
||||||
# Disabling publishing
|
|
||||||
#
|
|
||||||
# To avoid publishing the Traefik container, set this to `false`
|
|
||||||
publish: false
|
|
||||||
|
|
||||||
# Labels
|
|
||||||
#
|
|
||||||
# Additional labels to apply to the Traefik container
|
|
||||||
labels:
|
|
||||||
traefik.http.routers.catchall.entryPoints: http
|
|
||||||
traefik.http.routers.catchall.rule: PathPrefix(`/`)
|
|
||||||
traefik.http.routers.catchall.service: unavailable
|
|
||||||
traefik.http.routers.catchall.priority: "1"
|
|
||||||
traefik.http.services.unavailable.loadbalancer.server.port: "0"
|
|
||||||
|
|
||||||
# Arguments
|
|
||||||
#
|
|
||||||
# Additional arguments to pass to the Traefik container
|
|
||||||
args:
|
|
||||||
entryPoints.http.address: ":80"
|
|
||||||
entryPoints.http.forwardedHeaders.insecure: true
|
|
||||||
accesslog: true
|
|
||||||
accesslog.format: json
|
|
||||||
|
|
||||||
# Options
|
|
||||||
#
|
|
||||||
# Additional options to pass to `docker run`
|
|
||||||
options:
|
|
||||||
cpus: 2
|
|
||||||
|
|
||||||
# Environment variables
|
|
||||||
#
|
|
||||||
# See kamal docs env
|
|
||||||
env:
|
|
||||||
...
|
|
||||||
@@ -1,36 +1,29 @@
|
|||||||
class Kamal::Configuration::Env
|
class Kamal::Configuration::Env
|
||||||
include Kamal::Configuration::Validation
|
include Kamal::Configuration::Validation
|
||||||
|
|
||||||
attr_reader :secrets_keys, :clear, :secrets_file, :context
|
attr_reader :context, :secrets
|
||||||
|
attr_reader :clear, :secret_keys
|
||||||
delegate :argumentize, to: Kamal::Utils
|
delegate :argumentize, to: Kamal::Utils
|
||||||
|
|
||||||
def initialize(config:, secrets_file: nil, context: "env")
|
def initialize(config:, secrets:, context: "env")
|
||||||
@clear = config.fetch("clear", config.key?("secret") || config.key?("tags") ? {} : config)
|
@clear = config.fetch("clear", config.key?("secret") || config.key?("tags") ? {} : config)
|
||||||
@secrets_keys = config.fetch("secret", [])
|
@secrets = secrets
|
||||||
@secrets_file = secrets_file
|
@secret_keys = config.fetch("secret", [])
|
||||||
@context = context
|
@context = context
|
||||||
validate! config, context: context, with: Kamal::Configuration::Validator::Env
|
validate! config, context: context, with: Kamal::Configuration::Validator::Env
|
||||||
end
|
end
|
||||||
|
|
||||||
def args
|
def clear_args
|
||||||
[ "--env-file", secrets_file, *argumentize("--env", clear) ]
|
argumentize("--env", clear)
|
||||||
end
|
end
|
||||||
|
|
||||||
def secrets_io
|
def secrets_io
|
||||||
StringIO.new(Kamal::EnvFile.new(secrets).to_s)
|
Kamal::EnvFile.new(secret_keys.to_h { |key| [ key, secrets[key] ] }).to_io
|
||||||
end
|
|
||||||
|
|
||||||
def secrets
|
|
||||||
@secrets ||= secrets_keys.to_h { |key| [ key, ENV.fetch(key) ] }
|
|
||||||
end
|
|
||||||
|
|
||||||
def secrets_directory
|
|
||||||
File.dirname(secrets_file)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def merge(other)
|
def merge(other)
|
||||||
self.class.new \
|
self.class.new \
|
||||||
config: { "clear" => clear.merge(other.clear), "secret" => secrets_keys | other.secrets_keys },
|
config: { "clear" => clear.merge(other.clear), "secret" => secret_keys | other.secret_keys },
|
||||||
secrets_file: secrets_file || other.secrets_file
|
secrets: secrets
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
7
lib/kamal/configuration/env/tag.rb
vendored
7
lib/kamal/configuration/env/tag.rb
vendored
@@ -1,12 +1,13 @@
|
|||||||
class Kamal::Configuration::Env::Tag
|
class Kamal::Configuration::Env::Tag
|
||||||
attr_reader :name, :config
|
attr_reader :name, :config, :secrets
|
||||||
|
|
||||||
def initialize(name, config:)
|
def initialize(name, config:, secrets:)
|
||||||
@name = name
|
@name = name
|
||||||
@config = config
|
@config = config
|
||||||
|
@secrets = secrets
|
||||||
end
|
end
|
||||||
|
|
||||||
def env
|
def env
|
||||||
Kamal::Configuration::Env.new(config: config)
|
Kamal::Configuration::Env.new(config: config, secrets: secrets)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,63 +0,0 @@
|
|||||||
class Kamal::Configuration::Healthcheck
|
|
||||||
include Kamal::Configuration::Validation
|
|
||||||
|
|
||||||
attr_reader :healthcheck_config
|
|
||||||
|
|
||||||
def initialize(healthcheck_config:, context: "healthcheck")
|
|
||||||
@healthcheck_config = healthcheck_config || {}
|
|
||||||
validate! @healthcheck_config, context: context
|
|
||||||
end
|
|
||||||
|
|
||||||
def merge(other)
|
|
||||||
self.class.new healthcheck_config: healthcheck_config.deep_merge(other.healthcheck_config)
|
|
||||||
end
|
|
||||||
|
|
||||||
def cmd
|
|
||||||
healthcheck_config.fetch("cmd", http_health_check)
|
|
||||||
end
|
|
||||||
|
|
||||||
def port
|
|
||||||
healthcheck_config.fetch("port", 3000)
|
|
||||||
end
|
|
||||||
|
|
||||||
def path
|
|
||||||
healthcheck_config.fetch("path", "/up")
|
|
||||||
end
|
|
||||||
|
|
||||||
def max_attempts
|
|
||||||
healthcheck_config.fetch("max_attempts", 7)
|
|
||||||
end
|
|
||||||
|
|
||||||
def interval
|
|
||||||
healthcheck_config.fetch("interval", "1s")
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord
|
|
||||||
healthcheck_config.fetch("cord", "/tmp/kamal-cord")
|
|
||||||
end
|
|
||||||
|
|
||||||
def log_lines
|
|
||||||
healthcheck_config.fetch("log_lines", 50)
|
|
||||||
end
|
|
||||||
|
|
||||||
def set_port_or_path?
|
|
||||||
healthcheck_config["port"].present? || healthcheck_config["path"].present?
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_h
|
|
||||||
{
|
|
||||||
"cmd" => cmd,
|
|
||||||
"interval" => interval,
|
|
||||||
"max_attempts" => max_attempts,
|
|
||||||
"port" => port,
|
|
||||||
"path" => path,
|
|
||||||
"cord" => cord,
|
|
||||||
"log_lines" => log_lines
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
def http_health_check
|
|
||||||
"curl -f #{URI.join("http://localhost:#{port}", path)} || exit 1" if path.present? || port.present?
|
|
||||||
end
|
|
||||||
end
|
|
||||||
66
lib/kamal/configuration/proxy.rb
Normal file
66
lib/kamal/configuration/proxy.rb
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
class Kamal::Configuration::Proxy
|
||||||
|
include Kamal::Configuration::Validation
|
||||||
|
|
||||||
|
DEFAULT_LOG_REQUEST_HEADERS = [ "Cache-Control", "Last-Modified", "User-Agent" ]
|
||||||
|
CONTAINER_NAME = "kamal-proxy"
|
||||||
|
|
||||||
|
delegate :argumentize, :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
|
attr_reader :config, :proxy_config
|
||||||
|
|
||||||
|
def initialize(config:, proxy_config:, context: "proxy")
|
||||||
|
@config = config
|
||||||
|
@proxy_config = proxy_config
|
||||||
|
validate! @proxy_config, with: Kamal::Configuration::Validator::Proxy, context: context
|
||||||
|
end
|
||||||
|
|
||||||
|
def app_port
|
||||||
|
proxy_config.fetch("app_port", 80)
|
||||||
|
end
|
||||||
|
|
||||||
|
def ssl?
|
||||||
|
proxy_config.fetch("ssl", false)
|
||||||
|
end
|
||||||
|
|
||||||
|
def host
|
||||||
|
proxy_config["host"]
|
||||||
|
end
|
||||||
|
|
||||||
|
def deploy_options
|
||||||
|
{
|
||||||
|
host: proxy_config["host"],
|
||||||
|
tls: proxy_config["ssl"],
|
||||||
|
"deploy-timeout": seconds_duration(config.deploy_timeout),
|
||||||
|
"drain-timeout": seconds_duration(config.drain_timeout),
|
||||||
|
"health-check-interval": seconds_duration(proxy_config.dig("healthcheck", "interval")),
|
||||||
|
"health-check-timeout": seconds_duration(proxy_config.dig("healthcheck", "timeout")),
|
||||||
|
"health-check-path": proxy_config.dig("healthcheck", "path"),
|
||||||
|
"target-timeout": seconds_duration(proxy_config["response_timeout"]),
|
||||||
|
"buffer-requests": proxy_config.fetch("buffering", { "requests": true }).fetch("requests", true),
|
||||||
|
"buffer-responses": proxy_config.fetch("buffering", { "responses": true }).fetch("responses", true),
|
||||||
|
"buffer-memory": proxy_config.dig("buffering", "memory"),
|
||||||
|
"max-request-body": proxy_config.dig("buffering", "max_request_body"),
|
||||||
|
"max-response-body": proxy_config.dig("buffering", "max_response_body"),
|
||||||
|
"forward-headers": proxy_config.dig("forward_headers"),
|
||||||
|
"log-request-header": proxy_config.dig("logging", "request_headers") || DEFAULT_LOG_REQUEST_HEADERS,
|
||||||
|
"log-response-header": proxy_config.dig("logging", "response_headers")
|
||||||
|
}.compact
|
||||||
|
end
|
||||||
|
|
||||||
|
def deploy_command_args(target:)
|
||||||
|
optionize ({ target: "#{target}:#{app_port}" }).merge(deploy_options)
|
||||||
|
end
|
||||||
|
|
||||||
|
def remove_command_args(target:)
|
||||||
|
optionize({ target: "#{target}:#{app_port}" })
|
||||||
|
end
|
||||||
|
|
||||||
|
def merge(other)
|
||||||
|
self.class.new config: config, proxy_config: proxy_config.deep_merge(other.proxy_config)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def seconds_duration(value)
|
||||||
|
value ? "#{value}s" : nil
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,10 +1,11 @@
|
|||||||
class Kamal::Configuration::Registry
|
class Kamal::Configuration::Registry
|
||||||
include Kamal::Configuration::Validation
|
include Kamal::Configuration::Validation
|
||||||
|
|
||||||
attr_reader :registry_config
|
attr_reader :registry_config, :secrets
|
||||||
|
|
||||||
def initialize(config:)
|
def initialize(config:)
|
||||||
@registry_config = config.raw_config.registry || {}
|
@registry_config = config.raw_config.registry || {}
|
||||||
|
@secrets = config.secrets
|
||||||
validate! registry_config, with: Kamal::Configuration::Validator::Registry
|
validate! registry_config, with: Kamal::Configuration::Validator::Registry
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -23,7 +24,7 @@ class Kamal::Configuration::Registry
|
|||||||
private
|
private
|
||||||
def lookup(key)
|
def lookup(key)
|
||||||
if registry_config[key].is_a?(Array)
|
if registry_config[key].is_a?(Array)
|
||||||
ENV.fetch(registry_config[key].first).dup
|
secrets[registry_config[key].first]
|
||||||
else
|
else
|
||||||
registry_config[key]
|
registry_config[key]
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
class Kamal::Configuration::Role
|
class Kamal::Configuration::Role
|
||||||
include Kamal::Configuration::Validation
|
include Kamal::Configuration::Validation
|
||||||
|
|
||||||
CORD_FILE = "cord"
|
|
||||||
delegate :argumentize, :optionize, to: Kamal::Utils
|
delegate :argumentize, :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
attr_reader :name, :config, :specialized_env, :specialized_logging, :specialized_healthcheck
|
attr_reader :name, :config, :specialized_env, :specialized_logging, :specialized_proxy
|
||||||
|
|
||||||
alias to_s name
|
alias to_s name
|
||||||
|
|
||||||
@@ -18,16 +17,14 @@ class Kamal::Configuration::Role
|
|||||||
|
|
||||||
@specialized_env = Kamal::Configuration::Env.new \
|
@specialized_env = Kamal::Configuration::Env.new \
|
||||||
config: specializations.fetch("env", {}),
|
config: specializations.fetch("env", {}),
|
||||||
secrets_file: File.join(config.host_env_directory, "roles", "#{container_prefix}.env"),
|
secrets: config.secrets,
|
||||||
context: "servers/#{name}/env"
|
context: "servers/#{name}/env"
|
||||||
|
|
||||||
@specialized_logging = Kamal::Configuration::Logging.new \
|
@specialized_logging = Kamal::Configuration::Logging.new \
|
||||||
logging_config: specializations.fetch("logging", {}),
|
logging_config: specializations.fetch("logging", {}),
|
||||||
context: "servers/#{name}/logging"
|
context: "servers/#{name}/logging"
|
||||||
|
|
||||||
@specialized_healthcheck = Kamal::Configuration::Healthcheck.new \
|
initialize_specialized_proxy
|
||||||
healthcheck_config: specializations.fetch("healthcheck", {}),
|
|
||||||
context: "servers/#{name}/healthcheck"
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def primary_host
|
def primary_host
|
||||||
@@ -55,7 +52,7 @@ class Kamal::Configuration::Role
|
|||||||
end
|
end
|
||||||
|
|
||||||
def labels
|
def labels
|
||||||
default_labels.merge(traefik_labels).merge(custom_labels)
|
default_labels.merge(custom_labels)
|
||||||
end
|
end
|
||||||
|
|
||||||
def label_args
|
def label_args
|
||||||
@@ -70,6 +67,24 @@ class Kamal::Configuration::Role
|
|||||||
@logging ||= config.logging.merge(specialized_logging)
|
@logging ||= config.logging.merge(specialized_logging)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def proxy
|
||||||
|
@proxy ||= config.proxy.merge(specialized_proxy) if running_proxy?
|
||||||
|
end
|
||||||
|
|
||||||
|
def running_proxy?
|
||||||
|
@running_proxy
|
||||||
|
end
|
||||||
|
|
||||||
|
def ssl?
|
||||||
|
running_proxy? && proxy.ssl?
|
||||||
|
end
|
||||||
|
|
||||||
|
def stop_args
|
||||||
|
# When deploying with the proxy, kamal-proxy will drain request before returning so we don't need to wait.
|
||||||
|
timeout = running_proxy? ? nil : config.drain_timeout
|
||||||
|
|
||||||
|
[ *argumentize("-t", timeout) ]
|
||||||
|
end
|
||||||
|
|
||||||
def env(host)
|
def env(host)
|
||||||
@envs ||= {}
|
@envs ||= {}
|
||||||
@@ -77,7 +92,19 @@ class Kamal::Configuration::Role
|
|||||||
end
|
end
|
||||||
|
|
||||||
def env_args(host)
|
def env_args(host)
|
||||||
env(host).args
|
[ *env(host).clear_args, *argumentize("--env-file", secrets_path) ]
|
||||||
|
end
|
||||||
|
|
||||||
|
def env_directory
|
||||||
|
File.join(config.env_directory, "roles")
|
||||||
|
end
|
||||||
|
|
||||||
|
def secrets_io(host)
|
||||||
|
env(host).secrets_io
|
||||||
|
end
|
||||||
|
|
||||||
|
def secrets_path
|
||||||
|
File.join(config.env_directory, "roles", "#{name}.env")
|
||||||
end
|
end
|
||||||
|
|
||||||
def asset_volume_args
|
def asset_volume_args
|
||||||
@@ -85,72 +112,8 @@ class Kamal::Configuration::Role
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def health_check_args(cord: true)
|
|
||||||
if running_traefik? || healthcheck.set_port_or_path?
|
|
||||||
if cord && uses_cord?
|
|
||||||
optionize({ "health-cmd" => health_check_cmd_with_cord, "health-interval" => healthcheck.interval })
|
|
||||||
.concat(cord_volume.docker_args)
|
|
||||||
else
|
|
||||||
optionize({ "health-cmd" => healthcheck.cmd, "health-interval" => healthcheck.interval })
|
|
||||||
end
|
|
||||||
else
|
|
||||||
[]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def healthcheck
|
|
||||||
@healthcheck ||=
|
|
||||||
if running_traefik?
|
|
||||||
config.healthcheck.merge(specialized_healthcheck)
|
|
||||||
else
|
|
||||||
specialized_healthcheck
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def health_check_cmd_with_cord
|
|
||||||
"(#{healthcheck.cmd}) && (stat #{cord_container_file} > /dev/null || exit 1)"
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
def running_traefik?
|
|
||||||
if specializations["traefik"].nil?
|
|
||||||
primary?
|
|
||||||
else
|
|
||||||
specializations["traefik"]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def primary?
|
def primary?
|
||||||
self == @config.primary_role
|
name == @config.primary_role_name
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
def uses_cord?
|
|
||||||
running_traefik? && cord_volume && healthcheck.cmd.present?
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord_host_directory
|
|
||||||
File.join config.run_directory_as_docker_volume, "cords", [ container_prefix, config.run_id ].join("-")
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord_volume
|
|
||||||
if (cord = healthcheck.cord)
|
|
||||||
@cord_volume ||= Kamal::Configuration::Volume.new \
|
|
||||||
host_path: File.join(config.run_directory, "cords", [ container_prefix, config.run_id ].join("-")),
|
|
||||||
container_path: cord
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord_host_file
|
|
||||||
File.join cord_volume.host_path, CORD_FILE
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord_container_directory
|
|
||||||
health_check_options.fetch("cord", nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
def cord_container_file
|
|
||||||
File.join cord_volume.container_path, CORD_FILE
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
@@ -168,25 +131,52 @@ class Kamal::Configuration::Role
|
|||||||
end
|
end
|
||||||
|
|
||||||
def assets?
|
def assets?
|
||||||
asset_path.present? && running_traefik?
|
asset_path.present? && running_proxy?
|
||||||
end
|
end
|
||||||
|
|
||||||
def asset_volume(version = nil)
|
def asset_volume(version = config.version)
|
||||||
if assets?
|
if assets?
|
||||||
Kamal::Configuration::Volume.new \
|
Kamal::Configuration::Volume.new \
|
||||||
host_path: asset_volume_path(version), container_path: asset_path
|
host_path: asset_volume_directory(version), container_path: asset_path
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def asset_extracted_path(version = nil)
|
def asset_extracted_directory(version = config.version)
|
||||||
File.join config.run_directory, "assets", "extracted", container_name(version)
|
File.join config.assets_directory, "extracted", [ name, version ].join("-")
|
||||||
end
|
end
|
||||||
|
|
||||||
def asset_volume_path(version = nil)
|
def asset_volume_directory(version = config.version)
|
||||||
File.join config.run_directory, "assets", "volumes", container_name(version)
|
File.join config.assets_directory, "volumes", [ name, version ].join("-")
|
||||||
|
end
|
||||||
|
|
||||||
|
def ensure_one_host_for_ssl
|
||||||
|
if running_proxy? && proxy.ssl? && hosts.size > 1
|
||||||
|
raise Kamal::ConfigurationError, "SSL is only supported on a single server, found #{hosts.size} servers for role #{name}"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
def initialize_specialized_proxy
|
||||||
|
proxy_specializations = specializations["proxy"]
|
||||||
|
|
||||||
|
if primary?
|
||||||
|
# only false means no proxy for non-primary roles
|
||||||
|
@running_proxy = proxy_specializations != false
|
||||||
|
else
|
||||||
|
# false and nil both mean no proxy for non-primary roles
|
||||||
|
@running_proxy = !!proxy_specializations
|
||||||
|
end
|
||||||
|
|
||||||
|
if running_proxy?
|
||||||
|
proxy_config = proxy_specializations == true || proxy_specializations.nil? ? {} : proxy_specializations
|
||||||
|
|
||||||
|
@specialized_proxy = Kamal::Configuration::Proxy.new \
|
||||||
|
config: config,
|
||||||
|
proxy_config: proxy_config,
|
||||||
|
context: "servers/#{name}/proxy"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def tagged_hosts
|
def tagged_hosts
|
||||||
{}.tap do |tagged_hosts|
|
{}.tap do |tagged_hosts|
|
||||||
extract_hosts_from_config.map do |host_config|
|
extract_hosts_from_config.map do |host_config|
|
||||||
@@ -221,27 +211,6 @@ class Kamal::Configuration::Role
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def traefik_labels
|
|
||||||
if running_traefik?
|
|
||||||
{
|
|
||||||
# Setting a service property ensures that the generated service name will be consistent between versions
|
|
||||||
"traefik.http.services.#{traefik_service}.loadbalancer.server.scheme" => "http",
|
|
||||||
|
|
||||||
"traefik.http.routers.#{traefik_service}.rule" => "PathPrefix(`/`)",
|
|
||||||
"traefik.http.routers.#{traefik_service}.priority" => "2",
|
|
||||||
"traefik.http.middlewares.#{traefik_service}-retry.retry.attempts" => "5",
|
|
||||||
"traefik.http.middlewares.#{traefik_service}-retry.retry.initialinterval" => "500ms",
|
|
||||||
"traefik.http.routers.#{traefik_service}.middlewares" => "#{traefik_service}-retry@docker"
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def traefik_service
|
|
||||||
container_prefix
|
|
||||||
end
|
|
||||||
|
|
||||||
def custom_labels
|
def custom_labels
|
||||||
Hash.new.tap do |labels|
|
Hash.new.tap do |labels|
|
||||||
labels.merge!(config.labels) if config.labels.present?
|
labels.merge!(config.labels) if config.labels.present?
|
||||||
|
|||||||
@@ -26,8 +26,20 @@ class Kamal::Configuration::Ssh
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def keys_only
|
||||||
|
ssh_config["keys_only"]
|
||||||
|
end
|
||||||
|
|
||||||
|
def keys
|
||||||
|
ssh_config["keys"]
|
||||||
|
end
|
||||||
|
|
||||||
|
def key_data
|
||||||
|
ssh_config["key_data"]
|
||||||
|
end
|
||||||
|
|
||||||
def options
|
def options
|
||||||
{ user: user, port: port, proxy: proxy, logger: logger, keepalive: true, keepalive_interval: 30 }.compact
|
{ user: user, port: port, proxy: proxy, logger: logger, keepalive: true, keepalive_interval: 30, keys_only: keys_only, keys: keys, key_data: key_data }.compact
|
||||||
end
|
end
|
||||||
|
|
||||||
def to_h
|
def to_h
|
||||||
|
|||||||
@@ -1,60 +0,0 @@
|
|||||||
class Kamal::Configuration::Traefik
|
|
||||||
DEFAULT_IMAGE = "traefik:v2.10"
|
|
||||||
CONTAINER_PORT = 80
|
|
||||||
DEFAULT_ARGS = {
|
|
||||||
"log.level" => "DEBUG"
|
|
||||||
}
|
|
||||||
DEFAULT_LABELS = {
|
|
||||||
# These ensure we serve a 502 rather than a 404 if no containers are available
|
|
||||||
"traefik.http.routers.catchall.entryPoints" => "http",
|
|
||||||
"traefik.http.routers.catchall.rule" => "PathPrefix(`/`)",
|
|
||||||
"traefik.http.routers.catchall.service" => "unavailable",
|
|
||||||
"traefik.http.routers.catchall.priority" => 1,
|
|
||||||
"traefik.http.services.unavailable.loadbalancer.server.port" => "0"
|
|
||||||
}
|
|
||||||
|
|
||||||
include Kamal::Configuration::Validation
|
|
||||||
|
|
||||||
attr_reader :config, :traefik_config
|
|
||||||
|
|
||||||
def initialize(config:)
|
|
||||||
@config = config
|
|
||||||
@traefik_config = config.raw_config.traefik || {}
|
|
||||||
validate! traefik_config
|
|
||||||
end
|
|
||||||
|
|
||||||
def publish?
|
|
||||||
traefik_config["publish"] != false
|
|
||||||
end
|
|
||||||
|
|
||||||
def labels
|
|
||||||
DEFAULT_LABELS.merge(traefik_config["labels"] || {})
|
|
||||||
end
|
|
||||||
|
|
||||||
def env
|
|
||||||
Kamal::Configuration::Env.new \
|
|
||||||
config: traefik_config.fetch("env", {}),
|
|
||||||
secrets_file: File.join(config.host_env_directory, "traefik", "traefik.env"),
|
|
||||||
context: "traefik/env"
|
|
||||||
end
|
|
||||||
|
|
||||||
def host_port
|
|
||||||
traefik_config.fetch("host_port", CONTAINER_PORT)
|
|
||||||
end
|
|
||||||
|
|
||||||
def options
|
|
||||||
traefik_config.fetch("options", {})
|
|
||||||
end
|
|
||||||
|
|
||||||
def port
|
|
||||||
"#{host_port}:#{CONTAINER_PORT}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def args
|
|
||||||
DEFAULT_ARGS.merge(traefik_config.fetch("args", {}))
|
|
||||||
end
|
|
||||||
|
|
||||||
def image
|
|
||||||
traefik_config.fetch("image", DEFAULT_IMAGE)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -13,33 +13,40 @@ class Kamal::Configuration::Validator
|
|||||||
|
|
||||||
private
|
private
|
||||||
def validate_against_example!(validation_config, example)
|
def validate_against_example!(validation_config, example)
|
||||||
validate_type! validation_config, Hash
|
validate_type! validation_config, example.class
|
||||||
|
|
||||||
if (unknown_keys = validation_config.keys - example.keys).any?
|
if example.class == Hash
|
||||||
unknown_keys_error unknown_keys
|
check_unknown_keys! validation_config, example
|
||||||
end
|
|
||||||
|
|
||||||
validation_config.each do |key, value|
|
validation_config.each do |key, value|
|
||||||
with_context(key) do
|
next if extension?(key)
|
||||||
example_value = example[key]
|
with_context(key) do
|
||||||
|
example_value = example[key]
|
||||||
|
|
||||||
if example_value == "..."
|
if example_value == "..."
|
||||||
validate_type! value, *(Array if key == :servers), Hash
|
unless key.to_s == "proxy" && boolean?(value.class)
|
||||||
elsif key == "hosts"
|
validate_type! value, *(Array if key == :servers), Hash
|
||||||
validate_servers! value
|
end
|
||||||
elsif example_value.is_a?(Array)
|
elsif key == "hosts"
|
||||||
validate_array_of! value, example_value.first.class
|
validate_servers! value
|
||||||
elsif example_value.is_a?(Hash)
|
elsif example_value.is_a?(Array)
|
||||||
case key.to_s
|
if key == "arch"
|
||||||
when "options"
|
validate_array_of_or_type! value, example_value.first.class
|
||||||
validate_type! value, Hash
|
else
|
||||||
when "args", "labels"
|
validate_array_of! value, example_value.first.class
|
||||||
validate_hash_of! value, example_value.first[1].class
|
end
|
||||||
|
elsif example_value.is_a?(Hash)
|
||||||
|
case key.to_s
|
||||||
|
when "options", "args"
|
||||||
|
validate_type! value, Hash
|
||||||
|
when "labels"
|
||||||
|
validate_hash_of! value, example_value.first[1].class
|
||||||
|
else
|
||||||
|
validate_against_example! value, example_value
|
||||||
|
end
|
||||||
else
|
else
|
||||||
validate_against_example! value, example_value
|
validate_type! value, example_value.class
|
||||||
end
|
end
|
||||||
else
|
|
||||||
validate_type! value, example_value.class
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -70,6 +77,16 @@ class Kamal::Configuration::Validator
|
|||||||
value.is_a?(String) || value.is_a?(Symbol) || value.is_a?(Numeric) || value.is_a?(TrueClass) || value.is_a?(FalseClass)
|
value.is_a?(String) || value.is_a?(Symbol) || value.is_a?(Numeric) || value.is_a?(TrueClass) || value.is_a?(FalseClass)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def validate_array_of_or_type!(value, type)
|
||||||
|
if value.is_a?(Array)
|
||||||
|
validate_array_of! value, type
|
||||||
|
else
|
||||||
|
validate_type! value, type
|
||||||
|
end
|
||||||
|
rescue Kamal::ConfigurationError
|
||||||
|
type_error(Array, type)
|
||||||
|
end
|
||||||
|
|
||||||
def validate_array_of!(array, type)
|
def validate_array_of!(array, type)
|
||||||
validate_type! array, Array
|
validate_type! array, Array
|
||||||
|
|
||||||
@@ -137,4 +154,18 @@ class Kamal::Configuration::Validator
|
|||||||
ensure
|
ensure
|
||||||
@context = old_context
|
@context = old_context
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def allow_extensions?
|
||||||
|
false
|
||||||
|
end
|
||||||
|
|
||||||
|
def extension?(key)
|
||||||
|
key.to_s.start_with?("x-")
|
||||||
|
end
|
||||||
|
|
||||||
|
def check_unknown_keys!(config, example)
|
||||||
|
unknown_keys = config.keys - example.keys
|
||||||
|
unknown_keys.reject! { |key| extension?(key) } if allow_extensions?
|
||||||
|
unknown_keys_error unknown_keys if unknown_keys.present?
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
15
lib/kamal/configuration/validator/alias.rb
Normal file
15
lib/kamal/configuration/validator/alias.rb
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
class Kamal::Configuration::Validator::Alias < Kamal::Configuration::Validator
|
||||||
|
def validate!
|
||||||
|
super
|
||||||
|
|
||||||
|
name = context.delete_prefix("aliases/")
|
||||||
|
|
||||||
|
if name !~ /\A[a-z0-9_-]+\z/
|
||||||
|
error "Invalid alias name: '#{name}'. Must only contain lowercase letters, alphanumeric, hyphens and underscores."
|
||||||
|
end
|
||||||
|
|
||||||
|
if Kamal::Cli::Main.commands.include?(name)
|
||||||
|
error "Alias '#{name}' conflicts with a built-in command."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -5,5 +5,9 @@ class Kamal::Configuration::Validator::Builder < Kamal::Configuration::Validator
|
|||||||
if config["cache"] && config["cache"]["type"]
|
if config["cache"] && config["cache"]["type"]
|
||||||
error "Invalid cache type: #{config["cache"]["type"]}" unless [ "gha", "registry" ].include?(config["cache"]["type"])
|
error "Invalid cache type: #{config["cache"]["type"]}" unless [ "gha", "registry" ].include?(config["cache"]["type"])
|
||||||
end
|
end
|
||||||
|
|
||||||
|
error "Builder arch not set" unless config["arch"].present?
|
||||||
|
|
||||||
|
error "Cannot disable local builds, no remote is set" if config["local"] == false && config["remote"].blank?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
6
lib/kamal/configuration/validator/configuration.rb
Normal file
6
lib/kamal/configuration/validator/configuration.rb
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
class Kamal::Configuration::Validator::Configuration < Kamal::Configuration::Validator
|
||||||
|
private
|
||||||
|
def allow_extensions?
|
||||||
|
true
|
||||||
|
end
|
||||||
|
end
|
||||||
11
lib/kamal/configuration/validator/proxy.rb
Normal file
11
lib/kamal/configuration/validator/proxy.rb
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
class Kamal::Configuration::Validator::Proxy < Kamal::Configuration::Validator
|
||||||
|
def validate!
|
||||||
|
unless config.nil?
|
||||||
|
super
|
||||||
|
|
||||||
|
if config["host"].blank? && config["ssl"]
|
||||||
|
error "Must set a host to enable automatic SSL"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -15,6 +15,10 @@ class Kamal::EnvFile
|
|||||||
env_file.presence || "\n"
|
env_file.presence || "\n"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def to_io
|
||||||
|
StringIO.new(to_s)
|
||||||
|
end
|
||||||
|
|
||||||
alias to_str to_s
|
alias to_str to_s
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|||||||
@@ -9,6 +9,10 @@ module Kamal::Git
|
|||||||
`git config user.name`.strip
|
`git config user.name`.strip
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def email
|
||||||
|
`git config user.email`.strip
|
||||||
|
end
|
||||||
|
|
||||||
def revision
|
def revision
|
||||||
`git rev-parse HEAD`.strip
|
`git rev-parse HEAD`.strip
|
||||||
end
|
end
|
||||||
|
|||||||
37
lib/kamal/secrets.rb
Normal file
37
lib/kamal/secrets.rb
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
require "dotenv"
|
||||||
|
|
||||||
|
class Kamal::Secrets
|
||||||
|
attr_reader :secrets_files
|
||||||
|
|
||||||
|
Kamal::Secrets::Dotenv::InlineCommandSubstitution.install!
|
||||||
|
|
||||||
|
def initialize(destination: nil)
|
||||||
|
@secrets_files = \
|
||||||
|
[ ".kamal/secrets-common", ".kamal/secrets#{(".#{destination}" if destination)}" ].select { |f| File.exist?(f) }
|
||||||
|
@mutex = Mutex.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def [](key)
|
||||||
|
# Fetching secrets may ask the user for input, so ensure only one thread does that
|
||||||
|
@mutex.synchronize do
|
||||||
|
secrets.fetch(key)
|
||||||
|
end
|
||||||
|
rescue KeyError
|
||||||
|
if secrets_files
|
||||||
|
raise Kamal::ConfigurationError, "Secret '#{key}' not found in #{secrets_files.join(", ")}"
|
||||||
|
else
|
||||||
|
raise Kamal::ConfigurationError, "Secret '#{key}' not found, no secret files provided"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_h
|
||||||
|
secrets
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def secrets
|
||||||
|
@secrets ||= secrets_files.inject({}) do |secrets, secrets_file|
|
||||||
|
secrets.merge!(::Dotenv.parse(secrets_file))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
14
lib/kamal/secrets/adapters.rb
Normal file
14
lib/kamal/secrets/adapters.rb
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
require "active_support/core_ext/string/inflections"
|
||||||
|
module Kamal::Secrets::Adapters
|
||||||
|
def self.lookup(name)
|
||||||
|
name = "one_password" if name.downcase == "1password"
|
||||||
|
name = "last_pass" if name.downcase == "lastpass"
|
||||||
|
adapter_class(name)
|
||||||
|
end
|
||||||
|
|
||||||
|
def self.adapter_class(name)
|
||||||
|
Object.const_get("Kamal::Secrets::Adapters::#{name.camelize}").new
|
||||||
|
rescue NameError => e
|
||||||
|
raise RuntimeError, "Unknown secrets adapter: #{name}"
|
||||||
|
end
|
||||||
|
end
|
||||||
18
lib/kamal/secrets/adapters/base.rb
Normal file
18
lib/kamal/secrets/adapters/base.rb
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
class Kamal::Secrets::Adapters::Base
|
||||||
|
delegate :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
|
def fetch(secrets, account:, from: nil)
|
||||||
|
session = login(account)
|
||||||
|
full_secrets = secrets.map { |secret| [ from, secret ].compact.join("/") }
|
||||||
|
fetch_secrets(full_secrets, account: account, session: session)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def login(...)
|
||||||
|
raise NotImplementedError
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_secrets(...)
|
||||||
|
raise NotImplementedError
|
||||||
|
end
|
||||||
|
end
|
||||||
64
lib/kamal/secrets/adapters/bitwarden.rb
Normal file
64
lib/kamal/secrets/adapters/bitwarden.rb
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
class Kamal::Secrets::Adapters::Bitwarden < Kamal::Secrets::Adapters::Base
|
||||||
|
private
|
||||||
|
def login(account)
|
||||||
|
status = run_command("status")
|
||||||
|
|
||||||
|
if status["status"] == "unauthenticated"
|
||||||
|
run_command("login #{account.shellescape}", raw: true)
|
||||||
|
status = run_command("status")
|
||||||
|
end
|
||||||
|
|
||||||
|
if status["status"] == "locked"
|
||||||
|
session = run_command("unlock --raw", raw: true).presence
|
||||||
|
status = run_command("status", session: session)
|
||||||
|
end
|
||||||
|
|
||||||
|
raise RuntimeError, "Failed to login to and unlock Bitwarden" unless status["status"] == "unlocked"
|
||||||
|
|
||||||
|
run_command("sync", session: session, raw: true)
|
||||||
|
raise RuntimeError, "Failed to sync Bitwarden" unless $?.success?
|
||||||
|
|
||||||
|
session
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_secrets(secrets, account:, session:)
|
||||||
|
{}.tap do |results|
|
||||||
|
items_fields(secrets).each do |item, fields|
|
||||||
|
item_json = run_command("get item #{item.shellescape}", session: session, raw: true)
|
||||||
|
raise RuntimeError, "Could not read #{secret} from Bitwarden" unless $?.success?
|
||||||
|
item_json = JSON.parse(item_json)
|
||||||
|
|
||||||
|
if fields.any?
|
||||||
|
fields.each do |field|
|
||||||
|
item_field = item_json["fields"].find { |f| f["name"] == field }
|
||||||
|
raise RuntimeError, "Could not find field #{field} in item #{item} in Bitwarden" unless item_field
|
||||||
|
value = item_field["value"]
|
||||||
|
results["#{item}/#{field}"] = value
|
||||||
|
end
|
||||||
|
else
|
||||||
|
results[item] = item_json["login"]["password"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def items_fields(secrets)
|
||||||
|
{}.tap do |items|
|
||||||
|
secrets.each do |secret|
|
||||||
|
item, field = secret.split("/")
|
||||||
|
items[item] ||= []
|
||||||
|
items[item] << field
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def signedin?(account)
|
||||||
|
run_command("status")["status"] != "unauthenticated"
|
||||||
|
end
|
||||||
|
|
||||||
|
def run_command(command, session: nil, raw: false)
|
||||||
|
full_command = [ *("BW_SESSION=#{session.shellescape}" if session), "bw", command ].join(" ")
|
||||||
|
result = `#{full_command}`.strip
|
||||||
|
raw ? result : JSON.parse(result)
|
||||||
|
end
|
||||||
|
end
|
||||||
30
lib/kamal/secrets/adapters/last_pass.rb
Normal file
30
lib/kamal/secrets/adapters/last_pass.rb
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
class Kamal::Secrets::Adapters::LastPass < Kamal::Secrets::Adapters::Base
|
||||||
|
private
|
||||||
|
def login(account)
|
||||||
|
unless loggedin?(account)
|
||||||
|
`lpass login #{account.shellescape}`
|
||||||
|
raise RuntimeError, "Failed to login to 1Password" unless $?.success?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def loggedin?(account)
|
||||||
|
`lpass status --color never`.strip == "Logged in as #{account}."
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_secrets(secrets, account:, session:)
|
||||||
|
items = `lpass show #{secrets.map(&:shellescape).join(" ")} --json`
|
||||||
|
raise RuntimeError, "Could not read #{secrets} from 1Password" unless $?.success?
|
||||||
|
|
||||||
|
items = JSON.parse(items)
|
||||||
|
|
||||||
|
{}.tap do |results|
|
||||||
|
items.each do |item|
|
||||||
|
results[item["fullname"]] = item["password"]
|
||||||
|
end
|
||||||
|
|
||||||
|
if (missing_items = secrets - results.keys).any?
|
||||||
|
raise RuntimeError, "Could not find #{missing_items.join(", ")} in LassPass"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
61
lib/kamal/secrets/adapters/one_password.rb
Normal file
61
lib/kamal/secrets/adapters/one_password.rb
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
class Kamal::Secrets::Adapters::OnePassword < Kamal::Secrets::Adapters::Base
|
||||||
|
delegate :optionize, to: Kamal::Utils
|
||||||
|
|
||||||
|
private
|
||||||
|
def login(account)
|
||||||
|
unless loggedin?(account)
|
||||||
|
`op signin #{to_options(account: account, force: true, raw: true)}`.tap do
|
||||||
|
raise RuntimeError, "Failed to login to 1Password" unless $?.success?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def loggedin?(account)
|
||||||
|
`op account get --account #{account.shellescape} 2> /dev/null`
|
||||||
|
$?.success?
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_secrets(secrets, account:, session:)
|
||||||
|
{}.tap do |results|
|
||||||
|
vaults_items_fields(secrets).map do |vault, items|
|
||||||
|
items.each do |item, fields|
|
||||||
|
fields_json = JSON.parse(op_item_get(vault, item, fields, account: account, session: session))
|
||||||
|
fields_json = [ fields_json ] if fields.one?
|
||||||
|
|
||||||
|
fields_json.each do |field_json|
|
||||||
|
# The reference is in the form `op://vault/item/field[/field]`
|
||||||
|
field = field_json["reference"].delete_prefix("op://").delete_suffix("/password")
|
||||||
|
results[field] = field_json["value"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_options(**options)
|
||||||
|
optionize(options.compact).join(" ")
|
||||||
|
end
|
||||||
|
|
||||||
|
def vaults_items_fields(secrets)
|
||||||
|
{}.tap do |vaults|
|
||||||
|
secrets.each do |secret|
|
||||||
|
secret = secret.delete_prefix("op://")
|
||||||
|
vault, item, *fields = secret.split("/")
|
||||||
|
fields << "password" if fields.empty?
|
||||||
|
|
||||||
|
vaults[vault] ||= {}
|
||||||
|
vaults[vault][item] ||= []
|
||||||
|
vaults[vault][item] << fields.join(".")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def op_item_get(vault, item, fields, account:, session:)
|
||||||
|
labels = fields.map { |field| "label=#{field}" }.join(",")
|
||||||
|
options = to_options(vault: vault, fields: labels, format: "json", account: account, session: session.presence)
|
||||||
|
|
||||||
|
`op item get #{item.shellescape} #{options}`.tap do
|
||||||
|
raise RuntimeError, "Could not read #{fields.join(", ")} from #{item} in the #{vault} 1Password vault" unless $?.success?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
10
lib/kamal/secrets/adapters/test.rb
Normal file
10
lib/kamal/secrets/adapters/test.rb
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
class Kamal::Secrets::Adapters::Test < Kamal::Secrets::Adapters::Base
|
||||||
|
private
|
||||||
|
def login(account)
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_secrets(secrets, account:, session:)
|
||||||
|
secrets.to_h { |secret| [ secret, secret.reverse ] }
|
||||||
|
end
|
||||||
|
end
|
||||||
32
lib/kamal/secrets/dotenv/inline_command_substitution.rb
Normal file
32
lib/kamal/secrets/dotenv/inline_command_substitution.rb
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
class Kamal::Secrets::Dotenv::InlineCommandSubstitution
|
||||||
|
class << self
|
||||||
|
def install!
|
||||||
|
::Dotenv::Parser.substitutions.map! { |sub| sub == ::Dotenv::Substitutions::Command ? self : sub }
|
||||||
|
end
|
||||||
|
|
||||||
|
def call(value, _env, overwrite: false)
|
||||||
|
# Process interpolated shell commands
|
||||||
|
value.gsub(Dotenv::Substitutions::Command.singleton_class::INTERPOLATED_SHELL_COMMAND) do |*|
|
||||||
|
# Eliminate opening and closing parentheses
|
||||||
|
command = $LAST_MATCH_INFO[:cmd][1..-2]
|
||||||
|
|
||||||
|
if $LAST_MATCH_INFO[:backslash]
|
||||||
|
# Command is escaped, don't replace it.
|
||||||
|
$LAST_MATCH_INFO[0][1..]
|
||||||
|
else
|
||||||
|
if command =~ /\A\s*kamal\s*secrets\s+/
|
||||||
|
# Inline the command
|
||||||
|
inline_secrets_command(command)
|
||||||
|
else
|
||||||
|
# Execute the command and return the value
|
||||||
|
`#{command}`.chomp
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def inline_secrets_command(command)
|
||||||
|
Kamal::Cli::Main.start(command.shellsplit[1..] + [ "--inline" ]).chomp
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -3,6 +3,7 @@ require "sshkit/dsl"
|
|||||||
require "net/scp"
|
require "net/scp"
|
||||||
require "active_support/core_ext/hash/deep_merge"
|
require "active_support/core_ext/hash/deep_merge"
|
||||||
require "json"
|
require "json"
|
||||||
|
require "concurrent/atomic/semaphore"
|
||||||
|
|
||||||
class SSHKit::Backend::Abstract
|
class SSHKit::Backend::Abstract
|
||||||
def capture_with_info(*args, **kwargs)
|
def capture_with_info(*args, **kwargs)
|
||||||
|
|||||||
@@ -10,10 +10,11 @@ class Kamal::Tags
|
|||||||
|
|
||||||
def default_tags(config)
|
def default_tags(config)
|
||||||
{ recorded_at: Time.now.utc.iso8601,
|
{ recorded_at: Time.now.utc.iso8601,
|
||||||
performer: `whoami`.chomp,
|
performer: Kamal::Git.email.presence || `whoami`.chomp,
|
||||||
destination: config.destination,
|
destination: config.destination,
|
||||||
version: config.version,
|
version: config.version,
|
||||||
service_version: service_version(config) }
|
service_version: service_version(config),
|
||||||
|
service: config.service }
|
||||||
end
|
end
|
||||||
|
|
||||||
def service_version(config)
|
def service_version(config)
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user