Allow hosts to be tagged so we can have host specific env variables.
We might want host specific env variables for things like datacenter
specific tags or testing GC settings on a specific host.
Right now you either need to set up a separate role, or have the app
be host aware.
Now you can define tag env variables and assign those to hosts.
For example:
```
servers:
- 1.1.1.1
- 1.1.1.2: tag1
- 1.1.1.2: tag2
- 1.1.1.3: [ tag1, tag2 ]
env_tags:
tag1:
ENV1: value1
tag2:
ENV2: value2
```
The tag env supports the full env format, allowing you to set secret and
clear values.
264 lines
6.7 KiB
Ruby
264 lines
6.7 KiB
Ruby
class Kamal::Configuration::Role
|
|
CORD_FILE = "cord"
|
|
delegate :argumentize, :optionize, to: Kamal::Utils
|
|
|
|
attr_accessor :name
|
|
alias to_s name
|
|
|
|
def initialize(name, config:)
|
|
@name, @config = name.inquiry, config
|
|
@tagged_hosts ||= extract_tagged_hosts_from_config
|
|
end
|
|
|
|
def primary_host
|
|
hosts.first
|
|
end
|
|
|
|
def hosts
|
|
tagged_hosts.keys
|
|
end
|
|
|
|
def env_tags(host)
|
|
tagged_hosts.fetch(host).collect { |tag| config.env_tag(tag) }
|
|
end
|
|
|
|
def cmd
|
|
specializations["cmd"]
|
|
end
|
|
|
|
def option_args
|
|
if args = specializations["options"]
|
|
optionize args
|
|
else
|
|
[]
|
|
end
|
|
end
|
|
|
|
def labels
|
|
default_labels.merge(traefik_labels).merge(custom_labels)
|
|
end
|
|
|
|
def label_args
|
|
argumentize "--label", labels
|
|
end
|
|
|
|
def logging_args
|
|
args = config.logging || {}
|
|
args.deep_merge!(specializations["logging"]) if specializations["logging"].present?
|
|
|
|
if args.any?
|
|
optionize({ "log-driver" => args["driver"] }.compact) +
|
|
argumentize("--log-opt", args["options"])
|
|
else
|
|
config.logging_args
|
|
end
|
|
end
|
|
|
|
|
|
def env(host)
|
|
@envs ||= {}
|
|
@envs[host] ||= [ base_env, specialized_env, *env_tags(host).map(&:env) ].reduce(:merge)
|
|
end
|
|
|
|
def env_args(host)
|
|
env(host).args
|
|
end
|
|
|
|
def asset_volume_args
|
|
asset_volume&.docker_args
|
|
end
|
|
|
|
|
|
def health_check_args(cord: true)
|
|
if health_check_cmd.present?
|
|
if cord && uses_cord?
|
|
optionize({ "health-cmd" => health_check_cmd_with_cord, "health-interval" => health_check_interval })
|
|
.concat(cord_volume.docker_args)
|
|
else
|
|
optionize({ "health-cmd" => health_check_cmd, "health-interval" => health_check_interval })
|
|
end
|
|
else
|
|
[]
|
|
end
|
|
end
|
|
|
|
def health_check_cmd
|
|
health_check_options["cmd"] || http_health_check(port: health_check_options["port"], path: health_check_options["path"])
|
|
end
|
|
|
|
def health_check_cmd_with_cord
|
|
"(#{health_check_cmd}) && (stat #{cord_container_file} > /dev/null || exit 1)"
|
|
end
|
|
|
|
def health_check_interval
|
|
health_check_options["interval"] || "1s"
|
|
end
|
|
|
|
|
|
def running_traefik?
|
|
if specializations["traefik"].nil?
|
|
primary?
|
|
else
|
|
specializations["traefik"]
|
|
end
|
|
end
|
|
|
|
def primary?
|
|
self == @config.primary_role
|
|
end
|
|
|
|
|
|
def uses_cord?
|
|
running_traefik? && cord_volume && health_check_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 = health_check_options["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
|
|
|
|
|
|
def container_name(version = nil)
|
|
[ container_prefix, version || config.version ].compact.join("-")
|
|
end
|
|
|
|
def container_prefix
|
|
[ config.service, name, config.destination ].compact.join("-")
|
|
end
|
|
|
|
|
|
def asset_path
|
|
specializations["asset_path"] || config.asset_path
|
|
end
|
|
|
|
def assets?
|
|
asset_path.present? && running_traefik?
|
|
end
|
|
|
|
def asset_volume(version = nil)
|
|
if assets?
|
|
Kamal::Configuration::Volume.new \
|
|
host_path: asset_volume_path(version), container_path: asset_path
|
|
end
|
|
end
|
|
|
|
def asset_extracted_path(version = nil)
|
|
File.join config.run_directory, "assets", "extracted", container_name(version)
|
|
end
|
|
|
|
def asset_volume_path(version = nil)
|
|
File.join config.run_directory, "assets", "volumes", container_name(version)
|
|
end
|
|
|
|
private
|
|
attr_accessor :config, :tagged_hosts
|
|
|
|
def extract_tagged_hosts_from_config
|
|
{}.tap do |tagged_hosts|
|
|
extract_hosts_from_config.map do |host_config|
|
|
if host_config.is_a?(Hash)
|
|
raise ArgumentError, "Multiple hosts found: #{host_config.inspect}" unless host_config.size == 1
|
|
|
|
host, tags = host_config.first
|
|
tagged_hosts[host] = Array(tags)
|
|
elsif host_config.is_a?(String) || host_config.is_a?(Symbol)
|
|
tagged_hosts[host_config] = []
|
|
else
|
|
raise ArgumentError, "Invalid host config: #{host_config.inspect}"
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def extract_hosts_from_config
|
|
if config.servers.is_a?(Array)
|
|
config.servers
|
|
else
|
|
servers = config.servers[name]
|
|
servers.is_a?(Array) ? servers : Array(servers["hosts"])
|
|
end
|
|
end
|
|
|
|
def default_labels
|
|
{ "service" => config.service, "role" => name, "destination" => config.destination }
|
|
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
|
|
Hash.new.tap do |labels|
|
|
labels.merge!(config.labels) if config.labels.present?
|
|
labels.merge!(specializations["labels"]) if specializations["labels"].present?
|
|
end
|
|
end
|
|
|
|
def specializations
|
|
if config.servers.is_a?(Array) || config.servers[name].is_a?(Array)
|
|
{}
|
|
else
|
|
config.servers[name].except("hosts")
|
|
end
|
|
end
|
|
|
|
def specialized_env
|
|
Kamal::Configuration::Env.from_config config: specializations.fetch("env", {})
|
|
end
|
|
|
|
# Secrets are stored in an array, which won't merge by default, so have to do it by hand.
|
|
def base_env
|
|
Kamal::Configuration::Env.from_config \
|
|
config: config.env,
|
|
secrets_file: File.join(config.host_env_directory, "roles", "#{container_prefix}.env")
|
|
end
|
|
|
|
def http_health_check(port:, path:)
|
|
"curl -f #{URI.join("http://localhost:#{port}", path)} || exit 1" if path.present? || port.present?
|
|
end
|
|
|
|
def health_check_options
|
|
@health_check_options ||= begin
|
|
options = specializations["healthcheck"] || {}
|
|
options = config.healthcheck.merge(options) if running_traefik?
|
|
options
|
|
end
|
|
end
|
|
end
|