aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorFabrice Bellingard <bellingard@gmail.com>2012-06-22 14:04:44 +0200
committerFabrice Bellingard <bellingard@gmail.com>2012-06-22 14:04:44 +0200
commitbf998970a385d499990284f6db90ab1247d90029 (patch)
tree4cd109b5902e1d77ecdcb51f0fda1deb922072eb
parent60adc862836a9eae573587041ff35d090da55d38 (diff)
downloadsonarqube-bf998970a385d499990284f6db90ab1247d90029.tar.gz
sonarqube-bf998970a385d499990284f6db90ab1247d90029.zip
SONAR-3581 Tool to validate a l10n bundle based on multiple plugins
-rw-r--r--sonar-testing-harness/src/main/java/org/sonar/test/i18n/BundleSynchronizedMatcher.java81
-rw-r--r--sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java112
-rw-r--r--sonar-testing-harness/src/test/java/org/sonar/test/i18n/BundleSynchronizedTest.java45
-rw-r--r--sonar-testing-harness/src/test/java/org/sonar/test/i18n/I18nMatchersTest.java52
-rw-r--r--sonar-testing-harness/src/test/resources/org/sonar/l10n/abacus_fr.properties28
5 files changed, 251 insertions, 67 deletions
diff --git a/sonar-testing-harness/src/main/java/org/sonar/test/i18n/BundleSynchronizedMatcher.java b/sonar-testing-harness/src/main/java/org/sonar/test/i18n/BundleSynchronizedMatcher.java
index 7ed8b3384a2..4ae41847827 100644
--- a/sonar-testing-harness/src/main/java/org/sonar/test/i18n/BundleSynchronizedMatcher.java
+++ b/sonar-testing-harness/src/main/java/org/sonar/test/i18n/BundleSynchronizedMatcher.java
@@ -19,6 +19,7 @@
*/
package org.sonar.test.i18n;
+import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import org.apache.commons.io.IOUtils;
@@ -26,8 +27,16 @@ import org.hamcrest.BaseMatcher;
import org.hamcrest.Description;
import org.sonar.test.TestUtils;
-import java.io.*;
+import java.io.BufferedInputStream;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.OutputStream;
import java.net.MalformedURLException;
+import java.net.URI;
import java.net.URL;
import java.util.Collection;
import java.util.Map;
@@ -42,22 +51,32 @@ import static org.junit.Assert.fail;
public class BundleSynchronizedMatcher extends BaseMatcher<String> {
public static final String L10N_PATH = "/org/sonar/l10n/";
- private static final String GITHUB_RAW_FILE_PATH = "https://raw.github.com/SonarSource/sonar/master/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/";
private static final Collection<String> CORE_BUNDLES = Lists.newArrayList("checkstyle.properties", "core.properties",
- "findbugs.properties", "gwt.properties", "pmd.properties", "squidjava.properties");
+ "findbugs.properties", "gwt.properties", "pmd.properties", "squidjava.properties");
+ private static final String GITHUB_RAW_FILE_PATH = "https://raw.github.com/SonarSource/sonar/master/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/";
private String sonarVersion;
+ private URI referenceEnglishBundleURI;
// we use this variable to be able to unit test this class without looking at the real Github core bundles that change all the time
private String remote_file_path;
private String bundleName;
private SortedMap<String, String> missingKeys;
private SortedMap<String, String> additionalKeys;
+ public BundleSynchronizedMatcher() {
+ this(null, GITHUB_RAW_FILE_PATH);
+ }
+
+ public BundleSynchronizedMatcher(URI referenceEnglishBundleURI) {
+ this.referenceEnglishBundleURI = referenceEnglishBundleURI;
+ }
+
public BundleSynchronizedMatcher(String sonarVersion) {
this(sonarVersion, GITHUB_RAW_FILE_PATH);
}
- public BundleSynchronizedMatcher(String sonarVersion, String remote_file_path) {
+ @VisibleForTesting
+ BundleSynchronizedMatcher(String sonarVersion, String remote_file_path) {
this.sonarVersion = sonarVersion;
this.remote_file_path = remote_file_path;
}
@@ -75,6 +94,8 @@ public class BundleSynchronizedMatcher extends BaseMatcher<String> {
File defaultBundle;
if (isCoreBundle(defaultBundleName)) {
defaultBundle = getBundleFileFromGithub(defaultBundleName);
+ } else if (referenceEnglishBundleURI != null) {
+ defaultBundle = getBundleFileFromProvidedURI(defaultBundleName);
} else {
defaultBundle = getBundleFileFromClasspath(defaultBundleName);
}
@@ -167,18 +188,14 @@ public class BundleSynchronizedMatcher extends BaseMatcher<String> {
}
protected File getBundleFileFromGithub(String defaultBundleName) {
- File localBundle = new File("target/l10n/download/" + defaultBundleName);
+ String remoteFile = computeGitHubURL(defaultBundleName, sonarVersion);
+ URL remoteFileURL = null;
try {
- String remoteFile = computeGitHubURL(defaultBundleName, sonarVersion);
- saveUrlToLocalFile(remoteFile, localBundle);
+ remoteFileURL = new URL(remoteFile);
} catch (MalformedURLException e) {
fail("Could not download the original core bundle at: " + remote_file_path + defaultBundleName);
- } catch (IOException e) {
- fail("Could not download the original core bundle at: " + remote_file_path + defaultBundleName);
}
- assertThat("File 'target/tmp/" + defaultBundleName + "' has been downloaded but does not exist.", localBundle, notNullValue());
- assertThat("File 'target/tmp/" + defaultBundleName + "' has been downloaded but does not exist.", localBundle.exists(), is(true));
- return localBundle;
+ return downloadRemoteFile(defaultBundleName, remoteFileURL);
}
protected String computeGitHubURL(String defaultBundleName, String sonarVersion) {
@@ -196,18 +213,29 @@ public class BundleSynchronizedMatcher extends BaseMatcher<String> {
return bundle;
}
- protected String extractDefaultBundleName(String bundleName) {
- int firstUnderScoreIndex = bundleName.indexOf('_');
- assertThat("The bundle '" + bundleName + "' is a default bundle (without locale), so it can't be compared.", firstUnderScoreIndex > 0,
- is(true));
- return bundleName.substring(0, firstUnderScoreIndex) + ".properties";
+ private File getBundleFileFromProvidedURI(String defaultBundleName) {
+ URL remoteFileURL = null;
+ try {
+ remoteFileURL = referenceEnglishBundleURI.toURL();
+ } catch (MalformedURLException e) {
+ fail("Could not download the original bundle at: " + remote_file_path + defaultBundleName);
+ }
+ return downloadRemoteFile(defaultBundleName, remoteFileURL);
}
- protected boolean isCoreBundle(String defaultBundleName) {
- return CORE_BUNDLES.contains(defaultBundleName);
+ private File downloadRemoteFile(String defaultBundleName, URL remoteFileUrl) {
+ File localBundle = new File("target/l10n/download/" + defaultBundleName);
+ try {
+ saveUrlToLocalFile(remoteFileUrl, localBundle);
+ } catch (IOException e) {
+ fail("Could not download the original core bundle at: " + remoteFileUrl.toString() + defaultBundleName);
+ }
+ assertThat("File 'target/tmp/" + defaultBundleName + "' has been downloaded but does not exist.", localBundle, notNullValue());
+ assertThat("File 'target/tmp/" + defaultBundleName + "' has been downloaded but does not exist.", localBundle.exists(), is(true));
+ return localBundle;
}
- private void saveUrlToLocalFile(String url, File localFile) throws IOException {
+ private void saveUrlToLocalFile(URL url, File localFile) throws IOException {
if (localFile.exists()) {
localFile.delete();
}
@@ -216,7 +244,7 @@ public class BundleSynchronizedMatcher extends BaseMatcher<String> {
InputStream in = null;
OutputStream fout = null;
try {
- in = new BufferedInputStream(new URL(url).openStream());
+ in = new BufferedInputStream(url.openStream());
fout = new FileOutputStream(localFile);
byte data[] = new byte[1024];
@@ -230,4 +258,15 @@ public class BundleSynchronizedMatcher extends BaseMatcher<String> {
}
}
+ public static String extractDefaultBundleName(String bundleName) {
+ int firstUnderScoreIndex = bundleName.indexOf('_');
+ assertThat("The bundle '" + bundleName + "' is a default bundle (without locale), so it can't be compared.", firstUnderScoreIndex > 0,
+ is(true));
+ return bundleName.substring(0, firstUnderScoreIndex) + ".properties";
+ }
+
+ public static boolean isCoreBundle(String defaultBundleName) {
+ return CORE_BUNDLES.contains(defaultBundleName);
+ }
+
}
diff --git a/sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java b/sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java
index f053ae11621..ea4ac5f5965 100644
--- a/sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java
+++ b/sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java
@@ -19,18 +19,19 @@
*/
package org.sonar.test.i18n;
-import static org.junit.Assert.assertThat;
-import static org.junit.Assert.fail;
+import com.google.common.collect.Maps;
+import org.apache.commons.io.FileUtils;
+import org.apache.commons.lang.StringUtils;
+import org.sonar.test.TestUtils;
import java.io.File;
+import java.net.URI;
+import java.net.URISyntaxException;
import java.util.Collection;
import java.util.Map;
-import org.apache.commons.io.FileUtils;
-import org.apache.commons.lang.StringUtils;
-import org.sonar.test.TestUtils;
-
-import com.google.common.collect.Maps;
+import static org.junit.Assert.assertThat;
+import static org.junit.Assert.fail;
public final class I18nMatchers {
@@ -38,6 +39,9 @@ public final class I18nMatchers {
}
/**
+ * <p>
+ * <b>Used by language packs that translate Core bundles.</b>
+ * </p>
* Returns a matcher which checks that a translation bundle is up to date with the corresponding English Core bundle.
* <ul>
* <li>If a version of Sonar is specified, then the check is done against this version of the bundle found on Sonar Github repository.</li>
@@ -53,46 +57,109 @@ public final class I18nMatchers {
}
/**
- * Returns a matcher which checks that a translation bundle is up to date with the corresponding default one found in the same folder. <br>
- * <br>
- * This matcher is used for Sonar plugins that embed their own translations.
+ * <p>
+ * <b>Used by language packs that translate third-party bundles.</b>
+ * </p>
+ * Returns a matcher which checks that a translation bundle is up to date with the given reference English bundle from a third-party plugin.
+ *
+ * @param referenceEnglishBundleURI
+ * the URI referencing the English bundle to check against
+ * @return the matcher
+ */
+ public static BundleSynchronizedMatcher isBundleUpToDate(URI referenceEnglishBundleURI) {
+ return new BundleSynchronizedMatcher(referenceEnglishBundleURI);
+ }
+
+ /**
+ * <p>
+ * <b>Used only by independent plugins that embeds their own bundles for every language.</b>
+ * </p>
+ * Returns a matcher which checks that a translation bundle is up to date with the corresponding default one found in the same folder.
*
* @return the matcher
*/
public static BundleSynchronizedMatcher isBundleUpToDate() {
- return new BundleSynchronizedMatcher(null);
+ return new BundleSynchronizedMatcher();
+ }
+
+ /**
+ * <p>
+ * <b>Must be used only by independent plugins that embeds their own bundles for every language.</b>
+ * </p>
+ * Checks that all the translation bundles found on the classpath are up to date with the corresponding default one found in the same
+ * folder.
+ */
+ public static void assertAllBundlesUpToDate() {
+ try {
+ assertAllBundlesUpToDate(null, null);
+ } catch (URISyntaxException e) {
+ // Ignore, this can't happen here
+ }
}
/**
- * Checks that all the Core translation bundles found on the classpath are up to date with the corresponding English ones.
+ * <p>
+ * <b>Must be used only by language packs.</b>
+ * </p>
+ * <p>
+ * Depending on the parameters, this method does the following:
* <ul>
- * <li>If a version of Sonar is specified, then the check is done against this version of the bundles found on Sonar Github repository.</li>
- * <li>If sonarVersion is set to NULL, the check is done against the latest version of this bundles found on Github (master branch).</li>
+ * <li><b>sonarVersion</b>: checks that all the Core translation bundles found on the classpath are up to date with the corresponding English ones found on Sonar
+ * GitHub repository for the given Sonar version.
+ * <ul><li><i>Note: if sonarVersion is set to NULL, the check is done against the latest version of this bundles found the master branch of the GitHub repository.</i></li></ul>
+ * </li>
+ * <li><b>pluginIdsToBundleUrlMap</b>: checks that other translation bundles found on the classpath are up to date with the reference English bundles of the corresponding
+ * plugins given by the "pluginIdsToBundleUrlMap" parameter.
+ * </li>
* </ul>
+ * </p>
+ * <p><br>
+ * The following example will check that the translation of the Core bundles are up to date with version 3.2 of Sonar English Language Pack, and it
+ * will also check that the translation of the bundle of the Web plugin is up to date with the reference English bundle of version 1.2 of the Web plugin:
+ * <pre>
+ * Map<String, String> pluginIdsToBundleUrlMap = Maps.newHashMap();
+ * pluginIdsToBundleUrlMap.put("web", "http://svn.codehaus.org/sonar-plugins/tags/sonar-web-plugin-1.2/src/main/resources/org/sonar/l10n/web.properties");
+ * assertAllBundlesUpToDate("3.2", pluginIdsToBundleUrlMap);
+ * </pre>
+ * </p>
*
* @param sonarVersion
* the version of the bundles to check against, or NULL to check against the latest source on GitHub
+ * @param pluginIdsToBundleUrlMap
+ * a map that gives, for a given plugin, the URL of the English bundle that must be used to check the translation.
+ * @throws URISyntaxException if the provided URLs in the "pluginIdsToBundleUrlMap" parameter are not correct
*/
- public static void assertAllBundlesUpToDate(String sonarVersion) {
+ public static void assertAllBundlesUpToDate(String sonarVersion, Map<String, String> pluginIdsToBundleUrlMap) throws URISyntaxException {
File bundleFolder = TestUtils.getResource(BundleSynchronizedMatcher.L10N_PATH);
if (bundleFolder == null || !bundleFolder.isDirectory()) {
fail("No bundle found in '" + BundleSynchronizedMatcher.L10N_PATH + "'");
}
- Collection<File> bundles = FileUtils.listFiles(bundleFolder, new String[] { "properties" }, false);
+ Collection<File> bundles = FileUtils.listFiles(bundleFolder, new String[] {"properties"}, false);
Map<String, String> failedAssertionMessages = Maps.newHashMap();
for (File bundle : bundles) {
String bundleName = bundle.getName();
if (bundleName.indexOf('_') > 0) {
try {
- assertThat(bundleName, isBundleUpToDate(sonarVersion));
+ String baseBundleName = BundleSynchronizedMatcher.extractDefaultBundleName(bundleName);
+ String pluginId = StringUtils.substringBefore(baseBundleName, ".");
+ if (BundleSynchronizedMatcher.isCoreBundle(baseBundleName)) {
+ // this is a core bundle => must be checked againt the provided version of Sonar
+ assertThat(bundleName, isBundleUpToDate(sonarVersion));
+ } else if (pluginIdsToBundleUrlMap != null && pluginIdsToBundleUrlMap.get(pluginId) != null) {
+ // this is a third-party plugin translated by a language pack => must be checked against the provided URL
+ assertThat(bundleName, isBundleUpToDate(new URI(pluginIdsToBundleUrlMap.get(pluginId))));
+ } else {
+ // this is the case of a plugin that provides all the bundles for every language => check the bundles inside the plugin
+ assertThat(bundleName, isBundleUpToDate());
+ }
} catch (AssertionError e) {
failedAssertionMessages.put(bundleName, e.getMessage());
}
}
}
- if ( !failedAssertionMessages.isEmpty()) {
+ if (!failedAssertionMessages.isEmpty()) {
StringBuilder message = new StringBuilder();
message.append(failedAssertionMessages.size());
message.append(" bundles are not up-to-date: ");
@@ -102,13 +169,4 @@ public final class I18nMatchers {
fail(message.toString());
}
}
-
- /**
- * Checks that all the translation bundles found on the classpath are up to date with the corresponding default one found in the same
- * folder.
- */
- public static void assertAllBundlesUpToDate() {
- assertAllBundlesUpToDate(null);
- }
-
}
diff --git a/sonar-testing-harness/src/test/java/org/sonar/test/i18n/BundleSynchronizedTest.java b/sonar-testing-harness/src/test/java/org/sonar/test/i18n/BundleSynchronizedTest.java
index 8554d464aae..396f196927e 100644
--- a/sonar-testing-harness/src/test/java/org/sonar/test/i18n/BundleSynchronizedTest.java
+++ b/sonar-testing-harness/src/test/java/org/sonar/test/i18n/BundleSynchronizedTest.java
@@ -24,50 +24,56 @@ import org.junit.Test;
import org.sonar.test.TestUtils;
import java.io.File;
+import java.net.URI;
import java.util.SortedMap;
-import static org.hamcrest.Matchers.*;
-import static org.junit.Assert.*;
-import static org.sonar.test.i18n.I18nMatchers.isBundleUpToDate;
+import static org.hamcrest.Matchers.containsString;
+import static org.hamcrest.Matchers.hasItem;
+import static org.hamcrest.Matchers.is;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertThat;
+import static org.junit.Assert.assertTrue;
public class BundleSynchronizedTest {
- private static final String GITHUB_RAW_FILE_PATH = "https://raw.github.com/SonarSource/sonar/master/sonar-testing-harness/src/test/resources/org/sonar/l10n/";
+ private static final String GITHUB_RAW_TESTING_FILE_PATH = "https://raw.github.com/SonarSource/sonar/master/sonar-testing-harness/src/test/resources/org/sonar/l10n/";
private BundleSynchronizedMatcher matcher;
@Before
- public void test() throws Exception {
- matcher = new BundleSynchronizedMatcher(null);
+ public void init() {
+ matcher = new BundleSynchronizedMatcher();
}
@Test
- // The case of a Sonar plugin that embeds all the bundles for every language
- public void testBundlesInsideSonarPlugin() {
+ // The case of a Sonar Language Pack that translates the Core bundles
+ public void shouldMatchBundlesOfLanguagePack() {
// synchronized bundle
- assertThat("myPlugin_fr_CA.properties", isBundleUpToDate());
- assertFalse(new File("target/l10n/myPlugin_fr_CA.properties.report.txt").exists());
+ assertThat("core_fr_CA.properties", new BundleSynchronizedMatcher(null, GITHUB_RAW_TESTING_FILE_PATH));
// missing keys
try {
- assertThat("myPlugin_fr.properties", isBundleUpToDate());
- assertTrue(new File("target/l10n/myPlugin_fr.properties.report.txt").exists());
+ assertThat("core_fr.properties", new BundleSynchronizedMatcher(null, GITHUB_RAW_TESTING_FILE_PATH));
} catch (AssertionError e) {
assertThat(e.getMessage(), containsString("Missing translations are:\nsecond.prop"));
}
}
@Test
- public void shouldNotFailIfNoMissingKeysButAdditionalKeys() {
- assertThat("noMissingKeys_fr.properties", isBundleUpToDate());
+ // The case of a Sonar Language Pack that translates plugin bundles
+ public void shouldMatchWithProvidedURI() throws Exception {
+ matcher = new BundleSynchronizedMatcher(new URI("http://svn.codehaus.org/sonar-plugins/tags/sonar-abacus-plugin-0.1/src/main/resources/org/sonar/l10n/abacus.properties"));
+ assertThat("abacus_fr.properties", matcher);
}
@Test
- // The case of a Sonar Language Pack that translates the Core bundles
- public void testBundlesOfLanguagePack() {
+ // The case of a Sonar plugin that embeds all the bundles for every language
+ public void testBundlesInsideSonarPlugin() {
// synchronized bundle
- assertThat("core_fr_CA.properties", new BundleSynchronizedMatcher(null, GITHUB_RAW_FILE_PATH));
+ assertThat("myPlugin_fr_CA.properties", matcher);
+ assertFalse(new File("target/l10n/myPlugin_fr_CA.properties.report.txt").exists());
// missing keys
try {
- assertThat("core_fr.properties", new BundleSynchronizedMatcher(null, GITHUB_RAW_FILE_PATH));
+ assertThat("myPlugin_fr.properties", matcher);
+ assertTrue(new File("target/l10n/myPlugin_fr.properties.report.txt").exists());
} catch (AssertionError e) {
assertThat(e.getMessage(), containsString("Missing translations are:\nsecond.prop"));
}
@@ -85,7 +91,7 @@ public class BundleSynchronizedTest {
@Test
public void testGetBundleFileFromGithub() throws Exception {
- matcher = new BundleSynchronizedMatcher(null, GITHUB_RAW_FILE_PATH);
+ matcher = new BundleSynchronizedMatcher(null, GITHUB_RAW_TESTING_FILE_PATH);
matcher.getBundleFileFromGithub("core.properties");
assertTrue(new File("target/l10n/download/core.properties").exists());
}
@@ -133,4 +139,5 @@ public class BundleSynchronizedTest {
assertThat(matcher.computeGitHubURL("core.properties", "2.10"),
is("https://raw.github.com/SonarSource/sonar/2.10/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties"));
}
+
}
diff --git a/sonar-testing-harness/src/test/java/org/sonar/test/i18n/I18nMatchersTest.java b/sonar-testing-harness/src/test/java/org/sonar/test/i18n/I18nMatchersTest.java
new file mode 100644
index 00000000000..c97cb3951e5
--- /dev/null
+++ b/sonar-testing-harness/src/test/java/org/sonar/test/i18n/I18nMatchersTest.java
@@ -0,0 +1,52 @@
+/*
+ * Sonar, open source software quality management tool.
+ * Copyright (C) 2008-2012 SonarSource
+ * mailto:contact AT sonarsource DOT com
+ *
+ * Sonar 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.
+ *
+ * Sonar 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 Sonar; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02
+ */
+package org.sonar.test.i18n;
+
+import org.junit.Test;
+
+import java.io.File;
+
+import static org.hamcrest.Matchers.containsString;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertThat;
+import static org.junit.Assert.assertTrue;
+import static org.sonar.test.i18n.I18nMatchers.isBundleUpToDate;
+
+public class I18nMatchersTest {
+
+ @Test
+ public void testBundlesInsideSonarPlugin() {
+ // synchronized bundle
+ assertThat("myPlugin_fr_CA.properties", isBundleUpToDate());
+ assertFalse(new File("target/l10n/myPlugin_fr_CA.properties.report.txt").exists());
+ // missing keys
+ try {
+ assertThat("myPlugin_fr.properties", isBundleUpToDate());
+ assertTrue(new File("target/l10n/myPlugin_fr.properties.report.txt").exists());
+ } catch (AssertionError e) {
+ assertThat(e.getMessage(), containsString("Missing translations are:\nsecond.prop"));
+ }
+ }
+
+ @Test
+ public void shouldNotFailIfNoMissingKeysButAdditionalKeys() {
+ assertThat("noMissingKeys_fr.properties", isBundleUpToDate());
+ }
+}
diff --git a/sonar-testing-harness/src/test/resources/org/sonar/l10n/abacus_fr.properties b/sonar-testing-harness/src/test/resources/org/sonar/l10n/abacus_fr.properties
new file mode 100644
index 00000000000..03ee2266df9
--- /dev/null
+++ b/sonar-testing-harness/src/test/resources/org/sonar/l10n/abacus_fr.properties
@@ -0,0 +1,28 @@
+## -------- Test file for the BundleSynchronizedMatcher -------- ##
+widget.abacus.name=Abaques
+widget.abacus.description=Calcule la complexit\u00e9 de chaque composant pour vous aider \u00e0 utiliser vos abaques.
+
+widget.abacus.param.defaultColors=Liste de couleurs (format hexad\u00e9cimal) s\u00e9par\u00e9e par des virgules pour l'affichage du camembert.<br>Bien faire attention \u00e0 ce que le nombre de couleurs soit au minimum le m\u00eame que le nombre de niveaux de complexit\u00e9 des abaques.
+widget.abacus.param.defaultDisplay=Valeurs possibles :<ul class="bullet"><li>files (= nombre de fichiers)</li><li>percentage (= pourcentage)</li></ul>
+
+abacusTab.page=Abaques
+abacusTab.notComputed=Non calcul\u00e9e
+
+property.sonar.abacus.complexityThresholds.name=Seuils de complexit\u00e9 des abaques
+property.sonar.abacus.complexityThresholds.description=Usage : NomSeuil1:Complexit\u00e9Seuil1;NomSeuil2:Complexit\u00e9Seuil2;...;NomSeuilN<br>Valeur par d\u00e9faut : Simple:20;Medium:50;Complex:100;Very Complex
+
+metric.abacus-complexity.name=Complexit\u00e9 (Abaques)
+metric.abacus-complexity.description=Complexit\u00e9 (Abaques)
+metric.abacus-complexity-distribution.name=Distribution de la complexit\u00e9 (Abaques)
+metric.abacus-complexity-distribution.description=Distribution de la complexit\u00e9 (Abaques)
+
+abacus.componentComplexity=Complexit\u00e9 du composant
+abacus.componentComplexityDistribution.numberOfFiles=Distrib. complexit\u00e9 (fichiers)
+abacus.componentComplexityDistribution.percentage=Distrib. complexit\u00e9 (%)
+abacus.error.cannotDisplayWidget=Impossible d'afficher le widget.
+abacus.error.defaultColors.incorrectValue=La valeur du param\u00e8tre de wigdet <i>defaultColors</i> est incorrecte : moins de couleurs que de niveaux de complexit\u00e9 dans les abaques.
+abacus.error.defaultDisplay.incorrectValue=La valeur du param\u00e8tre de widget <i>defaultDisplay</i> est incorrecte : <i>{0}</i><br>Valeurs possibles :<ul class="bullet"><li>files (= nombre de fichiers)</li><li>percentage (= pourcentage)</li></ul>
+abacus.noData=Pas de donn\u00e9es
+abacus.numberOfFilesDistribution=Distribution en nombre de fichiers
+abacus.percentageDistribution=Distribution en %
+abacus.title=Abaques \ No newline at end of file