docker-py を使用して、Docker イメージをビルドおよび実行しています。
ドキュメントを読んでも、イメージの構築中にエラーが発生したかどうかをどのように検出すればよいかわかりません。 build()
エラーが発生しても例外を発生させません。そのため、戻ってくる応答を調査する必要があると思います。
docker-py の client.build() が失敗したかどうかを判断する最良の方法は何ですか?
共有の精神で、Fabien の回答 (非常に役に立ちました) の進歩である実装を示したいと思います。これにより、印刷、クリーンアップなどが行われ、状況が悪い場合に通知された例外がスローされます。
import json
import logging
import re
import docker
log = logging.getLogger(__name__)
class StreamLineBuildGenerator(object):
def __init__(self, json_data):
self.__dict__ = json.loads(json_data)
class Docker(object):
REGISTRY = "some_docker_registry"
def __init__(self):
self.client = docker.from_env()
self.api_client = docker.APIClient()
def build(self, path, repository):
tag = "{}/{}".format(Docker.REGISTRY, repository)
output = self.api_client.build(path=path, tag=tag)
self._process_output(output)
log.info("done building {}".format(repository))
def push(self, repository):
tag = "{}/{}".format(Docker.REGISTRY, repository)
output = self.client.images.push(tag)
self._process_output(output)
log.info("done pushing {}".format(tag))
def _process_output(self, output):
if type(output) == str:
output = output.split("\n")
for line in output:
if line:
errors = set()
try:
stream_line = StreamLineBuildGenerator(line)
if hasattr(stream_line, "status"):
log.info(stream_line.status)
elif hasattr(stream_line, "stream"):
stream = re.sub("^\n", "", stream_line.stream)
stream = re.sub("\n$", "", stream)
# found after newline to close (red) "error" blocks: 27 91 48 109
stream = re.sub("\n(\x1B\[0m)$", "\\1", stream)
if stream:
log.info(stream)
elif hasattr(stream_line, "aux"):
if hasattr(stream_line.aux, "Digest"):
log.info("digest: {}".format(stream_line.aux["Digest"]))
if hasattr(stream_line.aux, "ID"):
log.info("ID: {}".format(stream_line.aux["ID"]))
else:
log.info("not recognized (1): {}".format(line))
if hasattr(stream_line, "error"):
errors.add(stream_line.error)
if hasattr(stream_line, "errorDetail"):
errors.add(stream_line.errorDetail["message"])
if hasattr(stream_line.errorDetail, "code"):
error_code = stream_line.errorDetail["code"]
errors.add("Error code: {}".format(error_code))
except ValueError as e:
log.error("not recognized (2): {}".format(line))
if errors:
message = "problem executing Docker: {}".format(". ".join(errors))
raise SystemError(message)
StreamLineBuilder ジェネレーターを作成します。
import json
class StreamLineBuildGenerator(object):
def __init__(self, json_data):
self.__dict__ = json.loads(json_data)
そして、このジェネレーターを使用してストリームを解析します。
import docker
docker_client = docker.Client(version="1.18", base_url="unix:///var/run/docker.sock")
generator = docker_client.build(nocache=False, rm=True, stream=True, tag="my_image_tag", path="my_path")
for line in generator:
try:
stream_line = StreamLineBuildGenerator(line)
if hasattr(stream_line, "error"):
print(stream_line.error)
if hasattr(stream_line, "errorDetail"):
if not stream_line.error == stream_line.errorDetail["message"]:
if hasattr(stream_line.errorDetail, "code"):
print("[" + stream_line.errorDetail["code"] + "] ", False)
print(stream_line.errorDetail["message"])
except ValueError:
# If we are not able to deserialize the received line as JSON object, just print it out
print(line)
continue