/* * The MIT License * * Copyright (c) 2004-2011, Sun Microsystems, Inc., Kohsuke Kawaguchi, * Erik Ramfelt, Martin Eigenbrodt, Stephen Connolly, Tom Huybrechts * * 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 hudson.model; import com.google.common.collect.ImmutableSet; import hudson.DescriptorExtensionList; import hudson.FilePath; import hudson.Launcher; import hudson.Launcher.RemoteLauncher; import hudson.Util; import hudson.model.Descriptor.FormException; import hudson.remoting.Callable; import hudson.slaves.CommandLauncher; import hudson.slaves.ComputerLauncher; import hudson.slaves.DumbSlave; import hudson.slaves.JNLPLauncher; import hudson.slaves.NodeDescriptor; import hudson.slaves.NodeProperty; import hudson.slaves.NodePropertyDescriptor; import hudson.slaves.RetentionStrategy; import hudson.slaves.SlaveComputer; import hudson.util.ClockDifference; import hudson.util.DescribableList; import hudson.util.FormValidation; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.Serializable; import java.net.MalformedURLException; import java.net.URL; import java.net.URLConnection; import java.util.ArrayList; import java.util.Collection; import java.util.List; import java.util.Set; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; import javax.servlet.ServletException; import jenkins.model.Jenkins; import jenkins.security.MasterToSlaveCallable; import jenkins.slaves.WorkspaceLocator; import jenkins.util.SystemProperties; import org.apache.commons.io.IOUtils; import org.apache.commons.lang.StringUtils; import org.kohsuke.accmod.Restricted; import org.kohsuke.accmod.restrictions.NoExternalUse; import org.kohsuke.stapler.DataBoundSetter; import org.kohsuke.stapler.HttpResponse; import org.kohsuke.stapler.QueryParameter; import org.kohsuke.stapler.StaplerRequest; import org.kohsuke.stapler.StaplerResponse; /** * Information about a Hudson agent node. * * <p> * Ideally this would have been in the <tt>hudson.slaves</tt> package, * but for compatibility reasons, it can't. * * <p> * TODO: move out more stuff to {@link DumbSlave}. * * On Febrary, 2016 a general renaming was done internally: the "slave" term was replaced by * "Agent". This change was applied in: UI labels/HTML pages, javadocs and log messages. * Java classes, fields, methods, etc were not renamed to avoid compatibility issues. * See <a href="https://issues.jenkins-ci.org/browse/JENKINS-27268">JENKINS-27268</a>. * * @author Kohsuke Kawaguchi */ public abstract class Slave extends Node implements Serializable { /** * Name of this agent node. */ protected String name; /** * Description of this node. */ private String description; /** * Path to the root of the workspace from the view point of this node, such as "/hudson", this need not * be absolute provided that the launcher establishes a consistent working directory, such as "./.jenkins-slave" * when used with an SSH launcher. * * NOTE: if the administrator is using a relative path they are responsible for ensuring that the launcher used * provides a consistent working directory */ protected final String remoteFS; /** * Number of executors of this node. */ private int numExecutors = 2; /** * Job allocation strategy. */ private Mode mode = Mode.NORMAL; /** * Agent availablility strategy. */ private RetentionStrategy retentionStrategy; /** * The starter that will startup this agent. */ private ComputerLauncher launcher; /** * Whitespace-separated labels. */ private String label=""; private /*almost final*/ DescribableList<NodeProperty<?>,NodePropertyDescriptor> nodeProperties = new DescribableList<NodeProperty<?>,NodePropertyDescriptor>(Jenkins.getInstance().getNodesObject()); /** * Lazily computed set of labels from {@link #label}. */ private transient volatile Set<Label> labels; /** * Id of user which creates this agent {@link User}. */ private String userId; public Slave(String name, String nodeDescription, String remoteFS, String numExecutors, Mode mode, String labelString, ComputerLauncher launcher, RetentionStrategy retentionStrategy, List<? extends NodeProperty<?>> nodeProperties) throws FormException, IOException { this(name,nodeDescription,remoteFS,Util.tryParseNumber(numExecutors, 1).intValue(),mode,labelString,launcher,retentionStrategy, nodeProperties); } /** * @deprecated since 2009-02-20. */ @Deprecated public Slave(String name, String nodeDescription, String remoteFS, int numExecutors, Mode mode, String labelString, ComputerLauncher launcher, RetentionStrategy retentionStrategy) throws FormException, IOException { this(name, nodeDescription, remoteFS, numExecutors, mode, labelString, launcher, retentionStrategy, new ArrayList()); } public Slave(@Nonnull String name, String remoteFS, ComputerLauncher launcher) throws FormException, IOException { this.name = name; this.remoteFS = remoteFS; this.launcher = launcher; } /** * @deprecated as of 1.XXX * Use {@link #Slave(String, String, ComputerLauncher)} and set the rest through setters. */ public Slave(@Nonnull String name, String nodeDescription, String remoteFS, int numExecutors, Mode mode, String labelString, ComputerLauncher launcher, RetentionStrategy retentionStrategy, List<? extends NodeProperty<?>> nodeProperties) throws FormException, IOException { this.name = name; this.description = nodeDescription; this.numExecutors = numExecutors; this.mode = mode; this.remoteFS = Util.fixNull(remoteFS).trim(); this.label = Util.fixNull(labelString).trim(); this.launcher = launcher; this.retentionStrategy = retentionStrategy; getAssignedLabels(); // compute labels now this.nodeProperties.replaceBy(nodeProperties); Slave node = (Slave) Jenkins.getInstance().getNode(name); if(node!=null){ this.userId= node.getUserId(); //agent has already existed } else{ User user = User.current(); userId = user!=null ? user.getId() : "anonymous"; } if (name.equals("")) throw new FormException(Messages.Slave_InvalidConfig_NoName(), null); // if (remoteFS.equals("")) // throw new FormException(Messages.Slave_InvalidConfig_NoRemoteDir(name), null); if (this.numExecutors<=0) throw new FormException(Messages.Slave_InvalidConfig_Executors(name), null); } /** * Return id of user which created this agent * * @return id of user */ public String getUserId() { return userId; } public void setUserId(String userId){ this.userId = userId; } public ComputerLauncher getLauncher() { return launcher == null ? new JNLPLauncher() : launcher; } public void setLauncher(ComputerLauncher launcher) { this.launcher = launcher; } public String getRemoteFS() { return remoteFS; } public String getNodeName() { return name; } @Override public String toString() { return getClass().getName() + "[" + name + "]"; } public void setNodeName(String name) { this.name = name; } @DataBoundSetter public void setNodeDescription(String value) { this.description = value; } public String getNodeDescription() { return description; } public int getNumExecutors() { return numExecutors; } @DataBoundSetter public void setNumExecutors(int n) { this.numExecutors = n; } public Mode getMode() { return mode; } @DataBoundSetter public void setMode(Mode mode) { this.mode = mode; } public DescribableList<NodeProperty<?>, NodePropertyDescriptor> getNodeProperties() { assert nodeProperties != null; return nodeProperties; } @DataBoundSetter public void setNodeProperties(List<? extends NodeProperty<?>> properties) throws IOException { nodeProperties.replaceBy(properties); } public RetentionStrategy getRetentionStrategy() { return retentionStrategy == null ? RetentionStrategy.Always.INSTANCE : retentionStrategy; } @DataBoundSetter public void setRetentionStrategy(RetentionStrategy availabilityStrategy) { this.retentionStrategy = availabilityStrategy; } public String getLabelString() { return Util.fixNull(label).trim(); } @Override @DataBoundSetter public void setLabelString(String labelString) throws IOException { this.label = Util.fixNull(labelString).trim(); // Compute labels now. getAssignedLabels(); } @Override public Callable<ClockDifference,IOException> getClockDifferenceCallable() { return new GetClockDifference1(); } public Computer createComputer() { return new SlaveComputer(this); } public FilePath getWorkspaceFor(TopLevelItem item) { for (WorkspaceLocator l : WorkspaceLocator.all()) { FilePath workspace = l.locate(item, this); if (workspace != null) { return workspace; } } FilePath r = getWorkspaceRoot(); if(r==null) return null; // offline return r.child(item.getFullName()); } @CheckForNull public FilePath getRootPath() { final SlaveComputer computer = getComputer(); if (computer == null) { // if computer is null then channel is null and thus we were going to return null anyway return null; } else { return createPath(StringUtils.defaultString(computer.getAbsoluteRemoteFs(), remoteFS)); } } /** * Root directory on this agent where all the job workspaces are laid out. * @return * null if not connected. */ public @CheckForNull FilePath getWorkspaceRoot() { FilePath r = getRootPath(); if(r==null) return null; return r.child(WORKSPACE_ROOT); } /** * Web-bound object used to serve jar files for JNLP. */ public static final class JnlpJar implements HttpResponse { private final String fileName; public JnlpJar(String fileName) { this.fileName = fileName; } public void doIndex( StaplerRequest req, StaplerResponse rsp) throws IOException, ServletException { URLConnection con = connect(); // since we end up redirecting users to jnlpJars/foo.jar/, set the content disposition // so that browsers can download them in the right file name. // see http://support.microsoft.com/kb/260519 and http://www.boutell.com/newfaq/creating/forcedownload.html rsp.setHeader("Content-Disposition", "attachment; filename=" + fileName); InputStream in = con.getInputStream(); rsp.serveFile(req, in, con.getLastModified(), con.getContentLength(), "*.jar" ); in.close(); } public void generateResponse(StaplerRequest req, StaplerResponse rsp, Object node) throws IOException, ServletException { doIndex(req,rsp); } private URLConnection connect() throws IOException { URL res = getURL(); return res.openConnection(); } public URL getURL() throws MalformedURLException { String name = fileName; // Prevent the access to war contents & prevent the folder escaping (SECURITY-195) if (!ALLOWED_JNLPJARS_FILES.contains(name)) { throw new MalformedURLException("The specified file path " + fileName + " is not allowed due to security reasons"); } if (name.equals("hudson-cli.jar")) { name="jenkins-cli.jar"; } URL res = Jenkins.getInstance().servletContext.getResource("/WEB-INF/" + name); if(res==null) { // during the development this path doesn't have the files. res = new URL(new File(".").getAbsoluteFile().toURI().toURL(),"target/jenkins/WEB-INF/"+name); } return res; } public byte[] readFully() throws IOException { InputStream in = connect().getInputStream(); try { return IOUtils.toByteArray(in); } finally { in.close(); } } } /** * Creates a launcher for the agent. * * @return * If there is no computer it will return a {@link hudson.Launcher.DummyLauncher}, otherwise it * will return a {@link hudson.Launcher.RemoteLauncher} instead. */ public Launcher createLauncher(TaskListener listener) { SlaveComputer c = getComputer(); if (c == null) { listener.error("Issue with creating launcher for agent " + name + "."); return new Launcher.DummyLauncher(listener); } else { return new RemoteLauncher(listener, c.getChannel(), c.isUnix()).decorateFor(this); } } /** * Gets the corresponding computer object. */ public SlaveComputer getComputer() { return (SlaveComputer)toComputer(); } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; final Slave that = (Slave) o; return name.equals(that.name); } @Override public int hashCode() { return name.hashCode(); } /** * Invoked by XStream when this object is read into memory. */ protected Object readResolve() { // convert the old format to the new one if (launcher == null) { launcher = (agentCommand == null || agentCommand.trim().length() == 0) ? new JNLPLauncher() : new CommandLauncher(agentCommand); } if(nodeProperties==null) nodeProperties = new DescribableList<NodeProperty<?>,NodePropertyDescriptor>(Jenkins.getInstance().getNodesObject()); return this; } public SlaveDescriptor getDescriptor() { Descriptor d = Jenkins.getInstance().getDescriptorOrDie(getClass()); if (d instanceof SlaveDescriptor) return (SlaveDescriptor) d; throw new IllegalStateException(d.getClass()+" needs to extend from SlaveDescriptor"); } public static abstract class SlaveDescriptor extends NodeDescriptor { public FormValidation doCheckNumExecutors(@QueryParameter String value) { return FormValidation.validatePositiveInteger(value); } /** * Performs syntactical check on the remote FS for agents. */ public FormValidation doCheckRemoteFS(@QueryParameter String value) throws IOException, ServletException { if(Util.fixEmptyAndTrim(value)==null) return FormValidation.error(Messages.Slave_Remote_Director_Mandatory()); if(value.startsWith("\\\\") || value.startsWith("/net/")) return FormValidation.warning(Messages.Slave_Network_Mounted_File_System_Warning()); if (Util.isRelativePath(value)) { return FormValidation.warning(Messages.Slave_Remote_Relative_Path_Warning()); } return FormValidation.ok(); } /** * Returns the list of {@link ComputerLauncher} descriptors appropriate to the supplied {@link Slave}. * * @param it the {@link Slave} or {@code null} to assume the slave is of type {@link #clazz}. * @return the filtered list * @since 2.12 */ @Nonnull @Restricted(NoExternalUse.class) // intedned for use by Jelly EL only (plus hack in DelegatingComputerLauncher) public final List<Descriptor<ComputerLauncher>> computerLauncherDescriptors(@CheckForNull Slave it) { DescriptorExtensionList<ComputerLauncher, Descriptor<ComputerLauncher>> all = Jenkins.getInstance().<ComputerLauncher, Descriptor<ComputerLauncher>>getDescriptorList( ComputerLauncher.class); return it == null ? DescriptorVisibilityFilter.applyType(clazz, all) : DescriptorVisibilityFilter.apply(it, all); } /** * Returns the list of {@link RetentionStrategy} descriptors appropriate to the supplied {@link Slave}. * * @param it the {@link Slave} or {@code null} to assume the slave is of type {@link #clazz}. * @return the filtered list * @since 2.12 */ @Nonnull @SuppressWarnings("unchecked") // used by Jelly EL only @Restricted(NoExternalUse.class) // used by Jelly EL only public final List<Descriptor<RetentionStrategy<?>>> retentionStrategyDescriptors(@CheckForNull Slave it) { return it == null ? DescriptorVisibilityFilter.applyType(clazz, RetentionStrategy.all()) : DescriptorVisibilityFilter.apply(it, RetentionStrategy.all()); } /** * Returns the list of {@link NodePropertyDescriptor} appropriate to the supplied {@link Slave}. * * @param it the {@link Slave} or {@code null} to assume the slave is of type {@link #clazz}. * @return the filtered list * @since 2.12 */ @Nonnull @SuppressWarnings("unchecked") // used by Jelly EL only @Restricted(NoExternalUse.class) // used by Jelly EL only public final List<NodePropertyDescriptor> nodePropertyDescriptors(@CheckForNull Slave it) { List<NodePropertyDescriptor> result = new ArrayList<NodePropertyDescriptor>(); Collection<NodePropertyDescriptor> list = (Collection) Jenkins.getInstance().getDescriptorList(NodeProperty.class); for (NodePropertyDescriptor npd : it == null ? DescriptorVisibilityFilter.applyType(clazz, list) : DescriptorVisibilityFilter.apply(it, list)) { if (npd.isApplicable(clazz)) { result.add(npd); } } return result; } } // // backward compatibility // /** * Command line to launch the agent, like * "ssh myslave java -jar /path/to/hudson-remoting.jar" * @deprecated in 1.216 */ @Deprecated private transient String agentCommand; /** * Obtains the clock difference between this side and that side of the channel. * * <p> * This is a hack to wrap the whole thing into a simple {@link Callable}. * * <ol> * <li>When the callable is sent to remote, we capture the time (on this side) in {@link GetClockDifference2#startTime} * <li>When the other side receives the callable it is {@link GetClockDifference2}. * <li>We capture the time on the other side and {@link GetClockDifference3} gets sent from the other side * <li>When it's read on this side as a return value, it morphs itself into {@link ClockDifference}. * </ol> */ private static final class GetClockDifference1 extends MasterToSlaveCallable<ClockDifference,IOException> { public ClockDifference call() { // this method must be being invoked locally, which means the clock is in sync return new ClockDifference(0); } private Object writeReplace() { return new GetClockDifference2(); } private static final long serialVersionUID = 1L; } private static final class GetClockDifference2 extends MasterToSlaveCallable<GetClockDifference3,IOException> { /** * Capture the time on the master when this object is sent to remote, which is when * {@link GetClockDifference1#writeReplace()} is run. */ private final long startTime = System.currentTimeMillis(); public GetClockDifference3 call() { return new GetClockDifference3(startTime); } private static final long serialVersionUID = 1L; } private static final class GetClockDifference3 implements Serializable { private final long remoteTime = System.currentTimeMillis(); private final long startTime; public GetClockDifference3(long startTime) { this.startTime = startTime; } private Object readResolve() { long endTime = System.currentTimeMillis(); return new ClockDifference((startTime + endTime)/2-remoteTime); } } /** * Determines the workspace root file name for those who really really need the shortest possible path name. */ private static final String WORKSPACE_ROOT = SystemProperties.getString(Slave.class.getName()+".workspaceRoot","workspace"); /** * Provides a collection of file names, which are accessible via /jnlpJars link. */ private static final Set<String> ALLOWED_JNLPJARS_FILES = ImmutableSet.of("slave.jar", "remoting.jar", "jenkins-cli.jar", "hudson-cli.jar"); }