Class: Kamal::Configuration

Inherits:
Object
  • Object
show all
Includes:
Validation
Defined in:
lib/kamal/configuration.rb

Defined Under Namespace

Modules: Validation Classes: Accessory, Alias, Boot, Builder, Env, Logging, Proxy, Registry, Role, Servers, Ssh, Sshkit, Validator, Volume

Constant Summary collapse

PROXY_MINIMUM_VERSION =
"v0.8.4"
PROXY_HTTP_PORT =
80
PROXY_HTTPS_PORT =
443
PROXY_LOG_MAX_SIZE =
"10m"

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Methods included from Validation

#validate!, #validation_yml

Constructor Details

#initialize(raw_config, destination: nil, version: nil, validate: true) ⇒ Configuration

Returns a new instance of Configuration.



51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
# File 'lib/kamal/configuration.rb', line 51

def initialize(raw_config, destination: nil, version: nil, validate: true)
  @raw_config = ActiveSupport::InheritableOptions.new(raw_config)
  @destination = destination
  @declared_version = version

  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
  @servers = Servers.new(config: self)
  @registry = Registry.new(config: @raw_config, secrets: secrets)

  @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)
  @builder = Builder.new(config: self)
  @env = Env.new(config: @raw_config.env || {}, secrets: secrets)

  @logging = Logging.new(logging_config: @raw_config.logging)
  @proxy = Proxy.new(config: self, proxy_config: @raw_config.proxy || {})
  @ssh = Ssh.new(config: self)
  @sshkit = Sshkit.new(config: self)

  ensure_destination_if_required
  ensure_required_keys_present
  ensure_valid_kamal_version
  ensure_retain_containers_valid
  ensure_valid_service_name
  ensure_no_traefik_reboot_hooks
  ensure_one_host_for_ssl_roles
  ensure_unique_hosts_for_ssl_roles
end

Instance Attribute Details

#accessoriesObject (readonly)

Returns the value of attribute accessories.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def accessories
  @accessories
end

#aliasesObject (readonly)

Returns the value of attribute aliases.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def aliases
  @aliases
end

#bootObject (readonly)

Returns the value of attribute boot.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def boot
  @boot
end

#builderObject (readonly)

Returns the value of attribute builder.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def builder
  @builder
end

#destinationObject (readonly)

Returns the value of attribute destination.



12
13
14
# File 'lib/kamal/configuration.rb', line 12

def destination
  @destination
end

#envObject (readonly)

Returns the value of attribute env.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def env
  @env
end

#loggingObject (readonly)

Returns the value of attribute logging.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def logging
  @logging
end

#proxyObject (readonly)

Returns the value of attribute proxy.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def proxy
  @proxy
end

#raw_configObject (readonly)

Returns the value of attribute raw_config.



12
13
14
# File 'lib/kamal/configuration.rb', line 12

def raw_config
  @raw_config
end

#registryObject (readonly)

Returns the value of attribute registry.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def registry
  @registry
end

#secretsObject (readonly)

Returns the value of attribute secrets.



12
13
14
# File 'lib/kamal/configuration.rb', line 12

def secrets
  @secrets
end

#serversObject (readonly)

Returns the value of attribute servers.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def servers
  @servers
end

#sshObject (readonly)

Returns the value of attribute ssh.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def ssh
  @ssh
end

#sshkitObject (readonly)

Returns the value of attribute sshkit.



13
14
15
# File 'lib/kamal/configuration.rb', line 13

def sshkit
  @sshkit
end

Class Method Details

.create_from(config_file:, destination: nil, version: nil) ⇒ Object



23
24
25
26
27
28
29
# File 'lib/kamal/configuration.rb', line 23

def create_from(config_file:, destination: nil, version: nil)
  ENV["KAMAL_DESTINATION"] = destination

  raw_config = load_config_files(config_file, *destination_config_file(config_file, destination))

  new raw_config, destination: destination, version: version
end

Instance Method Details

#abbreviated_versionObject



93
94
95
96
97
98
99
100
101
102
# File 'lib/kamal/configuration.rb', line 93

def abbreviated_version
  if version
    # Don't abbreviate <sha>_uncommitted_<etc>
    if version.include?("_")
      version
    else
      version[0...7]
    end
  end
