/** * Copyright (c) 2009 Cliffano Subagio * * 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.plugins.sitemonitor; import hudson.Launcher; import hudson.model.AbstractBuild; import hudson.model.BuildListener; import hudson.plugins.sitemonitor.model.Result; import hudson.plugins.sitemonitor.model.Site; import hudson.plugins.sitemonitor.model.Status; import hudson.tasks.BuildStepMonitor; import hudson.tasks.Recorder; import java.io.IOException; import java.net.HttpURLConnection; import java.net.SocketTimeoutException; import java.net.URL; import java.net.UnknownHostException; import java.util.ArrayList; import java.util.List; /** * Performs the web site monitoring process. * @author cliffano */ public class SiteMonitorRecorder extends Recorder { /** * 1 sec = 1000 msecs . */ private static final int MILLISECS_IN_SECS = 1000; /** * The list of web sites to monitor. */ private List<Site> mSites; /** * Construct {@link SiteMonitorRecorder}. * @param sites * the list of web sites to monitor */ public SiteMonitorRecorder(final List<Site> sites) { mSites = sites; } /** * @return the list of web sites to monitor */ public final List<Site> getSites() { return mSites; } /** * Performs the web site monitoring by checking the response code of the * site's URL. * @param build * the build * @param launcher * the launcher * @param listener * the listener * @return true if all sites give success response codes, false otherwise * @throws InterruptedException * when there's an interruption * @throws IOException * when there's an IO error */ @Override public final boolean perform(final AbstractBuild<?, ?> build, final Launcher launcher, final BuildListener listener) throws InterruptedException, IOException { List<Result> results = new ArrayList<Result>(); SiteMonitorDescriptor descriptor = (SiteMonitorDescriptor) getDescriptor(); boolean hasFailure = false; for (Site site : mSites) { Integer responseCode = null; Status status; String note = ""; try { HttpURLConnection connection = (HttpURLConnection) (new URL( site.getUrl())).openConnection(); connection.setConnectTimeout(descriptor.getTimeout() * MILLISECS_IN_SECS); responseCode = connection.getResponseCode(); List<Integer> successResponseCodes = descriptor .getSuccessResponseCodes(); if (successResponseCodes.contains(responseCode)) { status = Status.UP; } else { status = Status.ERROR; } } catch (SocketTimeoutException ste) { listener.getLogger().println(ste + " - " + ste.getMessage()); status = Status.DOWN; } catch (Exception e) { note = e + " - " + e.getMessage(); listener.getLogger().println(note); status = Status.EXCEPTION; } note = "[" + status + "] " + note; listener.getLogger().println( "URL: " + site.getUrl() + ", response code: " + responseCode + ", status: " + status); if (!hasFailure && status != Status.UP) { hasFailure = true; } Result result = new Result(site, responseCode, status, note); results.add(result); } build.addAction(new SiteMonitorRootAction(results)); hudson.model.Result result; if (hasFailure) { result = hudson.model.Result.FAILURE; } else { result = hudson.model.Result.SUCCESS; } build.setResult(result); // the return value is not used when this class implements a Recorder, // it's left here just in case this class switches to a Builder. // http://n4.nabble.com/how-can-a-Recorder-mark-build-as-failure-td1746654.html return !hasFailure; } /** * Gets the required monitor service. * @return the BuildStepMonitor */ public final BuildStepMonitor getRequiredMonitorService() { return BuildStepMonitor.NONE; } }