Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Retry docker login/push #588

Merged
merged 1 commit into from
Jan 3, 2025
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
46 changes: 35 additions & 11 deletions services/orion-builder/src/orion_builder/push.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,13 @@
import argparse
import logging
import sys
from functools import wraps
from os import getenv
from pathlib import Path
from shutil import rmtree
from subprocess import PIPE
from subprocess import PIPE, CalledProcessError
from tempfile import mkdtemp
from time import sleep
from typing import List, Optional

import taskcluster
Expand All @@ -25,6 +27,24 @@
LOG = logging.getLogger(__name__)


def retry_call(f, retries=5, initial_delay=5):
"""Retry a call to subprocess.* which raises CalledProcessError."""

@wraps(f)
def wrapper(*args, **kwds):
delay = initial_delay
for _ in range(retries):
try:
return f(*args, **kwds)
except CalledProcessError:
LOG.warning("call failed: %s, retrying in %ds...", args, delay)
sleep(delay)
delay *= 2
return f(*args, **kwds)

return wrapper


class PushArgs(CommonArgs):
"""CLI arguments for Orion pusher"""

Expand Down Expand Up @@ -117,8 +137,7 @@ def main(argv: Optional[List[str]] = None) -> None:
LOG.info(
"Task %s artifact %s downloaded to: %s", task_id, artifact_name, img
)
existing_images = tool.list_images()
LOG.debug("Existing images before loading: %s", existing_images)
LOG.debug("Existing images before loading: %s", tool.list_images())

# 1. Load image/s artifact into the podman image store
load_result = tool.run(
Expand Down Expand Up @@ -166,10 +185,12 @@ def main(argv: Optional[List[str]] = None) -> None:
LOG.info(f"Manifest created: {create_result}")

# 3. Add the loaded images to the manifest
inspect_result = tool.run(
["manifest", "inspect", manifest_name], text=True, stdout=PIPE
LOG.debug(
"Manifest before adding images: %s",
tool.run(
["manifest", "inspect", manifest_name], text=True, stdout=PIPE
).stdout,
)
LOG.debug("Manifest before adding images: %s", inspect_result)
for arch in archs:
add_result = tool.run(
[
Expand All @@ -182,18 +203,21 @@ def main(argv: Optional[List[str]] = None) -> None:
stdout=PIPE,
)
LOG.info(f"Added: {add_result}")
inspect_result = tool.run(
["manifest", "inspect", manifest_name], text=True, stdout=PIPE
LOG.debug(
"Manifest after adding images: %s",
tool.run(
["manifest", "inspect", manifest_name], text=True, stdout=PIPE
).stdout,
)
LOG.debug("Manifest after adding images: %s", inspect_result)

# 4. Push the manifest (with images) to docker.io
tool.login(
retry_call(tool.login)(
config.docker["registry"],
config.docker["username"],
config.docker["password"],
)
push_result = tool.run(

push_result = retry_call(tool.run)(
[
"manifest",
"push",
Expand Down