end

#absolute_imageObject



156
157
158
# File 'lib/kamal/configuration.rb', line 156

def absolute_image
  "#{repository}:#{version}"
end

#accessory(name) ⇒ Object



116
117
118
# File 'lib/kamal/configuration.rb', line 116

def accessory(name)
  accessories.detect { |a| a.name == name.to_s }
end

#all_hostsObject



120
121
122
# File 'lib/kamal/configuration.rb', line 120

def all_hosts
  (roles + accessories).flat_map(&:hosts).uniq
end

#allow_empty_roles?Boolean

Returns:

  • (Boolean)


136
137
138
# File 'lib/kamal/configuration.rb', line 136

def allow_empty_roles?
  raw_config.allow_empty_roles
end

#app_directoryObject



212
213
214
# File 'lib/kamal/configuration.rb', line 212

def app_directory
  File.join apps_directory, [ service, destination ].compact.join("-")
end

#apps_directoryObject



208
209
210
# File 'lib/kamal/configuration.rb', line 208

def apps_directory
  File.join run_directory, "apps"
end

#asset_pathObject



228
229
230
# File 'lib/kamal/configuration.rb', line 228

def asset_path
  raw_config.asset_path
end

#assets_directoryObject



220
221
222
# File 'lib/kamal/configuration.rb', line 220

def assets_directory
  File.join app_directory, "assets"
end

#deploy_timeoutObject



196
197
198
# File 'lib/kamal/configuration.rb', line 196

def deploy_timeout
  raw_config.deploy_timeout || 30
end

#drain_timeoutObject



200
201
202
# File 'lib/kamal/configuration.rb', line 200

def drain_timeout
  raw_config.drain_timeout || 30
end

#env_directoryObject



216
217
218
# File 'lib/kamal/configuration.rb', line 216

def env_directory
  File.join app_directory, "env"
end

#env_tag(name) ⇒ Object



240
241
242
# File 'lib/kamal/configuration.rb', line 240

def env_tag(name)
  env_tags.detect { |t| t.name == name.to_s }
end

#env_tagsObject



232
233
234
235
236
237
238
# File 'lib/kamal/configuration.rb', line 232

def env_tags
  @env_tags ||= if (tags = raw_config.env["tags"])
    tags.collect { |name, config| Env::Tag.new(name, config: config, secrets: secrets) }
  else
    []
  end
end

#hooks_pathObject



224
225
226
# File 'lib/kamal/configuration.rb', line 224

def hooks_path
  raw_config.hooks_path || ".kamal/hooks"
end

#latest_imageObject



160
161
162
# File 'lib/kamal/configuration.rb', line 160

def latest_image
  "#{repository}:#{latest_tag}"
end

#latest_tagObject



164
165
166
# File 'lib/kamal/configuration.rb', line 164

def latest_tag
  [ "latest", *destination ].join("-")
end

#logging_argsObject



188
189
190
# File 'lib/kamal/configuration.rb', line 188

def logging_args
  logging.args
end

#minimum_versionObject



104
105
106
# File 'lib/kamal/configuration.rb', line 104

def minimum_version
  raw_config.minimum_version
end

#primary_hostObject



124
125
126
# File 'lib/kamal/configuration.rb', line 124

def primary_host
  primary_role&.primary_host
end

#primary_roleObject



132
133
134
# File 'lib/kamal/configuration.rb', line 132

def primary_role
  role(primary_role_name)
end

#primary_role_nameObject



128
129
130
# File 'lib/kamal/configuration.rb', line 128

def primary_role_name
  raw_config.primary_role || "web"
end

#proxy_container_nameObject



268
269
270
# File 'lib/kamal/configuration.rb', line 268

def proxy_container_name
  "kamal-proxy"
end

#proxy_directoryObject



272
273
274
# File 'lib/kamal/configuration.rb', line 272

def proxy_directory
  File.join run_directory, "proxy"
end

#proxy_hostsObject



148
149
150
# File 'lib/kamal/configuration.rb', line 148

def proxy_hosts
  proxy_roles.flat_map(&:hosts).uniq
end

#proxy_imageObject



264
265
266
# File 'lib/kamal/configuration.rb', line 264

