001: /********************************************************************************
002: * CruiseControl, a Continuous Integration Toolkit
003: * Copyright (c) 2007, ThoughtWorks, Inc.
004: * 200 E. Randolph, 25th Floor
005: * Chicago, IL 60601 USA
006: * All rights reserved.
007: *
008: * Redistribution and use in source and binary forms, with or without
009: * modification, are permitted provided that the following conditions
010: * are met:
011: *
012: * + Redistributions of source code must retain the above copyright
013: * notice, this list of conditions and the following disclaimer.
014: *
015: * + Redistributions in binary form must reproduce the above
016: * copyright notice, this list of conditions and the following
017: * disclaimer in the documentation and/or other materials provided
018: * with the distribution.
019: *
020: * + Neither the name of ThoughtWorks, Inc., CruiseControl, nor the
021: * names of its contributors may be used to endorse or promote
022: * products derived from this software without specific prior
023: * written permission.
024: *
025: * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
026: * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
027: * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
028: * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR
029: * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
030: * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
031: * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
032: * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
033: * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
034: * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
035: * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
036: ********************************************************************************/package net.sourceforge.cruisecontrol.dashboard;
037:
038: import java.io.File;
039: import java.util.ArrayList;
040: import java.util.Collections;
041: import java.util.HashMap;
042: import java.util.LinkedHashMap;
043: import java.util.List;
044: import java.util.Map;
045:
046: import net.sourceforge.cruisecontrol.dashboard.utils.CCDateFormatter;
047: import net.sourceforge.cruisecontrol.dashboard.utils.TimeConverter;
048:
049: import org.apache.commons.lang.StringUtils;
050: import org.joda.time.DateTime;
051:
052: public class BuildDetail implements Comparable, Build {
053:
054: private DateTime date;
055:
056: private String timeStamp;
057:
058: private Map pluginOutpus = new LinkedHashMap();
059:
060: private final Map props;
061:
062: private final LogFile logFile;
063:
064: private CurrentStatus currentStatus = CurrentStatus.WAITING;
065:
066: private final TimeConverter timeConverter;
067:
068: public BuildDetail(LogFile logFile) {
069: this (logFile, new HashMap(), new TimeConverter());
070: }
071:
072: public BuildDetail(LogFile logFile, Map props) {
073: this (logFile, props, new TimeConverter());
074: }
075:
076: public BuildDetail(LogFile logFile, Map props,
077: TimeConverter timeConverter) {
078: this .logFile = logFile;
079: this .timeConverter = timeConverter;
080: this .props = Collections.unmodifiableMap(props);
081: }
082:
083: public String getProjectName() {
084: return (String) props.get("projectname");
085: }
086:
087: public String getLabel() {
088: return (String) props.get("label");
089: }
090:
091: public String getDuration() {
092: return (String) props.get("duration");
093: }
094:
095: public LogFile getLogFile() {
096: return logFile;
097: }
098:
099: public String getLogFileName() {
100: return this .getBuildLogFilename();
101: }
102:
103: public File getLogFolder() {
104: return getLogFile().getParentFile();
105: }
106:
107: public List getTestSuites() {
108: return (List) props.get("testsuites");
109: }
110:
111: public String getTimeStamp() {
112: if (timeStamp == null) {
113: timeStamp = CCDateFormatter.yyyyMMddHHmmss(this
114: .getBuildDate());
115: }
116: return timeStamp;
117: }
118:
119: public int compareTo(Object o) {
120: return this .getBuildDate().compareTo(
121: ((BuildDetail) o).getBuildDate());
122: }
123:
124: public ModificationSet getModificationSet() {
125: return (ModificationSet) props.get("modifications");
126: }
127:
128: public int getNumberOfTests() {
129: int totalNumberOfTests = 0;
130:
131: List testSuites = getTestSuites();
132: for (int i = 0; i < testSuites.size(); i++) {
133: BuildTestSuite suite = (BuildTestSuite) testSuites.get(i);
134: totalNumberOfTests += suite.getNumberOfTests();
135: }
136:
137: return totalNumberOfTests;
138: }
139:
140: public int getNumberOfFailures() {
141: int numberOfFailures = 0;
142:
143: List testSuites = getTestSuites();
144: for (int i = 0; i < testSuites.size(); i++) {
145: BuildTestSuite suite = (BuildTestSuite) testSuites.get(i);
146: numberOfFailures += suite.getNumberOfFailures();
147: }
148:
149: return numberOfFailures;
150: }
151:
152: public int getNumberOfErrors() {
153: int numberOfErrors = 0;
154:
155: List testSuites = getTestSuites();
156: for (int i = 0; i < testSuites.size(); i++) {
157: BuildTestSuite suite = (BuildTestSuite) testSuites.get(i);
158: numberOfErrors += suite.getNumberOfErrors();
159: }
160:
161: return numberOfErrors;
162: }
163:
164: public boolean hasPassed() {
165: return StringUtils.contains(getLogFileName(), "L");
166: }
167:
168: public File getArtifactFolder() {
169: File folder = (File) props.get("artifactfolder");
170: return new File(folder, getTimeStamp());
171: }
172:
173: public List getArtifactFiles() {
174: List result = new ArrayList();
175: File[] files = getArtifactFolder().listFiles();
176: if (files == null) {
177: return result;
178: }
179: for (int i = 0; i < files.length; i++) {
180: if (!files[i].isHidden()) {
181: result.add(files[i]);
182: }
183: }
184: return result;
185: }
186:
187: public Map getPluginOutputs() {
188: return pluginOutpus;
189: }
190:
191: public void addPluginOutput(String category, Object output) {
192: pluginOutpus.put(category, output);
193: }
194:
195: public DateTime getBuildDate() {
196: if (date == null) {
197: date = CCDateFormatter.formatLogName(getLogFileName());
198: }
199: return date;
200: }
201:
202: public String getBuildLogFilename() {
203: return getLogFile().getName();
204: }
205:
206: public DateTime getBuildingSince() {
207: return getBuildDate();
208: }
209:
210: public CurrentStatus getCurrentStatus() {
211: return currentStatus;
212: }
213:
214: public void updateStatus(CurrentStatus currentStatus) {
215: this .currentStatus = currentStatus;
216: }
217:
218: public PreviousResult getPreviousBuildResult() {
219: return hasPassed() ? PreviousResult.PASSED
220: : PreviousResult.FAILED;
221: }
222:
223: public String getConvertedTime() {
224: return this.timeConverter.getConvertedTime(getBuildDate()
225: .toDate());
226: }
227: }
|