Java tutorial
/* * SonarQube * Copyright (C) 2009-2016 SonarSource SA * mailto:contact AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ package org.sonar.db.version.v50; import com.google.common.base.Splitter; import com.google.common.collect.ImmutableList; import java.io.ByteArrayOutputStream; import java.io.OutputStreamWriter; import java.io.StringReader; import java.util.Collections; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import javax.xml.stream.XMLInputFactory; import javax.xml.stream.XMLStreamException; import org.apache.commons.codec.digest.DigestUtils; import org.apache.commons.lang.StringUtils; import org.codehaus.stax2.XMLInputFactory2; import org.codehaus.staxmate.SMInputFactory; import org.codehaus.staxmate.in.SMHierarchicCursor; import org.codehaus.staxmate.in.SMInputCursor; import org.sonar.api.measures.CoreMetrics; import org.sonar.api.utils.KeyValueFormat; import org.sonar.api.utils.text.CsvWriter; import static java.nio.charset.StandardCharsets.UTF_8; class FileSourceDto { private static final String SPACE_CHARS = "\t\n\r "; private Iterator<String> sourceSplitter; private final Map<Integer, String> revisions; private final Map<Integer, String> authors; private final Map<Integer, String> dates; private final Map<Integer, String> utHits; private final Map<Integer, String> utConditions; private final Map<Integer, String> utCoveredConditions; private final Map<Integer, String> itHits; private final Map<Integer, String> itConditions; private final Map<Integer, String> itCoveredConditions; private final Map<Integer, String> overallHits; private final Map<Integer, String> overallConditions; private final Map<Integer, String> overallCoveredConditions; private final List<List<Block>> duplicationGroups; FileSourceDto(String source, String revisions, String authors, String dates, String utHits, String utConditions, String utCoveredConditions, String itHits, String itConditions, String itCoveredConditions, String overallHits, String overallConditions, String overallCoveredConditions, String duplicationData) { sourceSplitter = Splitter.onPattern("\r?\n|\r").split(source).iterator(); this.revisions = KeyValueFormat.parseIntString(revisions); this.authors = KeyValueFormat.parseIntString(authors); this.dates = KeyValueFormat.parseIntString(dates); this.utHits = KeyValueFormat.parseIntString(utHits); this.utConditions = KeyValueFormat.parseIntString(utConditions); this.utCoveredConditions = KeyValueFormat.parseIntString(utCoveredConditions); this.itHits = KeyValueFormat.parseIntString(itHits); this.itConditions = KeyValueFormat.parseIntString(itConditions); this.itCoveredConditions = KeyValueFormat.parseIntString(itCoveredConditions); this.overallHits = KeyValueFormat.parseIntString(overallHits); this.overallConditions = KeyValueFormat.parseIntString(overallConditions); this.overallCoveredConditions = KeyValueFormat.parseIntString(overallCoveredConditions); this.duplicationGroups = StringUtils.isNotBlank(duplicationData) ? parseDuplicationData(duplicationData) : Collections.<List<Block>>emptyList(); } String[] getSourceData() { String highlighting = ""; String symbolRefs = ""; Map<Integer, String> duplicationsPerLine = computeDuplicationsPerLine(duplicationGroups); ByteArrayOutputStream output = new ByteArrayOutputStream(); int line = 0; String sourceLine; CsvWriter csv = CsvWriter.of(new OutputStreamWriter(output, UTF_8)); StringBuilder lineHashes = new StringBuilder(); while (sourceSplitter.hasNext()) { line++; sourceLine = sourceSplitter.next(); lineHashes.append(lineChecksum(sourceLine)).append("\n"); csv.values(revisions.get(line), authors.get(line), dates.get(line), utHits.get(line), utConditions.get(line), utCoveredConditions.get(line), itHits.get(line), itConditions.get(line), itCoveredConditions.get(line), overallHits.get(line), overallConditions.get(line), overallCoveredConditions.get(line), highlighting, symbolRefs, duplicationsPerLine.get(line), sourceLine); } csv.close(); return new String[] { new String(output.toByteArray(), UTF_8), lineHashes.toString() }; } public static String lineChecksum(String line) { String reducedLine = StringUtils.replaceChars(line, SPACE_CHARS, ""); if (reducedLine.isEmpty()) { return ""; } return DigestUtils.md5Hex(reducedLine); } private Map<Integer, String> computeDuplicationsPerLine(List<List<Block>> duplicationGroups) { Map<Integer, String> result = new HashMap<>(); if (duplicationGroups.isEmpty()) { return result; } Map<Integer, StringBuilder> dupPerLine = new HashMap<>(); int blockId = 1; for (List<Block> group : duplicationGroups) { Block originBlock = group.get(0); addBlock(blockId, originBlock, dupPerLine); blockId++; for (int i = 1; i < group.size(); i++) { Block duplicate = group.get(i); if (duplicate.resourceKey.equals(originBlock.resourceKey)) { addBlock(blockId, duplicate, dupPerLine); blockId++; } } } for (Map.Entry<Integer, StringBuilder> entry : dupPerLine.entrySet()) { result.put(entry.getKey(), entry.getValue().toString()); } return result; } private static void addBlock(int blockId, Block block, Map<Integer, StringBuilder> dupPerLine) { int currentLine = block.start; for (int i = 0; i < block.length; i++) { if (dupPerLine.get(currentLine) == null) { dupPerLine.put(currentLine, new StringBuilder()); } if (dupPerLine.get(currentLine).length() > 0) { dupPerLine.get(currentLine).append(','); } dupPerLine.get(currentLine).append(blockId); currentLine++; } } /** * Parses data of {@link CoreMetrics#DUPLICATIONS_DATA}. */ private static List<List<Block>> parseDuplicationData(String data) { ImmutableList.Builder<List<Block>> groups = ImmutableList.builder(); try { StringReader reader = new StringReader(data); SMInputFactory inputFactory = initStax(); SMHierarchicCursor rootC = inputFactory.rootElementCursor(reader); // <duplications> rootC.advance(); SMInputCursor groupsCursor = rootC.childElementCursor("g"); while (groupsCursor.getNext() != null) { // <g> SMInputCursor blocksCursor = groupsCursor.childElementCursor("b"); ImmutableList.Builder<Block> group = ImmutableList.builder(); while (blocksCursor.getNext() != null) { // <b> String resourceKey = blocksCursor.getAttrValue("r"); int firstLine = getAttrIntValue(blocksCursor, "s"); int numberOfLines = getAttrIntValue(blocksCursor, "l"); group.add(new Block(resourceKey, firstLine, numberOfLines)); } groups.add(group.build()); } } catch (Exception e) { // SONAR-6174 Ignore any issue while parsing duplication measure. There is nothing user can do and things will get solved after // next analysis anyway } return groups.build(); } private static int getAttrIntValue(SMInputCursor cursor, String attrName) throws XMLStreamException { return cursor.getAttrIntValue(cursor.findAttrIndex(null, attrName)); } private static SMInputFactory initStax() { XMLInputFactory xmlFactory = XMLInputFactory2.newInstance(); return new SMInputFactory(xmlFactory); } private static class Block { final String resourceKey; final int start; final int length; public Block(String resourceKey, int s, int l) { this.resourceKey = resourceKey; this.start = s; this.length = l; } } }