/******************************************************************************* * * Copyright (c) 2004-2009 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, id:cactusman * * *******************************************************************************/ package hudson.model; import hudson.model.RunMap.Constructor; import hudson.Extension; import org.kohsuke.stapler.StaplerRequest; import org.kohsuke.stapler.StaplerResponse; import javax.servlet.ServletException; import javax.servlet.http.HttpServletResponse; import java.io.File; import java.io.IOException; /** * Job that runs outside Hudson whose result is submitted to Hudson (either via * web interface, or simply by placing files on the file system, for * compatibility.) * * @author Kohsuke Kawaguchi */ public class ExternalJob extends ViewJob<ExternalJob, ExternalRun> implements TopLevelItem { public ExternalJob(String name) { this(Hudson.getInstance(), name); } public ExternalJob(ItemGroup parent, String name) { super(parent, name); } @Override protected void reload() { this.runs.load(this, new Constructor<ExternalRun>() { public ExternalRun create(File dir) throws IOException { return new ExternalRun(ExternalJob.this, dir); } }); } // keep track of the previous time we started a build private transient long lastBuildStartTime; /** * Creates a new build of this project for immediate execution. * * Needs to be synchronized so that two {@link #newBuild()} invocations * serialize each other. */ public synchronized ExternalRun newBuild() throws IOException { // make sure we don't start two builds in the same second // so the build directories will be different too long timeSinceLast = System.currentTimeMillis() - lastBuildStartTime; if (timeSinceLast < 1000) { try { Thread.sleep(1000 - timeSinceLast); } catch (InterruptedException e) { } } lastBuildStartTime = System.currentTimeMillis(); ExternalRun run = new ExternalRun(this); runs.put(run); return run; } /** * Used to check if this is an external job and ready to accept a build * result. */ public void doAcceptBuildResult(StaplerResponse rsp) throws IOException, ServletException { rsp.setStatus(HttpServletResponse.SC_OK); } /** * Used to post the build result from a remote machine. */ public void doPostBuildResult(StaplerRequest req, StaplerResponse rsp) throws IOException, ServletException { checkPermission(AbstractProject.BUILD); ExternalRun run = newBuild(); run.acceptRemoteSubmission(req.getReader()); rsp.setStatus(HttpServletResponse.SC_OK); } public TopLevelItemDescriptor getDescriptor() { return DESCRIPTOR; } @Extension public static final TopLevelItemDescriptor DESCRIPTOR = new DescriptorImpl(); @Override public String getPronoun() { return Messages.ExternalJob_Pronoun(); } public static final class DescriptorImpl extends TopLevelItemDescriptor { public String getDisplayName() { return Messages.ExternalJob_DisplayName(); } public ExternalJob newInstance(ItemGroup parent, String name) { return new ExternalJob(parent, name); } } }