001    /*
002     * Sonar, open source software quality management tool.
003     * Copyright (C) 2009 SonarSource SA
004     * mailto:contact AT sonarsource DOT com
005     *
006     * Sonar is free software; you can redistribute it and/or
007     * modify it under the terms of the GNU Lesser General Public
008     * License as published by the Free Software Foundation; either
009     * version 3 of the License, or (at your option) any later version.
010     *
011     * Sonar is distributed in the hope that it will be useful,
012     * but WITHOUT ANY WARRANTY; without even the implied warranty of
013     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
014     * Lesser General Public License for more details.
015     *
016     * You should have received a copy of the GNU Lesser General Public
017     * License along with Sonar; if not, write to the Free Software
018     * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02
019     */
020    package org.sonar.api.utils;
021    
022    import com.google.common.collect.Multiset;
023    import org.apache.commons.collections.Bag;
024    import org.apache.commons.lang.StringUtils;
025    import org.apache.commons.lang.math.NumberUtils;
026    import org.sonar.api.rules.RulePriority;
027    import org.slf4j.LoggerFactory;
028    
029    import java.util.HashMap;
030    import java.util.Map;
031    
032    /**
033     * Util class to format key/value data. Output is a string representation ready to be
034     * injected into the database
035     *
036     * @since 1.10
037     */
038    public final class KeyValueFormat {
039    
040      private KeyValueFormat() {
041      }
042    
043      /**
044       * Transforms a string with the following format : "key1=value1;key2=value2..."
045       * into a Map<KEY, VALUE>. Requires to implement the transform(key,value) method
046       *
047       * @param data        the input string
048       * @param transformer the interface to implement
049       * @return a Map of <key, value>
050       */
051      public static <KEY, VALUE> Map<KEY, VALUE> parse(String data, Transformer<KEY, VALUE> transformer) {
052        Map<String, String> rawData = parse(data);
053        Map<KEY, VALUE> map = new HashMap<KEY, VALUE>();
054        for (Map.Entry<String, String> entry : rawData.entrySet()) {
055          KeyValue<KEY, VALUE> keyVal = transformer.transform(entry.getKey(), entry.getValue());
056          if (keyVal != null) {
057            map.put(keyVal.getKey(), keyVal.getValue());
058          }
059        }
060        return map;
061      }
062    
063      /**
064       * Transforms a string with the following format : "key1=value1;key2=value2..."
065       * into a Map<String,String>
066       *
067       * @param data the string to parse
068       * @return a map
069       */
070      public static Map<String, String> parse(String data) {
071        Map<String, String> map = new HashMap<String, String>();
072        String[] pairs = StringUtils.split(data, ";");
073        for (String pair : pairs) {
074          String[] keyValue = StringUtils.split(pair, "=");
075          String key = keyValue[0];
076          String value = (keyValue.length == 2 ? keyValue[1] : "");
077          map.put(key, value);
078        }
079        return map;
080      }
081    
082      /**
083       * Transforms a map<KEY,VALUE> into a string with the format : "key1=value1;key2=value2..."
084       *
085       * @param map the map to transform
086       * @return the formatted string
087       */
088      public static <KEY, VALUE> String format(Map<KEY, VALUE> map) {
089        StringBuilder sb = new StringBuilder();
090        boolean first = true;
091        for (Map.Entry<?, ?> entry : map.entrySet()) {
092          if (!first) {
093            sb.append(";");
094          }
095          sb.append(entry.getKey().toString());
096          sb.append("=");
097          if (entry.getValue() != null) {
098            sb.append(entry.getValue());
099          }
100          first = false;
101        }
102    
103        return sb.toString();
104      }
105    
106      /**
107       * @since 1.11
108       * @deprecated use Multiset from google collections instead of commons-collections bags
109       */
110      public static String format(Bag bag) {
111        return format(bag, 0);
112      }
113    
114      /**
115       * @since 1.11
116       * @deprecated use Multiset from google collections instead of commons-collections bags
117       */
118      public static String format(Bag bag, int var) {
119        StringBuilder sb = new StringBuilder();
120        boolean first = true;
121        for (Object obj : bag.uniqueSet()) {
122          if (!first) {
123            sb.append(";");
124          }
125          sb.append(obj.toString());
126          sb.append("=");
127          sb.append(bag.getCount(obj) + var);
128          first = false;
129        }
130    
131        return sb.toString();
132      }
133    
134      /**
135       * Transforms a Multiset<?> into a string with the format : "key1=count1;key2=count2..."
136       *
137       * @param set the set to transform
138       * @return the formatted string
139       */
140      public static String format(Multiset<?> set) {
141        StringBuilder sb = new StringBuilder();
142        boolean first = true;
143        for (Multiset.Entry<?> entry : set.entrySet()) {
144          if (!first) {
145            sb.append(";");
146          }
147          sb.append(entry.getElement().toString());
148          sb.append("=");
149          sb.append(entry.getCount());
150          first = false;
151        }
152        return sb.toString();
153      }
154    
155      /**
156       * Transforms a Object... into a string with the format : "object1=object2;object3=object4..."
157       *
158       * @param objects the object list to transform
159       * @return the formatted string
160       */
161      public static String format(Object... objects) {
162        StringBuilder sb = new StringBuilder();
163        boolean first = true;
164        if (objects != null) {
165          for (int i = 0; i < objects.length; i++) {
166            if (!first) {
167              sb.append(";");
168            }
169            sb.append(objects[i++].toString());
170            sb.append("=");
171            sb.append(objects[i]);
172            first = false;
173          }
174        }
175        return sb.toString();
176      }
177    
178      public interface Transformer<KEY, VALUE> {
179        KeyValue<KEY, VALUE> transform(String key, String value);
180      }
181    
182      /**
183       * Implementation of Transformer<String, Double>
184       */
185      public static class StringNumberPairTransformer implements Transformer<String, Double> {
186    
187        public KeyValue<String, Double> transform(String key, String value) {
188          return new KeyValue<String, Double>(key, toDouble(value));
189        }
190      }
191    
192      /**
193       * Implementation of Transformer<Double, Double>
194       */
195      public static class DoubleNumbersPairTransformer implements Transformer<Double, Double> {
196    
197        public KeyValue<Double, Double> transform(String key, String value) {
198          return new KeyValue<Double, Double>(toDouble(key), toDouble(value));
199        }
200      }
201    
202      /**
203       * Implementation of Transformer<Integer, Integer>
204       */
205      public static class IntegerNumbersPairTransformer implements Transformer<Integer, Integer> {
206    
207        public KeyValue<Integer, Integer> transform(String key, String value) {
208          return new KeyValue<Integer, Integer>(toInteger(key), toInteger(value));
209        }
210      }
211    
212      /**
213       * Implementation of Transformer<RulePriority, Integer>
214       */
215      public static class RulePriorityNumbersPairTransformer implements Transformer<RulePriority, Integer> {
216    
217        public KeyValue<RulePriority, Integer> transform(String key, String value) {
218          try {
219            if (StringUtils.isBlank(value)) { value = "0"; }
220            return new KeyValue<RulePriority, Integer>(RulePriority.valueOf(key.toUpperCase()), Integer.parseInt(value));
221          }
222          catch (Exception e) {
223            LoggerFactory.getLogger(RulePriorityNumbersPairTransformer.class).warn("Property " + key + " has invalid value: " + value, e);
224            return null;
225          }
226        }
227      }
228    
229      private static Double toDouble(String value) {
230        return StringUtils.isBlank(value) ? null : NumberUtils.toDouble(value);
231      }
232    
233      private static Integer toInteger(String value) {
234        return StringUtils.isBlank(value) ? null : NumberUtils.toInt(value);
235      }
236    }