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

Skip DFIU PRs for repos onboarded to Renovate Enterprise #1128

Merged
merged 11 commits into from
Feb 8, 2024
Merged
Show file tree
Hide file tree
Changes from 6 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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
*.ipr
*.iws
*.iml
*.DS_Store

# Vim files
*.swp
Expand Down
5 changes: 5 additions & 0 deletions dockerfile-image-update/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,11 @@


<dependencies>
<dependency>
<groupId>org.json</groupId>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

is there something we can't go with gson dependency that we already have https://github.com/avimanyum/dockerfile-image-update/blob/master/dockerfile-image-update/pom.xml#L116. or you are brining it in for easy of use?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah its just a cleaner and easier way to convert to Json object

<artifactId>json</artifactId>
<version>20240205</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-simple</artifactId>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,10 @@ static ArgumentParser getArgumentParser() {
ArgumentParsers.newFor("dockerfile-image-update").addHelp(true).build()
.description("Image Updates through Pull Request Automator");

parser.addArgument("-R", "--" + CHECK_FOR_RENOVATE)
.type(Boolean.class)
.setDefault(false) //To prevent null from being returned by the argument
.help("Check if Renovate app is being used to receive remediation PRs.");
parser.addArgument("-l", "--" + GIT_API_SEARCH_LIMIT)
.type(Integer.class)
.setDefault(1000)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@


import java.time.Duration;
import java.util.*;

/**
* @author minho-park
Expand Down Expand Up @@ -51,4 +52,7 @@ private Constants() {
public static final Duration DEFAULT_TOKEN_ADDING_RATE = Duration.ofMinutes(DEFAULT_CONSUMING_TOKEN_RATE);
public static final String FILENAME_DOCKERFILE = "dockerfile";
public static final String FILENAME_DOCKER_COMPOSE = "docker-compose";
public static final String CHECK_FOR_RENOVATE = "checkforrenovate";
//The Renovate configuration file can be in any one of the following locations. Refer to https://docs.renovatebot.com/configuration-options/
public static final List<String> RENOVATE_CONFIG_FILEPATHS = Arrays.asList("renovate.json", "renovate.json5", ".github/renovate.json", ".github/renovate.json5", ".gitlab/renovate.json", ".gitlab/renovate.json5", ".renovaterc", ".renovaterc.json", ".renovaterc.json5");
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
import com.salesforce.dockerfileimageupdate.model.*;
import com.salesforce.dockerfileimageupdate.process.*;
import net.sourceforge.argparse4j.inf.*;
import org.json.JSONObject;
import org.json.JSONTokener;
import org.kohsuke.github.*;
import org.slf4j.*;

Expand All @@ -27,10 +29,16 @@ public void prepareToCreate(final Namespace ns,
pathToDockerfilesInParentRepo.get(currUserRepo).stream().findFirst();
if (forkWithContentPaths.isPresent()) {
try {
dockerfileGitHubUtil.changeDockerfiles(ns,
pathToDockerfilesInParentRepo,
forkWithContentPaths.get(), skippedRepos,
gitForkBranch, rateLimiter);
//If the repository has been onboarded to renovate enterprise, skip sending the DFIU PR
if(ns.getBoolean(Constants.CHECK_FOR_RENOVATE)
&& (isRenovateEnabled(Constants.RENOVATE_CONFIG_FILEPATHS, forkWithContentPaths.get()))) {
log.info("Found a renovate configuration file in the repo %s. Skip sending DFIU PRs to this repository.", forkWithContentPaths.get().getParent().getFullName());
} else {
dockerfileGitHubUtil.changeDockerfiles(ns,
pathToDockerfilesInParentRepo,
forkWithContentPaths.get(), skippedRepos,
gitForkBranch, rateLimiter);
}
} catch (IOException | InterruptedException e) {
log.error(String.format("Error changing Dockerfile for %s", forkWithContentPaths.get().getParent().getFullName()), e);
exceptions.add((IOException) e);
Expand All @@ -41,5 +49,32 @@ public void prepareToCreate(final Namespace ns,
}
ResultsProcessor.processResults(skippedRepos, exceptions, log);
}

/**
* Check if the repository is onboarded to Renovate. The signal we are looking for are
* (1) The presence of a file where renovate configurations are stored in the repository
* (2) Ensuring that the file does not have the key "enabled" set to "false"
* @param filePaths the list that contains all the names of the files that we are searching for in the repo
* @param fork A GitHubContentToProcess object that contains the fork repository that is under process
* @return true if the file is found in the path specified and is not disabled, false otherwise
*/
private boolean isRenovateEnabled(List<String> filePaths, GitHubContentToProcess fork) throws IOException {
for (String filePath : filePaths) {
try {
GHContent fileContent = fork.getParent().getFileContent(filePath);
InputStream is = fileContent.read();
BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(is));
JSONTokener tokener = new JSONTokener(bufferedReader);
JSONObject json = new JSONObject(tokener);
is.close();
bufferedReader.close();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You might want to do this in a try-with-resources statement, see https://stackoverflow.com/a/56151320 . If is.close throws an exception then the close on buffered reader won't happen.

//If the file has the key 'enabled' set to false, it indicates that while the repo has been onboarded to renovate, it has been disabled for some reason
return json.optBoolean("enabled", true);
} catch (FileNotFoundException e) {
log.debug("The file with name %s not found in the repository.", filePath);
}
}
return false;
}
}

Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ public void testPullRequestsPrepareToCreateSuccessful() throws Exception {
"image", Constants.TAG,
"tag", Constants.STORE,
"store", Constants.SKIP_PR_CREATION,
false);
false, Constants.CHECK_FOR_RENOVATE, false);
Namespace ns = new Namespace(nsMap);
PullRequests pullRequests = new PullRequests();
GitHubPullRequestSender pullRequestSender = mock(GitHubPullRequestSender.class);
Expand Down Expand Up @@ -51,7 +51,7 @@ public void testPullRequestsPrepareThrowsException() throws Exception {
"image", Constants.TAG,
"tag", Constants.STORE,
"store", Constants.SKIP_PR_CREATION,
false);
false, Constants.CHECK_FOR_RENOVATE, false);
Namespace ns = new Namespace(nsMap);
PullRequests pullRequests = new PullRequests();
GitHubPullRequestSender pullRequestSender = mock(GitHubPullRequestSender.class);
Expand Down Expand Up @@ -89,7 +89,7 @@ public void testPullRequestsPrepareToCreateWhenNoDockerfileFound() throws Except
"image", Constants.TAG,
"tag", Constants.STORE,
"store", Constants.SKIP_PR_CREATION,
false);
false, Constants.CHECK_FOR_RENOVATE, false);
Namespace ns = new Namespace(nsMap);
PullRequests pullRequests = new PullRequests();
GitHubPullRequestSender pullRequestSender = mock(GitHubPullRequestSender.class);
Expand All @@ -110,4 +110,51 @@ public void testPullRequestsPrepareToCreateWhenNoDockerfileFound() throws Except
eq(pathToDockerfilesInParentRepo),
eq(gitHubContentToProcess), anyList(), eq(gitForkBranch),eq(rateLimiter));
}

