aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-testing-harness/src/main/java/org/sonar/test
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 /sonar-testing-harness/src/main/java/org/sonar/test
parent60adc862836a9eae573587041ff35d090da55d38 (diff)
downloadsonarqube-bf998970a385d499990284f6db90ab1247d90029.tar.gz
sonarqube-bf998970a385d499990284f6db90ab1247d90029.zip
SONAR-3581 Tool to validate a l10n bundle based on multiple plugins
Diffstat (limited to 'sonar-testing-harness/src/main/java/org/sonar/test')
-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
2 files changed, 145 insertions, 48 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);
- }
-
}