use std::collections::HashMap;
use crate::compose::types::{ComposeFile, Service};
use crate::error::{ComposeError, Result};
use crate::libpod::types::container::{LinuxResources, Namespace, SpecGenerator};
use crate::libpod::urlencoded;
use crate::libpod::API_PREFIX;
use crate::{ports, size};
mod resolve;
use resolve::{build_env, resolve_links, resolve_stop_signal, resolve_volume_name};
pub(crate) use resolve::{config_hash, resolve_bind_source};
use super::container_config::{
build_healthcheck, build_log_config, build_resource_limits, build_restart_policy,
build_ulimits, cdi_devices,
};
use super::container_fields::{
build_blkio_config, build_label_file_labels, parse_device, resolve_container_labels,
warn_swarm_only_deploy,
};
use super::network::resolve_network_mode;
use super::volume_mounts::build_mounts_all;
use super::Engine;
impl Engine {
pub(super) async fn create_and_start(
&self,
container_name: &str,
service_name: &str,
service: &Service,
file: &ComposeFile,
start: bool,
) -> Result<()> {
let derived_image;
let image: &str = if let Some(img) = service.image.as_deref() {
img
} else if service.build.is_some() {
derived_image = format!("{}:latest", service_name);
&derived_image
} else {
return Err(ComposeError::NoImageOrBuild(service_name.into()));
};
warn_swarm_only_deploy(service_name, service);
let env: HashMap<String, String> = build_env(service, &self.base_dir)?
.into_iter()
.filter_map(|s| match s.find('=') {
Some(idx) => Some((s[..idx].to_string(), s[idx + 1..].to_string())),
None => std::env::var(&s).ok().map(|v| (s, v)),
})
.collect();
let secret_binds = self.build_secret_binds(service, file)?;
let config_binds = self.build_config_binds(service, file)?;
let native_secrets = self.build_native_secrets(service, file).await?;
let (mut mounts, mut named_volumes) =
build_mounts_all(service, &self.base_dir, &secret_binds, &config_binds);
for m in &mut mounts {
if m.mount_type == "bind" {
if let Some(src) = m.source.take() {
m.source = Some(resolve_bind_source(&src, &self.base_dir));
}
}
}
for nv in &mut named_volumes {
nv.name = self.resolved_volume_name(&nv.name, file);
}
let parsed_ports = ports::parse_ports(&service.ports)?;
let portmappings = ports::to_libpod(&parsed_ports);
let mut expose: HashMap<u16, String> = parsed_ports
.iter()
.map(|p| (p.container_port, p.protocol.clone()))
.collect();
for raw in &service.expose {
let (port_str, proto) = if let Some(idx) = raw.rfind('/') {
(&raw[..idx], raw[idx + 1..].to_string())
} else {
(raw.as_str(), "tcp".to_string())
};
if let Ok(p) = port_str.parse::<u16>() {
expose.entry(p).or_insert(proto);
}
}
let (restart_policy, restart_tries) = build_restart_policy(service);
let log_configuration = build_log_config(service.logging.as_ref());
let (netns, networks) = resolve_network_mode(service_name, service, file, &self.project);
let label_file_labels = build_label_file_labels(service, &self.base_dir);
let mut labels = resolve_container_labels(service, label_file_labels);
labels.insert("podup.project".to_string(), self.project.clone());
labels.insert("podup.service".to_string(), service_name.to_string());
labels.insert("podup.config-hash".to_string(), config_hash(service, file)?);
let annotations: HashMap<String, String> = service.annotations.to_map();
let sysctl: HashMap<String, String> = service.sysctls.to_map();
let mut resource_limits = build_resource_limits(service);
if let Some(blkio) = build_blkio_config(service) {
resource_limits
.get_or_insert_with(LinuxResources::default)
.block_io = Some(blkio);
}
let ulimits = build_ulimits(service);
let devices: Vec<_> = service.devices.iter().map(|s| parse_device(s)).collect();
let pidns = service.pid.as_deref().map(Namespace::parse);
let ipcns = service.ipc.as_deref().map(Namespace::parse);
let utsns = service.uts.as_deref().map(Namespace::parse);
let cgroupns = service.cgroup.as_deref().map(Namespace::parse);
let userns = service.userns_mode.as_deref().map(Namespace::parse);
let (image_os, image_arch) = service
.platform
.as_deref()
.and_then(|p| p.split_once('/'))
.map(|(os, arch)| (Some(os.to_string()), Some(arch.to_string())))
.unwrap_or((None, None));
let links = resolve_links(service, file, &self.project);
let shm_size = service.shm_size.as_deref().and_then(size::parse_memory);
let stop_timeout = service
.stop_grace_period
.as_deref()
.and_then(size::parse_duration_secs);
if service.mac_address.is_some() {
tracing::warn!(
"service \"{service_name}\": top-level mac_address is deprecated; \
move it to networks.<network>.mac_address"
);
}
for warning in rootless_caveat_warnings(service_name, service) {
tracing::warn!("{warning}");
}
let stop_signal = service
.stop_signal
.as_deref()
.map(resolve_stop_signal)
.transpose()?;
let spec = SpecGenerator {
name: container_name.to_string(),
image: image.to_string(),
command: service.command.as_ref().map(|c| c.to_exec()),
entrypoint: service.entrypoint.as_ref().map(|c| c.to_exec()),
env,
terminal: service.tty,
stdin: service.stdin_open,
user: service.user.clone(),
work_dir: service.working_dir.clone(),
stop_signal,
stop_timeout,
hostname: service.hostname.clone(),
domainname: service.domainname.clone(),
labels,
annotations,
cap_add: service.cap_add.clone(),
cap_drop: service.cap_drop.clone(),
privileged: service.privileged,
read_only_filesystem: service.read_only,
security_opt: service.security_opt.clone(),
sysctl,
expose,
portmappings,
networks,
netns,
extra_hosts: service.extra_hosts.clone(),
dns_server: service.dns.to_list(),
dns_search: service.dns_search.to_list(),
dns_option: service.dns_opt.to_list(),
mounts,
volumes: named_volumes,
volumes_from: service.volumes_from.clone(),
secrets: native_secrets,
userns,
pidns,
ipcns,
utsns,
cgroupns,
cgroup_parent: service.cgroup_parent.clone(),
resource_limits,
ulimits,
shm_size,
healthconfig: service.healthcheck.as_ref().map(build_healthcheck),
log_configuration,
init: service.init,
restart_policy,
restart_tries,
devices,
cdi_devices: cdi_devices(service),
device_cgroup_rule: service.device_cgroup_rules.clone(),
groups: service.group_add.clone(),
oom_score_adj: service.oom_score_adj,
runtime: service.runtime.clone(),
links,
image_os,
image_arch,
storage_opts: service.storage_opt.clone(),
..Default::default()
};
let rm_path = format!(
"{API_PREFIX}/containers/{}?force=true&v={}",
urlencoded(container_name),
self.renew_anon_volumes,
);
if let Err(e) = self.client.delete_ok(&rm_path).await {
tracing::debug!("pre-create delete {container_name}: {e}");
}
self.client
.post_json::<_, serde_json::Value>(&format!("{API_PREFIX}/containers/create"), &spec)
.await
.map_err(ComposeError::Podman)?;
if start {
let start_path = format!(
"{API_PREFIX}/containers/{}/start",
urlencoded(container_name)
);
self.client
.post_empty_ok(&start_path)
.await
.map_err(ComposeError::Podman)?;
}
Ok(())
}
fn resolved_volume_name(&self, reference: &str, file: &ComposeFile) -> String {
resolve_volume_name(reference, &self.project, file)
}
}
fn rootless_caveat_warnings(name: &str, service: &Service) -> Vec<String> {
let mut out = Vec::new();
if service.privileged == Some(true) {
out.push(format!(
"service \"{name}\": privileged has reduced effect under rootless Podman — a \
container cannot gain more privileges than the user that launched it"
));
}
if service.oom_kill_disable.is_some() {
out.push(format!(
"service \"{name}\": oom_kill_disable is not supported on cgroups v2 systems and \
is ignored"
));
}
if service.mem_swappiness.is_some() {
out.push(format!(
"service \"{name}\": mem_swappiness is only supported on cgroups v1 rootful systems \
and is ignored otherwise"
));
}
if service.cpu_rt_runtime.is_some() || service.cpu_rt_period.is_some() {
out.push(format!(
"service \"{name}\": cpu_rt_runtime/cpu_rt_period are only supported on cgroups v1 \
rootful systems; the container may fail to start rootless"
));
}
if !service.links.is_empty() {
out.push(format!(
"service \"{name}\": links has no effect under rootless Podman networking — put the \
services on a shared network and reach them by service name instead"
));
}
if !service.external_links.is_empty() {
out.push(format!(
"service \"{name}\": external_links has no effect under rootless Podman networking — \
attach the target container to a shared network and reach it by service name instead"
));
}
out
}
#[cfg(test)]
mod tests {
use super::rootless_caveat_warnings;
use crate::compose::types::Service;
#[test]
fn no_caveat_warnings_for_plain_service() {
assert!(rootless_caveat_warnings("web", &Service::default()).is_empty());
}
#[test]
fn warns_for_each_rootless_caveat_field() {
let service = Service {
privileged: Some(true),
oom_kill_disable: Some(true),
mem_swappiness: Some(10),
cpu_rt_runtime: Some(1000),
links: vec!["db".into()],
external_links: vec!["legacy_db:db".into()],
..Service::default()
};
let warnings = rootless_caveat_warnings("web", &service);
assert_eq!(warnings.len(), 6);
let joined = warnings.join("\n");
for needle in [
"privileged",
"oom_kill_disable",
"mem_swappiness",
"cpu_rt_runtime",
"links",
"external_links",
] {
assert!(joined.contains(needle), "missing warning for {needle}");
}
}
}