@Test
public void testPullRequestsPrepareSkipsSendingPRIfRepoOnboardedToRenovate() throws Exception {
Map<String, Object> nsMap = ImmutableMap.of(
Constants.IMG, "image",
Constants.TAG, "tag",
Constants.STORE,"store",
Constants.SKIP_PR_CREATION,false,
Constants.CHECK_FOR_RENOVATE, true);


Namespace ns = new Namespace(nsMap);
PullRequests pullRequests = new PullRequests();
GitHubPullRequestSender pullRequestSender = mock(GitHubPullRequestSender.class);
PagedSearchIterable<GHContent> contentsFoundWithImage = mock(PagedSearchIterable.class);
GitForkBranch gitForkBranch = mock(GitForkBranch.class);
DockerfileGitHubUtil dockerfileGitHubUtil = mock(DockerfileGitHubUtil.class);
RateLimiter rateLimiter = Mockito.spy(new RateLimiter());
Multimap<String, GitHubContentToProcess> pathToDockerfilesInParentRepo = ArrayListMultimap.create();
GitHubContentToProcess gitHubContentToProcess = mock(GitHubContentToProcess.class);
pathToDockerfilesInParentRepo.put("repo1", gitHubContentToProcess);
pathToDockerfilesInParentRepo.put("repo2", gitHubContentToProcess);
pathToDockerfilesInParentRepo.put("repo3", gitHubContentToProcess);
GHContent content = mock(GHContent.class);
InputStream inputStream1 = new ByteArrayInputStream("{someKey:someValue}".getBytes());
InputStream inputStream2 = new ByteArrayInputStream("{enabled:false}".getBytes());
GHRepository ghRepository = mock(GHRepository.class);

when(pullRequestSender.forkRepositoriesFoundAndGetPathToDockerfiles(contentsFoundWithImage, gitForkBranch)).thenReturn(pathToDockerfilesInParentRepo);
when(gitHubContentToProcess.getParent()).thenReturn(ghRepository);
//Fetch the content of the renovate.json file for the 3 repos.
// The first one returns a file with regular json content.
// The second one returns a file with the key 'enabled' set to 'false' to replicate a repo that has been onboarded to renovate but has it disabled
// The third repo does not have the renovate.json file
when(ghRepository.getFileContent(anyString())).thenReturn(content).thenReturn(content).thenThrow(new FileNotFoundException());
when(ghRepository.getFullName()).thenReturn("org/repo");
when(content.read()).thenReturn(inputStream1).thenReturn(inputStream2);

pullRequests.prepareToCreate(ns, pullRequestSender, contentsFoundWithImage,
gitForkBranch, dockerfileGitHubUtil, rateLimiter);

//Verify that the DFIU PR is skipped for the first repo, but is sent to the other two repos
verify(dockerfileGitHubUtil, times(2)).changeDockerfiles(eq(ns),
eq(pathToDockerfilesInParentRepo),
eq(gitHubContentToProcess), anyList(), eq(gitForkBranch),
eq(rateLimiter));
}
}
Loading