def proxy_image
  "basecamp/kamal-proxy:#{PROXY_MINIMUM_VERSION}"
end

#proxy_logging_args(max_size) ⇒ Object



256
257
258
# File 'lib/kamal/configuration.rb', line 256

def proxy_logging_args(max_size)
  argumentize "--log-opt", "max-size=#{max_size}" if max_size.present?
end

#proxy_options_defaultObject



260
261
262
# File 'lib/kamal/configuration.rb', line 260

def proxy_options_default
  [ *proxy_publish_args(PROXY_HTTP_PORT, PROXY_HTTPS_PORT), *proxy_logging_args(PROXY_LOG_MAX_SIZE) ]
end

#proxy_options_fileObject



276
277
278
# File 'lib/kamal/configuration.rb', line 276

def proxy_options_file
  File.join proxy_directory, "options"
end

#proxy_publish_args(http_port, https_port, bind_ips = nil) ⇒ Object



244
245
246
247
248
249
250
251
252
253
254
# File 'lib/kamal/configuration.rb', line 244

def proxy_publish_args(http_port, https_port, bind_ips = nil)
  ensure_valid_bind_ips(bind_ips)

  (bind_ips || [ nil ]).map do |bind_ip|
    bind_ip = format_bind_ip(bind_ip)
    publish_http = [ bind_ip, http_port, PROXY_HTTP_PORT ].compact.join(":")
    publish_https = [ bind_ip, https_port, PROXY_HTTPS_PORT ].compact.join(":")

    argumentize "--publish", [ publish_http, publish_https ]
  end.join(" ")
end

#proxy_role_namesObject



144
145
146
# File 'lib/kamal/configuration.rb', line 144

def proxy_role_names
  proxy_roles.flat_map(&:name)
end

#proxy_rolesObject



140
141
142
# File 'lib/kamal/configuration.rb', line 140

def proxy_roles
  roles.select(&:running_proxy?)
end

#readiness_delayObject



192
193
194
# File 'lib/kamal/configuration.rb', line 192

def readiness_delay
  raw_config.readiness_delay || 7
end

#repositoryObject



152
153
154
# File 'lib/kamal/configuration.rb', line 152

def repository
  [ registry.server, image ].compact.join("/")
end

#require_destination?Boolean

Returns:

  • (Boolean)


172
173
174
# File 'lib/kamal/configuration.rb', line 172

def require_destination?
  raw_config.require_destination
end

#retain_containersObject



176
177
178
# File 'lib/kamal/configuration.rb', line 176

def retain_containers
  raw_config.retain_containers || 5
end

#role(name) ⇒ Object



112
113
114
# File 'lib/kamal/configuration.rb', line 112

def role(name)
  roles.detect { |r| r.name == name.to_s }
end

#rolesObject



108
109
110
# File 'lib/kamal/configuration.rb', line 108

def roles
  servers.roles
end

#run_directoryObject



204
205
206
# File 'lib/kamal/configuration.rb', line 204

def run_directory
  ".kamal"
end

#service_with_versionObject



168
169
170
# File 'lib/kamal/configuration.rb', line 168

def service_with_version
  "#{service}-#{version}"
end

#to_hObject



280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
# File 'lib/kamal/configuration.rb', line 280

def to_h
  {
    roles: role_names,
    hosts: all_hosts,
    primary_host: primary_host,
    version: version,
    repository: repository,
    absolute_image: absolute_image,
    service_with_version: service_with_version,
    volume_args: volume_args,
    ssh_options: ssh.to_h,
    sshkit: sshkit.to_h,
    builder: builder.to_h,
    accessories: raw_config.accessories,
    logging: logging_args
  }.compact
end

#versionObject



89
90
91
# File 'lib/kamal/configuration.rb', line 89

def version
  @declared_version.presence || ENV["VERSION"] || git_version
end

#version=(version) ⇒ Object



85
86
87
# File 'lib/kamal/configuration.rb', line 85

def version=(version)
  @declared_version = version
end

#volume_argsObject



180
181
182
183
184
185
186
# File 'lib/kamal/configuration.rb', line 180

def volume_args
  if raw_config.volumes.present?
    argumentize "--volume", raw_config.volumes
  else
    []
  end
end