Skip to content

Commit

Permalink
Merge pull request jenkinsci#55 from jglick/SCMSourceRetriever.clone
Browse files Browse the repository at this point in the history
[JENKINS-70869] New option “clone” option to SCM-based retrievers to avoid `workspace/xxx@libs/`
  • Loading branch information
jglick authored Apr 11, 2023
2 parents 06b0eff + 6d4bd49 commit aaceeb6
Show file tree
Hide file tree
Showing 13 changed files with 526 additions and 266 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,328 @@
/*
* The MIT License
*
* Copyright 2023 CloudBees, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/

package org.jenkinsci.plugins.workflow.libs;

import edu.umd.cs.findbugs.annotations.CheckForNull;
import edu.umd.cs.findbugs.annotations.NonNull;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import hudson.AbortException;
import hudson.Extension;
import hudson.FilePath;
import hudson.Functions;
import hudson.Util;
import hudson.model.Computer;
import hudson.model.Item;
import hudson.model.Job;
import hudson.model.Node;
import hudson.model.Run;
import hudson.model.TaskListener;
import hudson.model.TopLevelItem;
import hudson.model.listeners.ItemListener;
import hudson.scm.SCM;
import hudson.slaves.WorkspaceList;
import hudson.util.FormValidation;
import java.io.File;
import java.io.IOException;
import java.io.InterruptedIOException;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.regex.Pattern;
import jenkins.model.Jenkins;
import org.jenkinsci.plugins.workflow.steps.scm.GenericSCMStep;
import org.jenkinsci.plugins.workflow.steps.scm.SCMStep;
import org.kohsuke.accmod.Restricted;
import org.kohsuke.accmod.restrictions.DoNotUse;
import org.kohsuke.stapler.DataBoundSetter;
import org.kohsuke.stapler.QueryParameter;
import org.kohsuke.stapler.verb.POST;

/**
* Functionality common to {@link SCMSourceRetriever} and {@link SCMRetriever}.
*/
public abstract class SCMBasedRetriever extends LibraryRetriever {

private static final Logger LOGGER = Logger.getLogger(SCMBasedRetriever.class.getName());

@SuppressFBWarnings(value = "MS_SHOULD_BE_FINAL", justification = "Non-final for write access via the Script Console")
public static boolean INCLUDE_SRC_TEST_IN_LIBRARIES = Boolean.getBoolean(SCMSourceRetriever.class.getName() + ".INCLUDE_SRC_TEST_IN_LIBRARIES");

/**
* Matches ".." in positions where it would be treated as the parent directory.
*
* <p>Used to prevent {@link #libraryPath} from being used for directory traversal.
*/
static final Pattern PROHIBITED_DOUBLE_DOT = Pattern.compile("(^|.*[\\\\/])\\.\\.($|[\\\\/].*)");

private boolean clone;

/**
* The path to the library inside of the SCM.
*
* {@code null} is the default and means that the library is in the root of the repository. Otherwise, the value is
* considered to be a relative path inside of the repository and always ends in a forward slash
*
* @see #setLibraryPath
*/
private @CheckForNull String libraryPath;

public boolean isClone() {
return clone;
}

@DataBoundSetter public void setClone(boolean clone) {
this.clone = clone;
}

public String getLibraryPath() {
return libraryPath;
}

@DataBoundSetter public void setLibraryPath(String libraryPath) {
libraryPath = Util.fixEmptyAndTrim(libraryPath);
if (libraryPath != null && !libraryPath.endsWith("/")) {
libraryPath += '/';
}
this.libraryPath = libraryPath;
}

protected final void doRetrieve(String name, boolean changelog, @NonNull SCM scm, FilePath target, Run<?, ?> run, TaskListener listener) throws Exception {
if (libraryPath != null && PROHIBITED_DOUBLE_DOT.matcher(libraryPath).matches()) {
throw new AbortException("Library path may not contain '..'");
}
if (clone && changelog) {
listener.getLogger().println("WARNING: ignoring request to compute changelog in clone mode");
changelog = false;
}
// Adapted from CpsScmFlowDefinition:
SCMStep delegate = new GenericSCMStep(scm);
delegate.setPoll(false); // TODO we have no API for determining if a given SCMHead is branch-like or tag-like; would we want to turn on polling if the former?
delegate.setChangelog(changelog);
Node node = Jenkins.get();
if (clone) {
if (libraryPath == null) {
retrySCMOperation(listener, () -> {
delegate.checkout(run, target, listener, Jenkins.get().createLauncher(listener));
WorkspaceList.tempDir(target).deleteRecursive();
return null;
});
} else {
FilePath root = target.child("root");
retrySCMOperation(listener, () -> {
delegate.checkout(run, root, listener, Jenkins.get().createLauncher(listener));
WorkspaceList.tempDir(root).deleteRecursive();
return null;
});
FilePath subdir = root.child(libraryPath);
if (!subdir.isDirectory()) {
throw new AbortException("Did not find " + libraryPath + " in checkout");
}
for (String content : List.of("src", "vars", "resources")) {
FilePath contentDir = subdir.child(content);
if (contentDir.isDirectory()) {
LOGGER.fine(() -> "Moving " + content + " to top level in " + target);
contentDir.renameTo(target.child(content));
}
}
// root itself will be deleted below
}
if (!INCLUDE_SRC_TEST_IN_LIBRARIES) {
FilePath srcTest = target.child("src/test");
if (srcTest.isDirectory()) {
listener.getLogger().println("Excluding src/test/ from checkout of " + scm.getKey() + " so that library test code cannot be accessed by Pipelines.");
listener.getLogger().println("To remove this log message, move the test code outside of src/. To restore the previous behavior that allowed access to files in src/test/, pass -D" + SCMSourceRetriever.class.getName() + ".INCLUDE_SRC_TEST_IN_LIBRARIES=true to the java command used to start Jenkins.");
srcTest.deleteRecursive();
}
}
for (FilePath child : target.list()) {
String subdir = child.getName();
switch (subdir) {
case "src":
// TODO delete everything that is not *.groovy
break;
case "vars":
// TODO delete everything that is not *.groovy or *.txt, incl. subdirs
break;
case "resources":
// OK, leave it all
break;
default:
child.deleteRecursive();
LOGGER.fine(() -> "Deleted " + child);
}
}
} else { // !clone
FilePath dir;
if (run.getParent() instanceof TopLevelItem) {
FilePath baseWorkspace = node.getWorkspaceFor((TopLevelItem) run.getParent());
if (baseWorkspace == null) {
throw new IOException(node.getDisplayName() + " may be offline");
}
String checkoutDirName = LibraryRecord.directoryNameFor(scm.getKey());
dir = baseWorkspace.withSuffix(getFilePathSuffix() + "libs").child(checkoutDirName);
} else { // should not happen, but just in case:
throw new AbortException("Cannot check out in non-top-level build");
}
Computer computer = node.toComputer();
if (computer == null) {
throw new IOException(node.getDisplayName() + " may be offline");
}
try (WorkspaceList.Lease lease = computer.getWorkspaceList().allocate(dir)) {
// Write the SCM key to a file as a debugging aid.
lease.path.withSuffix("-scm-key.txt").write(scm.getKey(), "UTF-8");
retrySCMOperation(listener, () -> {
delegate.checkout(run, lease.path, listener, node.createLauncher(listener));
return null;
});
if (libraryPath == null) {
libraryPath = ".";
}
String excludes = INCLUDE_SRC_TEST_IN_LIBRARIES ? null : "src/test/";
if (lease.path.child(libraryPath).child("src/test").exists()) {
listener.getLogger().println("Excluding src/test/ from checkout of " + scm.getKey() + " so that library test code cannot be accessed by Pipelines.");
listener.getLogger().println("To remove this log message, move the test code outside of src/. To restore the previous behavior that allowed access to files in src/test/, pass -D" + SCMSourceRetriever.class.getName() + ".INCLUDE_SRC_TEST_IN_LIBRARIES=true to the java command used to start Jenkins.");
}
// Cannot add WorkspaceActionImpl to private CpsFlowExecution.flowStartNodeActions; do we care?
// Copy sources with relevant files from the checkout:
lease.path.child(libraryPath).copyRecursiveTo("src/**/*.groovy,vars/*.groovy,vars/*.txt,resources/", excludes, target);
}
}
}

protected static <T> T retrySCMOperation(TaskListener listener, Callable<T> task) throws Exception{
T ret = null;
for (int retryCount = Jenkins.get().getScmCheckoutRetryCount(); retryCount >= 0; retryCount--) {
try {
ret = task.call();
break;
}
catch (AbortException e) {
// abort exception might have a null message.
// If so, just skip echoing it.
if (e.getMessage() != null) {
listener.error(e.getMessage());
}
}
catch (InterruptedIOException e) {
throw e;
}
catch (Exception e) {
// checkout error not yet reported
Functions.printStackTrace(e, listener.error("Checkout failed"));
}

if (retryCount == 0) // all attempts failed
throw new AbortException("Maximum checkout retry attempts reached, aborting");

listener.getLogger().println("Retrying after 10 seconds");
Thread.sleep(10000);
}
return ret;
}

// TODO there is WorkspaceList.tempDir but no API to make other variants
private static String getFilePathSuffix() {
return System.getProperty(WorkspaceList.class.getName(), "@");
}

protected abstract static class SCMBasedRetrieverDescriptor extends LibraryRetrieverDescriptor {

@POST
public FormValidation doCheckLibraryPath(@QueryParameter String libraryPath) {
libraryPath = Util.fixEmptyAndTrim(libraryPath);
if (libraryPath == null) {
return FormValidation.ok();
} else if (PROHIBITED_DOUBLE_DOT.matcher(libraryPath).matches()) {
return FormValidation.error(Messages.SCMSourceRetriever_library_path_no_double_dot());
}
return FormValidation.ok();
}

}

@Restricted(DoNotUse.class)
@Extension
public static class WorkspaceListener extends ItemListener {

@Override
public void onDeleted(Item item) {
deleteLibsDir(item, item.getFullName());
}

@Override
public void onLocationChanged(Item item, String oldFullName, String newFullName) {
deleteLibsDir(item, oldFullName);
}

private static void deleteLibsDir(Item item, String itemFullName) {
if (item instanceof Job
&& item.getClass()
.getName()
.equals("org.jenkinsci.plugins.workflow.job.WorkflowJob")) {
synchronized (item) {
String base =
expandVariablesForDirectory(
Jenkins.get().getRawWorkspaceDir(),
itemFullName,
item.getRootDir().getPath());
FilePath dir =
new FilePath(new File(base)).withSuffix(getFilePathSuffix() + "libs");
try {
if (dir.isDirectory()) {
LOGGER.log(
Level.INFO,
() -> "Deleting obsolete library workspace " + dir);
dir.deleteRecursive();
}
} catch (IOException | InterruptedException e) {
LOGGER.log(
Level.WARNING,
e,
() -> "Could not delete obsolete library workspace " + dir);
}
}
}
}

private static String expandVariablesForDirectory(
String base, String itemFullName, String itemRootDir) {
// If the item is moved, it is too late to look up its original workspace location by
// the time we get the notification. See:
// https://github.com/jenkinsci/jenkins/blob/f03183ab09ce5fb8f9f4cc9ccee42a3c3e6b2d3e/core/src/main/java/jenkins/model/Jenkins.java#L2567-L2576
Map<String, String> properties = new HashMap<>();
properties.put("JENKINS_HOME", Jenkins.get().getRootDir().getPath());
properties.put("ITEM_ROOTDIR", itemRootDir);
properties.put("ITEM_FULLNAME", itemFullName); // legacy, deprecated
properties.put(
"ITEM_FULL_NAME", itemFullName.replace(':', '$')); // safe, see JENKINS-12251
return Util.replaceMacro(base, Collections.unmodifiableMap(properties));
}
}

}
Loading

0 comments on commit aaceeb6

Please sign in to comment.