1 /*******************************************************************************
2  * Copyright (c) 2009, 2018 Mountainminds GmbH & Co. KG and Contributors
3  * All rights reserved. This program and the accompanying materials
4  * are made available under the terms of the Eclipse Public License v1.0
5  * which accompanies this distribution, and is available at
6  * http://www.eclipse.org/legal/epl-v10.html
7  *
8  * Contributors:
9  *    Marc R. Hoffmann - initial API and implementation
10  *
11  *******************************************************************************/
12 package org.jacoco.report.check;
13 
14 import java.math.BigDecimal;
15 import java.math.RoundingMode;
16 import java.util.Collections;
17 import java.util.HashMap;
18 import java.util.Map;
19 
20 import org.jacoco.core.analysis.ICounter.CounterValue;
21 import org.jacoco.core.analysis.ICoverageNode;
22 import org.jacoco.core.analysis.ICoverageNode.CounterEntity;
23 
24 /**
25  * Descriptor for a limit which is given by a {@link Rule}.
26  */
27 public class Limit {
28 
29 	private static final Map<CounterValue, String> VALUE_NAMES;
30 	private static final Map<CounterEntity, String> ENTITY_NAMES;
31 
32 	static {
33 		final Map<CounterValue, String> values = new HashMap<CounterValue, String>();
values.put(CounterValue.TOTALCOUNT, "total count")34 		values.put(CounterValue.TOTALCOUNT, "total count");
values.put(CounterValue.MISSEDCOUNT, "missed count")35 		values.put(CounterValue.MISSEDCOUNT, "missed count");
values.put(CounterValue.COVEREDCOUNT, "covered count")36 		values.put(CounterValue.COVEREDCOUNT, "covered count");
values.put(CounterValue.MISSEDRATIO, "missed ratio")37 		values.put(CounterValue.MISSEDRATIO, "missed ratio");
values.put(CounterValue.COVEREDRATIO, "covered ratio")38 		values.put(CounterValue.COVEREDRATIO, "covered ratio");
39 		VALUE_NAMES = Collections.unmodifiableMap(values);
40 
41 		final Map<CounterEntity, String> entities = new HashMap<CounterEntity, String>();
entities.put(CounterEntity.INSTRUCTION, "instructions")42 		entities.put(CounterEntity.INSTRUCTION, "instructions");
entities.put(CounterEntity.BRANCH, "branches")43 		entities.put(CounterEntity.BRANCH, "branches");
entities.put(CounterEntity.COMPLEXITY, "complexity")44 		entities.put(CounterEntity.COMPLEXITY, "complexity");
entities.put(CounterEntity.LINE, "lines")45 		entities.put(CounterEntity.LINE, "lines");
entities.put(CounterEntity.METHOD, "methods")46 		entities.put(CounterEntity.METHOD, "methods");
entities.put(CounterEntity.CLASS, "classes")47 		entities.put(CounterEntity.CLASS, "classes");
48 		ENTITY_NAMES = Collections.unmodifiableMap(entities);
49 	}
50 
51 	private CounterEntity entity;
52 
53 	private CounterValue value;
54 
55 	private BigDecimal minimum;
56 
57 	private BigDecimal maximum;
58 
59 	/**
60 	 * Creates a new instance with the following defaults:
61 	 * <ul>
62 	 * <li>counter entity: {@link CounterEntity#INSTRUCTION}
63 	 * <li>counter value: {@link CounterValue#COVEREDRATIO}
64 	 * <li>minimum: no limit
65 	 * <li>maximum: no limit
66 	 * </ul>
67 	 */
Limit()68 	public Limit() {
69 		this.entity = CounterEntity.INSTRUCTION;
70 		this.value = CounterValue.COVEREDRATIO;
71 	}
72 
73 	/**
74 	 * @return the configured counter entity to check
75 	 */
getEntity()76 	public CounterEntity getEntity() {
77 		return entity;
78 	}
79 
80 	/**
81 	 * Sets the counter entity to check.
82 	 *
83 	 * @param entity
84 	 *            counter entity to check
85 	 * TODO: use CounterEntity directly once Maven 3 is required.
86 	 */
setCounter(final String entity)87 	public void setCounter(final String entity) {
88 		this.entity = CounterEntity.valueOf(entity);
89 	}
90 
91 	/**
92 	 * @return the configured value to check
93 	 */
getValue()94 	public CounterValue getValue() {
95 		return value;
96 	}
97 
98 	/**
99 	 * Sets the value to check.
100 	 *
101 	 * @param value
102 	 *            value to check
103 	 * TODO: use CounterValue directly once Maven 3 is required.
104 	 */
setValue(final String value)105 	public void setValue(final String value) {
106 		this.value = CounterValue.valueOf(value);
107 	}
108 
109 	/**
110 	 * @return configured minimum value, or <code>null</code> if no minimum is
111 	 *         given
112 	 */
getMinimum()113 	public String getMinimum() {
114 		return minimum == null ? null : minimum.toPlainString();
115 	}
116 
117 	/**
118 	 * Sets allowed maximum value as decimal string or percent representation.
119 	 * The given precision is also considered in error messages. Coverage ratios
120 	 * are given in the range from 0.0 to 1.0.
121 	 *
122 	 * @param minimum
123 	 *            allowed minimum or <code>null</code>, if no minimum should be
124 	 *            checked
125 	 */
setMinimum(final String minimum)126 	public void setMinimum(final String minimum) {
127 		this.minimum = parseValue(minimum);
128 	}
129 
130 	/**
131 	 * @return configured maximum value, or <code>null</code> if no maximum is
132 	 *         given
133 	 */
getMaximum()134 	public String getMaximum() {
135 		return maximum == null ? null : maximum.toPlainString();
136 	}
137 
138 	/**
139 	 * Sets allowed maximum value as decimal string or percent representation.
140 	 * The given precision is also considered in error messages. Coverage ratios
141 	 * are given in the range from 0.0 to 1.0.
142 	 *
143 	 * @param maximum
144 	 *            allowed maximum or <code>null</code>, if no maximum should be
145 	 *            checked
146 	 */
setMaximum(final String maximum)147 	public void setMaximum(final String maximum) {
148 		this.maximum = parseValue(maximum);
149 	}
150 
parseValue(final String value)151 	private static BigDecimal parseValue(final String value) {
152 		if (value == null) {
153 			return null;
154 		}
155 
156 		final String trimmedValue = value.trim();
157 		if (trimmedValue.endsWith("%")) {
158 			final String percent = trimmedValue.substring(0, trimmedValue.length() - 1);
159 			return new BigDecimal(percent).movePointLeft(2);
160 		}
161 
162 		return new BigDecimal(trimmedValue);
163 	}
164 
check(final ICoverageNode node)165 	String check(final ICoverageNode node) {
166 		final double d = node.getCounter(entity).getValue(value);
167 		if (Double.isNaN(d)) {
168 			return null;
169 		}
170 		final BigDecimal bd = BigDecimal.valueOf(d);
171 		if (minimum != null && minimum.compareTo(bd) > 0) {
172 			return message("minimum", bd, minimum, RoundingMode.FLOOR);
173 		}
174 		if (maximum != null && maximum.compareTo(bd) < 0) {
175 			return message("maximum", bd, maximum, RoundingMode.CEILING);
176 		}
177 		return null;
178 	}
179 
message(final String minmax, final BigDecimal v, final BigDecimal ref, final RoundingMode mode)180 	private String message(final String minmax, final BigDecimal v,
181 			final BigDecimal ref, final RoundingMode mode) {
182 		final BigDecimal rounded = v.setScale(ref.scale(), mode);
183 		return String.format("%s %s is %s, but expected %s is %s",
184 				ENTITY_NAMES.get(entity), VALUE_NAMES.get(value),
185 				rounded.toPlainString(), minmax, ref.toPlainString());
186 	}
187 
188 }
189