/******************************************************************************* * * Copyright (c) 2004-2010 Oracle Corporation. * * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * * Kohsuke Kawaguchi, Erik Ramfelt, Martin Eigenbrodt, Stephen Connolly, Tom Huybrechts * * *******************************************************************************/ package hudson.model; import hudson.FilePath; import hudson.Launcher; import hudson.Util; import hudson.Launcher.RemoteLauncher; import hudson.diagnosis.OldDataMonitor; import hudson.model.Descriptor.FormException; import hudson.remoting.Callable; import hudson.remoting.VirtualChannel; 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.List; import java.util.Set; import javax.servlet.ServletException; import org.apache.commons.io.IOUtils; import org.kohsuke.stapler.DataBoundConstructor; import org.kohsuke.stapler.HttpResponse; import org.kohsuke.stapler.QueryParameter; import org.kohsuke.stapler.StaplerRequest; import org.kohsuke.stapler.StaplerResponse; /** * Information about a Hudson slave 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}. * * @author Kohsuke Kawaguchi */ public abstract class Slave extends Node implements Serializable { /** * Name of this slave node. */ protected String name; /** * Description of this node. */ private final String description; /** * Path to the root of the workspace from the view point of this node, such * as "/hudson" */ protected final String remoteFS; /** * Number of executors of this node. */ private int numExecutors = 2; /** * Job allocation strategy. */ private Mode mode; /** * Slave availablility strategy. */ private RetentionStrategy retentionStrategy; /** * The starter that will startup this slave. */ private ComputerLauncher launcher; /** * Whitespace-separated labels. */ private String label = ""; private /*almost final*/ DescribableList<NodeProperty<?>, NodePropertyDescriptor> nodeProperties = new DescribableList<NodeProperty<?>, NodePropertyDescriptor>(Hudson.getInstance()); /** * Lazily computed set of labels from {@link #label}. */ private transient volatile Set<Label> labels; @DataBoundConstructor 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(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); 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); } } 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; } public void setNodeName(String name) { this.name = name; } public String getNodeDescription() { return description; } public int getNumExecutors() { return numExecutors; } public Mode getMode() { return mode; } public void setMode(Mode mode) { this.mode = mode; } public DescribableList<NodeProperty<?>, NodePropertyDescriptor> getNodeProperties() { return nodeProperties; } public RetentionStrategy getRetentionStrategy() { return retentionStrategy == null ? RetentionStrategy.Always.INSTANCE : retentionStrategy; } public void setRetentionStrategy(RetentionStrategy availabilityStrategy) { this.retentionStrategy = availabilityStrategy; } public String getLabelString() { return Util.fixNull(label).trim(); } public ClockDifference getClockDifference() throws IOException, InterruptedException { VirtualChannel channel = getChannel(); if (channel == null) { throw new IOException(getNodeName() + " is offline"); } long startTime = System.currentTimeMillis(); long slaveTime = channel.call(new GetSystemTime()); long endTime = System.currentTimeMillis(); return new ClockDifference((startTime + endTime) / 2 - slaveTime); } public Computer createComputer() { return new SlaveComputer(this); } public FilePath getWorkspaceFor(TopLevelItem item) { FilePath r = getWorkspaceRoot(); if (r == null) { return null; // offline } return r.child(item.getName()); } public FilePath getRootPath() { return createPath(remoteFS); } /** * Root directory on this slave where all the job workspaces are laid out. * * @return null if not connected. */ public 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) { if (fileName.startsWith("/")){ fileName = fileName.substring(1); } 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 { URL res = Hudson.getInstance().servletContext.getResource("/WEB-INF/" + fileName); if (res == null) { // during the development this path doesn't have the files. res = new URL(new File(".").getAbsoluteFile().toURI().toURL(), "target/generated-resources/WEB-INF/" + fileName); } return res; } public byte[] readFully() throws IOException { InputStream in = connect().getInputStream(); try { return IOUtils.toByteArray(in); } finally { in.close(); } } } public Launcher createLauncher(TaskListener listener) { SlaveComputer c = getComputer(); 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. */ public Object readResolve() { // convert the old format to the new one if (command != null && agentCommand == null) { if (command.length() > 0) { command += ' '; } agentCommand = command + "java -jar ~/bin/slave.jar"; } if (command != null || localFS != null) { OldDataMonitor.report(Hudson.getInstance(), "1.69"); } if (launcher == null) { launcher = (agentCommand == null || agentCommand.trim().length() == 0) ? new JNLPLauncher() : new CommandLauncher(agentCommand); } if (nodeProperties == null) { nodeProperties = new DescribableList<NodeProperty<?>, NodePropertyDescriptor>(Hudson.getInstance()); } return this; } public SlaveDescriptor getDescriptor() { Descriptor d = Hudson.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 slaves. */ 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()); } return FormValidation.ok(); } } // // backward compatibility // /** * In Hudson < 1.69 this was used to store the local file path to the remote * workspace. No longer in use. * * * @deprecated ... but still in use during the transition. */ private File localFS; /** * In Hudson < 1.69 this was used to store the command to connect to the * remote machine, like "ssh myslave". * * * @deprecated */ private transient String command; /** * Command line to launch the agent, like "ssh myslave java -jar * /path/to/hudson-remoting.jar" */ private transient String agentCommand; /** * Obtains the system clock. */ private static final class GetSystemTime implements Callable<Long, RuntimeException> { public Long call() { return System.currentTimeMillis(); } private static final long serialVersionUID = 1L; } /** * Determines the workspace root file name for those who really really need * the shortest possible path name. */ private static final String WORKSPACE_ROOT = System.getProperty(Slave.class.getName() + ".workspaceRoot", "workspace"); }