Skip to content

Commit

Permalink
Add docker options for pull retry (#75)
Browse files Browse the repository at this point in the history
  • Loading branch information
BarDweller authored Jul 16, 2024
1 parent fb6f150 commit 8a44d72
Show file tree
Hide file tree
Showing 11 changed files with 186 additions and 70 deletions.
4 changes: 3 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,9 @@ The [`BuildpackConfig`](client/src/main/java/dev/snowdrop/buildpack/BuildConfig.

- run/build/output Image can be specified
- docker can be configured with..
- pull timeout
- pull timeout
- pull retry count (will retry image pull on failure)
- pull retry timeout increase (increases timeout each time pull is retried)
- host
- network
- docker socket path
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,7 @@ public BuilderImage(DockerConfig dc, PlatformConfig pc, ImageReference runImage,
image = builderImage;

// pull and inspect the builderImage to obtain builder metadata.
ImageUtils.pullImages(dc.getDockerClient(),
dc.getPullTimeout(),
builderImage.getReference());
ImageUtils.pullImages(dc, builderImage.getReference());

ImageInfo ii = ImageUtils.inspectImage(dc.getDockerClient(),
builderImage.getReference());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,17 @@ public static DockerConfigBuilder builder() {
return new DockerConfigBuilder();
}

public static enum PullPolicy {ALWAYS, IF_NOT_PRESENT};

private static final Integer DEFAULT_PULL_TIMEOUT = 60;
private static final Integer DEFAULT_PULL_RETRY_INCREASE = 15;
private static final Integer DEFAULT_PULL_RETRY_COUNT = 3;
private static final PullPolicy DEFAULT_PULL_POLICY = PullPolicy.IF_NOT_PRESENT;

private Integer pullTimeoutSeconds;
private Integer pullRetryCount;
private Integer pullRetryIncreaseSeconds;
private PullPolicy pullPolicy;
private String dockerHost;
private String dockerSocket;
private String dockerNetwork;
Expand All @@ -23,13 +31,19 @@ public static DockerConfigBuilder builder() {

public DockerConfig(
Integer pullTimeoutSeconds,
Integer pullRetryCount,
Integer pullRetryIncreaseSeconds,
PullPolicy pullPolicy,
String dockerHost,
String dockerSocket,
String dockerNetwork,
Boolean useDaemon,
DockerClient dockerClient
){
this.pullTimeoutSeconds = pullTimeoutSeconds != null ? pullTimeoutSeconds : DEFAULT_PULL_TIMEOUT;
this.pullTimeoutSeconds = pullTimeoutSeconds != null ? Integer.max(0,pullTimeoutSeconds) : DEFAULT_PULL_TIMEOUT;
this.pullRetryCount = pullRetryCount != null ? Integer.max(0,pullRetryCount) : DEFAULT_PULL_RETRY_COUNT;
this.pullRetryIncreaseSeconds = pullRetryIncreaseSeconds != null ? Integer.max(0,pullRetryIncreaseSeconds) : DEFAULT_PULL_RETRY_INCREASE;
this.pullPolicy = pullPolicy != null ? pullPolicy : DEFAULT_PULL_POLICY;
this.dockerHost = dockerHost != null ? dockerHost : DockerClientUtils.getDockerHost();
this.dockerSocket = dockerSocket != null ? dockerSocket : (this.dockerHost.startsWith("unix://") ? this.dockerHost.substring("unix://".length()) : "/var/run/docker.sock");
this.dockerNetwork = dockerNetwork;
Expand All @@ -47,6 +61,18 @@ public Integer getPullTimeout(){
return this.pullTimeoutSeconds;
}

public Integer getPullRetryCount(){
return this.pullRetryCount;
}

public Integer getPullRetryIncrease(){
return this.pullRetryIncreaseSeconds;
}

public PullPolicy getPullPolicy(){
return this.pullPolicy;
}

public String getDockerHost(){
return this.dockerHost;
}
Expand Down
100 changes: 74 additions & 26 deletions client/src/main/java/dev/snowdrop/buildpack/docker/ImageUtils.java
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,14 @@

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand All @@ -15,7 +18,10 @@
import com.github.dockerjava.api.command.InspectImageResponse;
import com.github.dockerjava.api.command.PullImageResultCallback;
import com.github.dockerjava.api.model.Image;

import com.github.dockerjava.api.exception.DockerClientException;
import com.github.dockerjava.api.exception.NotFoundException;

import dev.snowdrop.buildpack.config.DockerConfig;
import dev.snowdrop.buildpack.BuildpackException;
/**
* Higher level docker image api
Expand All @@ -30,51 +36,93 @@ public static class ImageInfo {
}

/**
* Util method to pull images if they don't exist to the local docker yet.
* Util method to pull images, configure behavior via dockerconfig.
*/
public static void pullImages(DockerClient dc, int timeoutSeconds, String... imageNames) {
@SuppressWarnings("resource")
public static void pullImages(DockerConfig config, String... imageNames) {
Set<String> imageNameSet = new HashSet<>(Arrays.asList(imageNames));

// list the current known images
List<Image> li = dc.listImagesCmd().exec();
for (Image i : li) {
if (i.getRepoTags() == null) {
continue;
}
for (String it : i.getRepoTags()) {
if (imageNameSet.contains(it)) {
imageNameSet.remove(it);
DockerClient dc = config.getDockerClient();

//if using ifnotpresent, filter set to unknown images.
if(config.getPullPolicy() == DockerConfig.PullPolicy.IF_NOT_PRESENT) {
// list the current known images
List<Image> li = dc.listImagesCmd().exec();
for (Image i : li) {
if (i.getRepoTags() == null) {
continue;
}
for (String it : i.getRepoTags()) {
if (imageNameSet.contains(it)) {
imageNameSet.remove(it);
}
}
}
}

if (imageNameSet.isEmpty()) {
// fast exit if all images are already known to the local docker.
log.debug("Nothing to pull, all of " + Arrays.asList(imageNames) + " are known");
return;
if (imageNameSet.isEmpty()) {
// fast exit if all images are already known to the local docker.
log.debug("Nothing to pull, all of " + Arrays.asList(imageNames) + " are known");
return;
}
}

// pull the images not known
List<PullImageResultCallback> pircs = new ArrayList<>();
int retryCount = 0;
Map<String,PullImageResultCallback> pircMap = new HashMap<>();

// pull the images still in set.
for (String stillNeeded : imageNameSet) {
log.debug("pulling '" + stillNeeded + "'");
PullImageResultCallback pirc = new PullImageResultCallback();
dc.pullImageCmd(stillNeeded).exec(pirc);
pircs.add(pirc);
pircMap.put(stillNeeded,pirc);
}

// wait for pulls to complete.
for (PullImageResultCallback pirc : pircs) {
try {
pirc.awaitCompletion(timeoutSeconds, TimeUnit.SECONDS);
} catch (InterruptedException e) {
throw BuildpackException.launderThrowable(e);
RuntimeException lastSeen = null;
boolean allDone = false;
while(!allDone && retryCount<=config.getPullRetryCount()){
allDone = true;
long thisWait = config.getPullTimeout()+(retryCount*config.getPullRetryIncrease());
for (Entry<String, PullImageResultCallback> e : pircMap.entrySet()) {
boolean done = false;
try {
if(e.getValue()==null) continue;
log.debug("waiting on image "+e.getKey()+" for "+thisWait+" seconds");
done = e.getValue().awaitCompletion( thisWait, TimeUnit.SECONDS);
log.debug("success for image "+e.getKey());
} catch (InterruptedException ie) {
throw BuildpackException.launderThrowable(ie);
} catch (DockerClientException dce) {
//error occurred during pull for this pirc, need to pause & retry the pull op
lastSeen = dce;
} catch (NotFoundException nfe) {
lastSeen = nfe;
}
if(!done){
String imageName = e.getKey();
PullImageResultCallback newPirc = new PullImageResultCallback();
dc.pullImageCmd(imageName).exec(newPirc);
e.setValue(newPirc);
allDone=false;
}else{
e.setValue(null);
}
}
retryCount++;
if(retryCount<=config.getPullRetryCount()){
if(lastSeen!=null){
log.debug("Error during pull "+lastSeen.getMessage());
}
log.debug("Retrying ("+retryCount+") for "+pircMap.entrySet().stream().filter(e -> e.getValue()!=null).collect(Collectors.toList()));
}
}

// TODO: progress tracking..
if(lastSeen!=null && !allDone){
throw lastSeen;
}
}


/**
* Util method to retrieve info for a given docker image.
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -105,7 +105,7 @@ public int execute() {
}

//pull the new image..
ImageUtils.pullImages(config.getDockerConfig().getDockerClient(), factory.getDockerConfig().getPullTimeout(), newRunImage);
ImageUtils.pullImages(config.getDockerConfig(), newRunImage);

//update run image associated with our builder image.
factory.getBuilderImage().getRunImages(activePlatformLevel).clear();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,9 +20,7 @@ public class LifecycleMetadata {
public LifecycleMetadata(DockerConfig dc, ImageReference lifecycleImage) throws BuildpackException {

// pull and inspect the builderImage to obtain builder metadata.
ImageUtils.pullImages(dc.getDockerClient(),
dc.getPullTimeout(),
lifecycleImage.getReference());
ImageUtils.pullImages(dc,lifecycleImage.getReference());

ImageInfo ii = ImageUtils.inspectImage(dc.getDockerClient(),
lifecycleImage.getReference());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.mockito.Mockito.lenient;
import static org.mockito.Mockito.when;

import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
Expand All @@ -20,20 +19,21 @@
public class DockerConfigTest {
@Test
void checkTimeout() {
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null);
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertEquals(60, dc1.getPullTimeout());

DockerConfig dc2 = new DockerConfig(245017, null, null, null, null, null);
DockerConfig dc2 = new DockerConfig(245017, null, null, null, null, null, null, null, null);
assertEquals(dc2.getPullTimeout(), 245017);
}

@Test
void checkDockerHost(@Mock DockerClient dockerClient, @Mock PingCmd pingCmd) {
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null);
lenient().when(dockerClient.pingCmd()).thenReturn(pingCmd);

DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertNotNull(dc1.getDockerHost());

when(dockerClient.pingCmd()).thenReturn(pingCmd);
DockerConfig dc2 = new DockerConfig(null, "tcp://stilettos", null, null, null, dockerClient);
DockerConfig dc2 = new DockerConfig(null, null, null, null, "tcp://stilettos", null, null, null, dockerClient);
assertEquals("tcp://stilettos", dc2.getDockerHost());
}

Expand All @@ -42,48 +42,77 @@ void checkDockerSocket(@Mock DockerClient dockerClient, @Mock PingCmd pingCmd) {

lenient().when(dockerClient.pingCmd()).thenReturn(pingCmd);

DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null);
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertNotNull(dc1.getDockerSocket());

DockerConfig dc2 = new DockerConfig(null, "unix:///stilettos", null, null, null, dockerClient);
DockerConfig dc2 = new DockerConfig(null, null, null, null, "unix:///stilettos", null, null, null, dockerClient);
assertEquals("/stilettos", dc2.getDockerSocket());

DockerConfig dc3 = new DockerConfig(null, "tcp://stilettos", null, null, null, dockerClient);
DockerConfig dc3 = new DockerConfig(null, null, null, null, "tcp://stilettos", null, null, null, dockerClient);
assertEquals("/var/run/docker.sock", dc3.getDockerSocket());

DockerConfig dc4 = new DockerConfig(null, null, "fish", null, null, null);
DockerConfig dc4 = new DockerConfig(null, null, null, null, null, "fish", null, null, null);
assertEquals("fish", dc4.getDockerSocket());
}

@Test
void checkDockerNetwork() {
DockerConfig dc1 = new DockerConfig(null, null, null, "kitten", null, null);
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, "kitten", null, null);
assertEquals("kitten", dc1.getDockerNetwork());

DockerConfig dc2 = new DockerConfig(null, null, null, null, null, null);
DockerConfig dc2 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertNull(dc2.getDockerNetwork());
}

@Test
void checkUseDaemon() {
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null);
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertTrue(dc1.getUseDaemon());

DockerConfig dc2 = new DockerConfig(null, null, null, null, true, null);
DockerConfig dc2 = new DockerConfig(null, null, null, null, null, null, null, true, null);
assertTrue(dc2.getUseDaemon());

DockerConfig dc3 = new DockerConfig(null, null, null, null, false, null);
DockerConfig dc3 = new DockerConfig(null, null, null, null, null, null, null, false, null);
assertFalse(dc3.getUseDaemon());
}

@Test
void checkDockerClient(@Mock DockerClient dockerClient, @Mock PingCmd pingCmd){
lenient().when(dockerClient.pingCmd()).thenReturn(pingCmd);

DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null);
DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, null);
assertNotNull(dc1.getDockerClient());

DockerConfig dc2 = new DockerConfig(null, null, null, null, null, dockerClient);
DockerConfig dc2 = new DockerConfig(null, null, null, null, null, null, null, null, dockerClient);
assertEquals(dockerClient, dc2.getDockerClient());
}

@Test
void checkPullPolicy(@Mock DockerClient dockerClient, @Mock PingCmd pingCmd){
lenient().when(dockerClient.pingCmd()).thenReturn(pingCmd);

DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, dockerClient);
assertEquals(DockerConfig.PullPolicy.IF_NOT_PRESENT, dc1.getPullPolicy());

DockerConfig dc2 = new DockerConfig(null, null, null, DockerConfig.PullPolicy.IF_NOT_PRESENT, null, null, null, null, dockerClient);
assertEquals(DockerConfig.PullPolicy.IF_NOT_PRESENT, dc2.getPullPolicy());

DockerConfig dc3 = new DockerConfig(null, null, null, DockerConfig.PullPolicy.ALWAYS, null, null, null, null, dockerClient);
assertEquals(DockerConfig.PullPolicy.ALWAYS, dc3.getPullPolicy());
}


@Test
void checkPullRetry(@Mock DockerClient dockerClient, @Mock PingCmd pingCmd){
lenient().when(dockerClient.pingCmd()).thenReturn(pingCmd);

DockerConfig dc1 = new DockerConfig(null, null, null, null, null, null, null, null, dockerClient);
assertEquals(3, dc1.getPullRetryCount());

DockerConfig dc2 = new DockerConfig(null, 5, null, null, null, null, null, null, dockerClient);
assertEquals(5, dc2.getPullRetryCount());

DockerConfig dc3 = new DockerConfig(null, 0, null, null, null, null, null, null, dockerClient);
assertEquals(0, dc3.getPullRetryCount());
}
}
Loading

0 comments on commit 8a44d72

Please sign in to comment.