All Downloads are FREE. Search and download functionalities are using the official Maven repository.

hudson.util.DoubleLaunchChecker Maven / Gradle / Ivy

The newest version!
/*******************************************************************************
 *
 * 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
 *     
 *
 *******************************************************************************/ 

package hudson.util;

import hudson.model.Hudson;
import hudson.stapler.WebAppController;
import hudson.triggers.SafeTimerTask;
import hudson.triggers.Trigger;
import org.apache.commons.io.FileUtils;
import org.kohsuke.stapler.StaplerRequest;
import org.kohsuke.stapler.StaplerResponse;

import javax.servlet.ServletException;
import javax.servlet.ServletContext;
import static javax.servlet.http.HttpServletResponse.SC_INTERNAL_SERVER_ERROR;
import java.io.File;
import java.io.IOException;
import java.util.Random;
import java.util.logging.Logger;
import java.util.logging.Level;
import java.lang.management.ManagementFactory;
import java.lang.reflect.Method;

/**
 * Makes sure that no other Hudson uses our HUDSON_HOME directory,
 * to forestall the problem of running multiple instances of Hudson that point to the same data directory.
 *
 * 

* This set up error occasionally happens especialy when the user is trying to reassign the context path of the app, * and it results in a hard-to-diagnose error, so we actively check this. * *

* The mechanism is simple. This class occasionally updates a known file inside the hudson home directory, * and whenever it does so, it monitors the timestamp of the file to make sure no one else is updating * this file. In this way, while we cannot detect the problem right away, within a reasonable time frame * we can detect the collision. * *

* More traditional way of doing this is to use a lock file with PID in it, but unfortunately in Java, * there's no reliabe way to obtain PID. * * @author Kohsuke Kawaguchi * @since 1.178 */ public class DoubleLaunchChecker { /** * The timestamp of the owner file when we updated it for the last time. * 0 to indicate that there was no update before. */ private long lastWriteTime = 0L; /** * Once the error is reported, the user can choose to ignore and proceed anyway, * in which case the flag is set to true. */ private boolean ignore = false; private final Random random = new Random(); public final File home; /** * ID string of the other Hudson that we are colliding with. * Can be null. */ private String collidingId; public DoubleLaunchChecker() { home = Hudson.getInstance().getRootDir(); } protected void execute() { File timestampFile = new File(home,".owner"); long t = timestampFile.lastModified(); if(t!=0 && lastWriteTime!=0 && t!=lastWriteTime && !ignore) { try { collidingId = FileUtils.readFileToString(timestampFile); } catch (IOException e) { LOGGER.log(Level.SEVERE, "Failed to read collision file", e); } // we noticed that someone else have updated this file. // switch GUI to display this error. WebAppController.get().install(this); LOGGER.severe("Collision detected. timestamp="+t+", expected="+lastWriteTime); // we need to continue updating this file, so that the other Hudson would notice the problem, too. } try { FileUtils.writeStringToFile(timestampFile, getId()); lastWriteTime = timestampFile.lastModified(); } catch (IOException e) { // if failed to write, err on the safe side and assume things are OK. lastWriteTime=0; } schedule(); } /** * Figures out a string that identifies this instance of Hudson. */ public String getId() { Hudson h = Hudson.getInstance(); // in servlet 2.5, we can get the context path String contextPath=""; try { Method m = ServletContext.class.getMethod("getContextPath"); contextPath=" contextPath=\""+m.invoke(h.servletContext)+"\""; } catch (Exception e) { // maybe running with Servlet 2.4 } return h.hashCode()+contextPath+" at "+ManagementFactory.getRuntimeMXBean().getName(); } public String getCollidingId() { return collidingId; } /** * Schedules the next execution. */ public void schedule() { // randomize the scheduling so that multiple Hudson instances will write at the file at different time long MINUTE = 1000*60; Trigger.timer.schedule(new SafeTimerTask() { protected void doRun() { execute(); } },(random.nextInt(30)+60)*MINUTE); } /** * Serve all URLs with the index view. */ public void doDynamic(StaplerRequest req, StaplerResponse rsp) throws IOException, ServletException { rsp.setStatus(SC_INTERNAL_SERVER_ERROR); req.getView(this,"index.jelly").forward(req,rsp); } /** * Ignore the problem and go back to using Hudson. */ public void doIgnore(StaplerRequest req, StaplerResponse rsp) throws IOException { ignore = true; WebAppController.get().install(Hudson.getInstance()); rsp.sendRedirect2(req.getContextPath()+'/'); } private static final Logger LOGGER = Logger.getLogger(DoubleLaunchChecker.class.getName()); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy