diff options
author | Fabrice Bellingard <bellingard@gmail.com> | 2012-06-22 14:04:44 +0200 |
---|---|---|
committer | Fabrice Bellingard <bellingard@gmail.com> | 2012-06-22 14:04:44 +0200 |
commit | bf998970a385d499990284f6db90ab1247d90029 (patch) | |
tree | 4cd109b5902e1d77ecdcb51f0fda1deb922072eb /sonar-testing-harness/src/main/java/org/sonar/test | |
parent | 60adc862836a9eae573587041ff35d090da55d38 (diff) | |
download | sonarqube-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.java | 81 | ||||
-rw-r--r-- | sonar-testing-harness/src/main/java/org/sonar/test/i18n/I18nMatchers.java | 112 |
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); - } - } |