/******************************************************************************** * CruiseControl, a Continuous Integration Toolkit * Copyright (c) 2007, ThoughtWorks, Inc. * 200 E. Randolph, 25th Floor * Chicago, IL 60601 USA * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * + Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * + Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * + Neither the name of ThoughtWorks, Inc., CruiseControl, nor the * names of its contributors may be used to endorse or promote * products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ********************************************************************************/ package net.sourceforge.cruisecontrol.dashboard.web; import net.sourceforge.cruisecontrol.dashboard.BuildDetail; import net.sourceforge.cruisecontrol.dashboard.BuildLiveDetail; import net.sourceforge.cruisecontrol.dashboard.BuildSummary; import net.sourceforge.cruisecontrol.dashboard.CurrentStatus; import net.sourceforge.cruisecontrol.dashboard.PreviousResult; import net.sourceforge.cruisecontrol.dashboard.service.BuildLoopQueryService; import net.sourceforge.cruisecontrol.dashboard.service.BuildService; import net.sourceforge.cruisecontrol.dashboard.service.BuildSummaryUIService; import net.sourceforge.cruisecontrol.dashboard.service.HistoricalBuildSummariesService; import net.sourceforge.cruisecontrol.dashboard.service.WidgetPluginService; import net.sourceforge.cruisecontrol.dashboard.utils.CCDateFormatter; import net.sourceforge.cruisecontrol.dashboard.utils.DashboardUtils; import net.sourceforge.cruisecontrol.dashboard.widgets.Widget; import net.sourceforge.cruisecontrol.util.DateUtil; import org.joda.time.DateTime; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.HashMap; import java.util.Map; import java.text.ParseException; public class BuildDetailController extends BaseMultiActionController { private final BuildService buildService; private final HistoricalBuildSummariesService buildSummarySerivce; private final WidgetPluginService widgetPluginService; private final BuildSummaryUIService buildSummaryUIService; private final BuildLoopQueryService buildLoopQueryService; public BuildDetailController(BuildService buildService, HistoricalBuildSummariesService buildSummarySerivce, WidgetPluginService widgetPluginService, BuildSummaryUIService buildSummaryUIService, BuildLoopQueryService buildLoopQueryService) { this.buildService = buildService; this.buildSummarySerivce = buildSummarySerivce; this.widgetPluginService = widgetPluginService; this.buildSummaryUIService = buildSummaryUIService; this.buildLoopQueryService = buildLoopQueryService; this.setSupportedMethods(new String[]{"GET"}); } public ModelAndView latest(HttpServletRequest request, HttpServletResponse response) throws Exception { String[] url = DashboardUtils.urlToParams(request.getRequestURI()); String projectName = DashboardUtils.decode(url[url.length - 1]); if (isBuilding(projectName)) { return live(projectName); } else { BuildSummary latest = this.buildSummarySerivce.getLatest(projectName); BuildDetail build = this.buildService.getBuild(projectName, latest.getBuildLogFileDateTime()); loadAllWidgets(request, build); return new ModelAndView("page_build_detail", buildDetail(projectName, build)); } } public ModelAndView history(HttpServletRequest request, HttpServletResponse response) throws Exception { String[] url = DashboardUtils.urlToParams(request.getRequestURI()); String projectName = DashboardUtils.decode(url[url.length - 2]); String yyyyMMddHHmmss = url[url.length - 1]; BuildDetail build = buildService.getBuild(projectName, yyyyMMddHHmmss); loadAllWidgets(request, build); return new ModelAndView("page_build_detail", buildDetail(projectName, build)); } public ModelAndView live(String projectName) throws Exception { PreviousResult lastBuildStatus = buildSummaryUIService.getLastBuildStatus(projectName); BuildLiveDetail build = buildService.getActiveBuild(projectName, lastBuildStatus); Map model = buildDetail(projectName, build); model.put("buildSince", buildStartTime(projectName)); model.put("buildDuration", buildSummaryUIService.getLastBuildDuration(projectName)); return new ModelAndView("page_building", model); } private String buildStartTime(String projectName) throws ParseException { String buildStartTime = buildLoopQueryService.getProjectInfo(projectName).getBuildStartTime(); DateTime startTime = new DateTime(DateUtil.parseIso8601(buildStartTime)); return CCDateFormatter.getDateStringInHumanBeingReadingStyle(startTime); } private boolean isBuilding(String projectName) { String buildStatus = getCurrentBuildStatus(projectName); return CurrentStatus.BUILDING.equals(CurrentStatus.getProjectBuildStatus(buildStatus)); } public ModelAndView handleError(HttpServletRequest request, HttpServletResponse response, Exception e) throws Exception { String[] url = DashboardUtils.urlToParams(request.getRequestURI()); Map data = new HashMap(); String projectName = ""; final String partial; if (url.length == 6) { partial = "projectlog"; projectName = DashboardUtils.decode(url[url.length - 2]); data.put("log", url[url.length - 1]); } else if (url.length == 5) { partial = "project"; projectName = DashboardUtils.decode(url[url.length - 1]); } else { partial = "noproject"; } data.put("projectName", projectName); return new ModelAndView("forward:/exceptions/builddetail/" + partial, data); } private Map buildDetail(String projectName, BuildDetail build) { Map model = new HashMap(); model.put("historicalBuildCmds", buildSummaryUIService.transform(buildSummarySerivce .getLastest25(projectName))); model.put("buildCmd", buildSummaryUIService.transformWithLevel(build)); model.put("logfile", build.getBuildLogFilename()); model.put("durationToSuccessfulBuild", buildSummarySerivce.getDurationFromLastSuccessfulBuild( projectName, build.getBuildDate())); return model; } private String getCurrentBuildStatus(String projectName) { return buildLoopQueryService.getAllProjectsStatus().get(projectName); } private void loadAllWidgets(HttpServletRequest request, BuildDetail build) { HashMap contextProperties = new HashMap(); contextProperties.put(Widget.PARAM_WEB_CONTEXT_PATH, request.getContextPath()); widgetPluginService.mergePluginOutput(build, contextProperties); } }