-
-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathpodman.py
140 lines (109 loc) · 4.06 KB
/
podman.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
"""
Podman containers.
https://podman.io/
"""
import asyncio
import logging
import uuid
import shlex
import subprocess
from . import Container, ContainerConfigFile
class Podman(Container):
"""
Represents a pdoman container.
"""
def __init__(self, cfg):
super().__init__(cfg)
self.running_image = None
self.container_id = None
@classmethod
def dynamic_ssh_config(cls) -> bool:
return False
@classmethod
def config(cls, machine_id, cfgdata, cfgpath):
cfg = ContainerConfigFile(machine_id, cfgdata, cfgpath)
cfg.base_image = cfgdata["base_image"]
cfg.command = cfgdata["command"]
return cfg
async def prepare(self, manage=False):
"""
No need to prepare the container image as we can directly run it
"""
if manage:
raise RuntimeError("Docker image cannot be started in management mode")
async def launch(self):
logging.debug("[podman] launching container with ssh port %d", self.ssh_port)
self.running_image = (self.cfg.base_image.replace('/', '-').replace(':', '-')
+ "-" + str(uuid.uuid4()))
command = []
for part in shlex.split(self.cfg.command):
part = part.replace("{BASE_IMAGE}", str(self.cfg.base_image))
part = part.replace("{SSHPORT}", str(self.ssh_port))
part = part.replace("{IMAGENAME}", str(self.running_image))
command.append(part)
logging.debug(f"[podman] $ {' '.join(command)}")
process = await asyncio.create_subprocess_exec(
*command,
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=None
)
process.stdin.close()
# podman echoes the container id
line = await process.stdout.readline()
if line:
self.container_id = line.strip().decode()
if self.container_id:
logging.debug("[podman] spawned container with hash %s" % self.container_id)
else:
raise Exception("no container id was provided by podman, "
"pls investigate launch command")
ret = await process.wait()
if ret != 0:
self.running_image = None
self.container_id = None
raise Exception("failed to start podman container")
async def is_running(self):
if not self.running_image:
return False
command = ['podman', 'inspect', '-f', '\'{{.State.Running}}\'', self.running_image]
logging.debug(f"[podman] $ {' '.join(command)}")
process = await asyncio.create_subprocess_exec(
*command,
)
out, err = await process.communicate()
return 'true' in out.decode()
async def wait_for_shutdown(self, timeout=60):
if not self.running_image:
return
command = ['podman', 'wait', self.running_image]
logging.debug(f"[podman] $ {' '.join(command)}")
process = await asyncio.create_subprocess_exec(
*command,
)
try:
await asyncio.wait_for(process.wait(), timeout)
return True
except asyncio.TimeoutError:
logging.warning("[podman] shutdown wait timed out "
f"for container {self.running_image}")
return False
async def terminate(self):
if not self.running_image:
return
command = ['podman', 'stop', self.running_image]
logging.debug(f"[podman] $ {' '.join(command)}")
process = await asyncio.create_subprocess_exec(
*command,
)
await asyncio.wait_for(process.wait(), timeout=20)
self.process = None
async def cleanup(self):
if not self.running_image:
return
command = ["podman", "rm", "-f", self.running_image]
logging.debug(f"[podman] $ {' '.join(command)}")
process = await asyncio.create_subprocess_exec(
*command,
)
await asyncio.wait_for(process.wait(), timeout=20)