[rfc][icedtea-web]Localized Man Page Script
Jiri Vanek
jvanek at redhat.com
Fri Jul 11 07:31:03 UTC 2014
On 07/10/2014 09:31 PM, Jie Kang wrote:
> Hello
>
> I have modified the code a bit and added test cases in response to your suggestions. I have also created a OptionParser utility class to take care of parsing the arguments given to main(String[] args). A future patch will make use of this in other classes (such as Boot) in order to remove duplication. Feedback is appreciated!
>
> ----- Original Message -----
>> >On 06/26/2014 07:39 PM, Jie Kang wrote:
>>> > >Hello,
>>> > >
>>> > >I've started working on a tiny bit of the implementation for the Java-based
>>> > >Local Man Page Generator.
>> >
>> >Thank you for taking care of it and thank you for sharing thought so early.
>>> > >
>>> > >At the moment it lets you create man pages through main() by specifying
>>> > >locale and name of man page.
>> >
>> >The class looks quite ok, but I'm not sure of it usefulness.
>> >
>> >Few nits to class itself
>> > * I would put all this work to package
>> >net.sourceforge.jnlp.resources....generator?util?manpages?infogenerator?
> Okay.
Looking to you new class. I would really like to put it one level inside resources:
netx/net/sourceforge/jnlp/resources/infogenerator/..allClassesEndEnumsWithOtherPossiblePackages.java
>
>> > * I would not use private classes (non of the enums here have the right to
>> > be ) - and each public
>
> I disagree a little bit. Traditonally, constants have been provided using 'static final Object'. However, enums were introduced in order to enhance and improve the way constants are defined in Java. The enums in this situation define constants for the class. I can agree that my implementation is not the very best but I would prefer enums over private static final objects.
>
> As well, with the enums we can easily provide help messages for all of our modules (this work is in progress).
Oh you misunderstand me. The enum is best for this case! My compalins were against hiding as inner
class.
>
> Alternatively we can define these enum constants in some public class so that all of ITW can use them. This prevents duplication which occurs in the code already.
Why not class per enum? It will be hidden in package, so it will be msot clean of all.,
>
> E.g. If you look at MessageProperties.java in test-extensions/ : it basically has the same enum and some methods that my code could use, but is in test-extensions. As well there are other files such as JNLPRuntime that use constants for "Message.properties" and "net.sourceforge.jnlp.resources" so we can remove this duplication with public enum class. This fix should be a separate patch.
>
> However, if you wish I can remove the enums and replace them with 'private static final' objects.
Of course not :)) Sorry for saying it so wrongly.
But go with regular public enums.
>
>> >member should have tests on each method ;)
>
> I have added unit tests for the exposed methods.
>
>> >
>> >Well the usecase - I think that you do need this class at all.
>> >
>> >I believe, that before the generation, you can simply set LANG
>> >
>> >(
>> >LANGBACKUP=$LANG
>> >for E in "" ".UTF-8" ; do #some more encoding? Do we need to bother with
>> >encodings at all?
>> >for L in "" en_US$E pl_PL$E de_DE$E cs_CZ$E ; do #is "" wonted/needed?
>> >export LANG=$L
>> >java -CP=... net.sourceforge.jnlp.resources.some_main_in_subpackage
>> >-output={javaws|policieditor|itwsettings|icedtea-web} -outputDir=someDir
>> >done
>> >done
>> >LANG=$LANGBACKUP
>> >export LANG
>> >)
> The current approach can be used very similar to the script you gave above. (I don't think we need to bother with encodings, subject to change but it is very easy to modify) It can work as follows:
>
> for L in "" en_US$E pl_PL$E de_DE$E cs_CZ$E ; do
> java -CP=... net.sourceforge.jnlp.resources.some_main_in_subpackage
> -lang $L -name javaws policyeditor itw-settings icedtea-web #we can add outputDir argument if necessary, but we should be able to generate it based off $L and $M
>
>> >
>> >and use JNLPRuntime + Translator to get correct messages.
>> >
You will not like me, but looking to your class, it is moving e to usage of LANG and runtime even more.
Usage of JnlpRuntime is no overhad. in this case, absolute zero. The overhead is to have class which
is pretending to do the same.
> Using JNLPRuntime means JNLPRuntime being initialized multiple times.
> The JNLPRuntime initialization includes a lot of things we don't need for this generator so I
don't think this is a good idea.
They does not matter.
> Also setting LANG over and over is unnecessary when we can supply arguments to main().
No. there is crucial difference. When you put it in as arg, then you are constructing the properties
filename again. Thats unnecessary and evil.
> Using the JNLPRuntime + Translator seems like a huge amount of extra overhead for what we need here. Using JNLPRuntime means JNLPRuntime being initialized multiple times. The JNLPRuntime initialization includes a lot of things we don't need for this generator so I don't think this is a good idea. Also setting LANG over and over is unnecessary when we can supply arguments to main().
>
>
>
>
>> >But this approach must be verified.
>> >
>> >Otherwise this will bring up duplicated code.
> This is a very valid point for using JNLPRuntime. However, I think we can consider extracting a few methods from JNLPRuntime into something like a ResourceBundleManager (similar to how we OptionParser addition is an extraction from Boot) in order to reduce the duplication.
I like this idea. If you separte this part (taking care of Trasnaltor) of jnlpruntime to sublass,
which you can then reuse without the whole (rest of) runtime it will be probably the best
>
>> >
>> >Also I guess the correct location inside outputDir can be guessed from
>> >locale (or if it more simple
>> >to generate it from $L and $E then I probablky do not care)
>> >
>> >
>> >well my $0.02 :)
>> >
>> >
>>> > >This is for during build process.
>> >
>> >Yes, thats correct. Build time is the place when this is supposed to happen.
>> >
>>> > >
>>> > >
>>> > >It is attached. Jiri, could you look over it? I am still unsure if this is
>>> > >the direction you want to go in.
>> >
>> >Thank you for keeping me in loop!
>>> > >
>>> > >----- Original Message -----
>>>> > >>On 05/20/2014 09:53 PM, Andrew Azores wrote:
>>>>> > >>>On 05/20/2014 03:52 PM, Jie Kang wrote:
>>>>>> > >>>>Hello,
>>>>>> > >>>>
>>>>>> > >>>>I've made all the suggested fixes and am working on the makefile at the
>>>>>> > >>>>moment.
>>>>>> > >>>>
>>>>>> > >>>>Thanks,
>>>>>> > >>>>
>>>>>> > >>>>
>>>>> > >>>
>>>>> > >>>Looks good to me. Please post the new patch when you have your script
>>>>> > >>>integrated and tested working
>>>>> > >>>in the Makefile.
>>>>> > >>>
>>>>> > >>>Thanks,
>>>>> > >>>
>>>> > >>
>>>> > >>
>>>> > >>hi!
>>>> > >>
>>>> > >>The creepy voice from behind the sea have nasty words to say :(
>>>> > >>
>>>> > >>I'm afraid we ca not use this approach. And as it is it must not be
>>>> > >>pushed.
>>>> > >>
>>>> > >>The original idea of this feature was to*reuse* already existing,
>>>> > >>localized
>>>> > >>properties files.
>>>> > >>Also it was intended to share output between manpages*and* applications
>>>> > >>(javaws, policieditor,
>>>> > >>itw-settngs) --help option. Also maybe add icedtea-web manpage, and
>>>> > >>(probably) also have an
>>>> > >>possibility to generate more then man pages - eg html "man pages" or
>>>> > >>whatever.
>>>> > >>
>>>> > >>You can see that -help swihhc already provides eg full list of programs
>>>> > >>switches, or some about
>>>> > >>sntences....
>>>> > >>
>>>> > >>Well your approach do not implement any of this. Well, as opposite it
>>>> > >>file
>>>> > >>new file to localize
>>>> > >>and even more fiels to maintain.
>>>> > >>
>>>> > >>My original idea on this to have java based generator, included in netx
>>>> > >>itself.
>>>> > >>Its "main" class will be able to :
>>>> > >> - be called during the build (just put $CLASSES to CLASSPATH and invoke
>>>> > >> main)
>>>> > >> - generate man pages in reqested language
>>>> > >> - genreate html man pages (well I think there is some man2html, so
>>>> > >> maybe
>>>> > >> this is waste of
>>>> > >>everything)
>>>> > >> - the html files may save as base for help in gui modes
>>>> > >> - be called from javaws/itwsettings/policieditor and so
>>>> > >> - generate -help output
>>>> > >> - implementation detail - do not add "language swithc" generate files
>>>> > >> baed
>>>> > >> of currently set LOCALE
>>>> > >> - much more :)
>>>> > >>
>>>> > >>As benefit you will get
>>>> > >> - all the messages will come form properties
>>>> > >> - only properties to maintain!
>>>> > >> - parsing of properties in free - unicode support, and insertion of
>>>> > >> parametres from JDK itself
>>>> > >> - improved -help output (from man pages)
>>>> > >> - improved manpages form -help outputs
>>>> > >> - list of all switches
>>>> > >> - all default file locations!
>>>> > >> - get rid of about/help code from main classes of itw
>>>> > >> - much more;)
>>>> > >>
>>>> > >>If some messages are in man pages, but not in properties, you can add them
>>>> > >>into proeprties, but
>>>> > >>please verify that there is no similar in properties. And of course vice
>>>> > >>versa.
>>>> > >>
>>>> > >>Well most ofhttp://icedtea.classpath.org/wiki/IcedTea-Web#Release_Plans
>>>> > >>since 1.3 are mostly
>>>> > >>somehow inserted by me. Sometime from need, sometimes from wish (feel free
>>>> > >>to
>>>> > >>add yours, or suggest
>>>> > >>modifications!) But pelase, always consult a bit before first patch.
>>>> > >>Mostly
>>>> > >>IRCis enough :)
>>>> > >>
>>>> > >>Sorry for this email:( But this is really stop show for this approach).
>>>> > >>
>>>> > >>Please try to post the patch ins small hunks, but it can be harder then
>>>> > >>write patch itself, andnot
>>>> > >>always possible, or meaningful.
>>>> > >>I hope I havenot forget something :( But we can tune it during the
>>>> > >>process.
>>>> > >>But I'm afraid nowyou
>>>> > >>are dammed to wait with final push on this topic to me returned from PTO
>>>> > >>:o)
>>>> > >>
>>>> > >>All the best!
>>>> > >> J
>>>> > >>
>>>> > >>
>>>> > >>Few notes to code itself just "education purposes":
>>>> > >> - changelog - the first line of changes should be caption. Eg "Added
>>>> > >> generator of man pages" for
>>>> > >>this particualr setthsi
>>>> > >>
>>>> > >>date-name-email
>>>> > >>[emtyline]
>>>> > >>[tab]caption
>>>> > >>[tab]*file:changes
>>>> > >>
>>>> > >>
>>>> > >>In changelog - always clearly say that it is new file. like
>>>> > >>[tab[ * file: new file, its purpose
>>>> > >>
>>>> > >>
>>>> > >>I'm not sure how good templates for man pages will be suitable in java
>>>> > >>based
>>>> > >>solution.. Well then
>>>> > >>may be. We will see. I hoped to have some class like "Content provider"
>>>> > >>which
>>>> > >>will use some class
>>>> > >>"markup" and together they will provide
>>>> > >> - context of desired verbosity
>>>> > >> - marked in markup you wont
>>>> > >>
>>>> > >>You can think about it as execise from some "system architecture" because
>>>> > >>it
>>>> > >>is isolated system in
>>>> > >>netx :) (whch you are designing from ground)
>>>> > >>
>>>> > >>In case of template - I would suggest %{key} rather then simple @KEY
>>>> > >>(although for your sh based
>>>> > >>solution @KEY is most correct)
>>>> > >>
>>>> > >>Otherwise splendid work on code, and sorry for wasting it:(
>>>> > >>
>>>> > >>
>>> > >
>> >
>> >
>
> Thanks,
>
> -- Jie Kang
>
Pleasure on my side, and really sorry with disagreing with your design so fiercely.
Please see also inline:
>
> localmanscript-2.patch
>
>
> diff --git a/netx/net/sourceforge/jnlp/resources/InfoGenerator.java b/netx/net/sourceforge/jnlp/resources/InfoGenerator.java
> new file mode 100644
> --- /dev/null
> +++ b/netx/net/sourceforge/jnlp/resources/InfoGenerator.java
As already told, please move to resources' subpackage
> @@ -0,0 +1,192 @@
> +/*Copyright (C) 2014 Red Hat, Inc.
> +
> +This file is part of IcedTea.
> +
> +IcedTea is free software; you can redistribute it and/or
> +modify it under the terms of the GNU General Public License as published by
> +the Free Software Foundation, version 2.
> +
> +IcedTea 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
> +General Public License for more details.
> +
> +You should have received a copy of the GNU General Public License
> +along with IcedTea; see the file COPYING. If not, write to
> +the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
> +02110-1301 USA.
> +
> +Linking this library statically or dynamically with other modules is
> +making a combined work based on this library. Thus, the terms and
> +conditions of the GNU General Public License cover the whole
> +combination.
> +
> +As a special exception, the copyright holders of this library give you
> +permission to link this library with independent modules to produce an
> +executable, regardless of the license terms of these independent
> +modules, and to copy and distribute the resulting executable under
> +terms of your choice, provided that you also meet, for each linked
> +independent module, the terms and conditions of the license of that
> +module. An independent module is a module which is not derived from
> +or based on this library. If you modify this library, you may extend
> +this exception to your version of the library, but you are not
> +obligated to do so. If you do not wish to do so, delete this
> +exception statement from your version.
> + */
> +
> +package net.sourceforge.jnlp.resources;
> +
> +import java.io.IOException;
> +import java.nio.file.Files;
> +import java.nio.file.Path;
> +import java.nio.file.Paths;
> +import java.util.Arrays;
> +import java.util.List;
> +import java.util.Locale;
> +import java.util.ResourceBundle;
> +
> +import net.sourceforge.jnlp.util.OptionParser;
> +
> +public class InfoGenerator {
> + private enum FileNames {
> + PACKAGE("net.sourceforge.jnlp.resources"),
> + MESSAGES("Messages"),
> + TEMPLATE("_template"),
> + NAME("MAN");
> +
> + private String value;
> +
> + private FileNames(String value) {
> + this.value = value;
> + }
> + @Override
> + public String toString() {
> + return this.value;
> + }
> + }
> +
> + private enum ManPages {
> + POLICYEDITOR("policyeditor", "PE"),
> + ITWEB_SETTINGS("itweb-settings", "WS"),
> + ITWEB("itweb", "IT"),
> + JAVAWS("javaws", "JV");
> +
> + private String value;
> + private String abbreviation;
> +
> + private ManPages(String value, String abbreviation) {
> + this.value = value;
> + this.abbreviation = abbreviation;
> + }
> +
> + @Override
> + public String toString() {
> + return this.value;
> + }
> +
> + public String getHelpManual() {
> + //TODO
> + /*
> + * Return localized help/about information
> + * Called from javaws/itweb-settings/policyeditor
> + *
> + */
> + return this.value;
I think that here you can see an pitfall. Once you are not using the jnlpruntime + lang, then you
cannot correctly reuse the code shared between man pages geenrator and help generator.
Less switches, less work. You have a lot of switches :(
> + }
> + public static String getAbbreviation(String name) {
> + for (ManPages page : ManPages.values()) {
> + if (name.equals(page.abbreviation)) {
> + return page.abbreviation;
> + }
> + }
> + return "";
> + }
> + public static boolean contains(String name) {
> + for (ManPages page : ManPages.values()) {
> + if (name.equals(page.value)) {
> + return true;
> + }
> + }
> + return false;
> + }
> + }
Those enums will not be needed once thew lang + (jenlpruntime +) transaltor are used.
If they will remain. please made them as public (separate file) enums.
> +
> + private static final List<String> supportedLocales = Arrays.asList(new String[]
> + {"en", "cs", "de", "pl"});
thsi will prbably also not needed.
> +
> + private static void createManPage(Path templatePath, Path outputPath, String name, String bundleName) throws IOException {
> + createManPage(templatePath, outputPath, Locale.getDefault(), name, bundleName);
> + }
I dont think we wont to create manpages for defautl locale.
> +
> + public static void createManPage(Path templatePath, Path outputPath, Locale locale, String name, String bundleName) throws IOException {
> + String templateContent = new String(Files.readAllBytes(templatePath));
> + String manPageContent = replaceTemplateWithBundleContent(templateContent, locale, name, bundleName);
> +
> + Files.write(outputPath, manPageContent.getBytes());
> + }
As you know - we eont messages only for CURRENT locale O:)
> +
> + private static String replaceTemplateWithBundleContent(String templateContent, Locale locale, String name, String bundleName) throws IOException {
> + ResourceBundle bundle = ResourceBundle.getBundle(bundleName, locale);
> +
> + String abbreviation = ManPages.getAbbreviation(name);
> + if (name == "") {
> + throw new IOException("Name abbreviation not found.");
> + }
> +
> + for (String key : bundle.keySet()) {
> + if (key.startsWith(FileNames.NAME.toString() + abbreviation)) {
> + templateContent = templateContent.replaceAll(key, bundle.getString(key));
> + }
> + }
> + return templateContent;
Is this tempalte already somewhere?
I would stronly discourage you from tempaltes. I would go with some "Markuper class" - with impl for
html and man.
> + }
> +
> + /*
> + * Expected arguments:against not
> + * 0 : language : ex. en
> + * 1 : manual name : ex. policyeditor
> + *
> + *
> + */
> + public static void main(String[] args) {
> +
> + String[] acceptedArgs = {"-lang", "-name"};
> + if (args.length < 4) {
> + System.err.println("Error: Expects at least four arguments. No work has been done.");
> + return;
Quit a lot for such simple case isn't it?
> + }
> +
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> + String[] languages = parser.getOptionValues("-lang");
> + if (languages.length != 1) {
> + System.err.println("Error: Language argument has incorrect format. Example: \"-lang en\"");
> + }
> + String language = languages[0];
> + if (!supportedLocales.contains(language)) {
> + System.err.println("Error: Locale: " + language + " not supported. No work has been done.");
> + return;
> + }
> +
> + String[] names = parser.getOptionValues("-name");
> + if (names.length < 1) {
> + System.err.println("Error: Name argument has incorrect format. Example: \"-name policyeditor javaws\"");
> + }
> + for (String name : names) {
> + if (!ManPages.contains(name)) {
> + System.err.println("Warning: " + name + " not supported. No work has been done for this manual.");
> + } else {
> + try {
> + Locale locale = new Locale(language);
> + Path template = Paths.get(name + FileNames.TEMPLATE);
> + Path output = Paths.get(name + "_" + locale.getLanguage());
> + String bundleName = FileNames.PACKAGE.toString() + "." + FileNames.MESSAGES.toString();
> +
> + createManPage(template, output, locale, name, bundleName);
> + } catch (IOException e) {
> + e.printStackTrace();
> + }
> + }
> + }
> + }
> +
> +}
> diff --git a/netx/net/sourceforge/jnlp/util/OptionParser.java b/netx/net/sourceforge/jnlp/util/OptionParser.java
> new file mode 100644
> --- /dev/null
> +++ b/netx/net/sourceforge/jnlp/util/OptionParser.java
> @@ -0,0 +1,80 @@
> +/*Copyright (C) 2014 Red Hat, Inc.
> +
> +This file is part of IcedTea.
> +
> +IcedTea is free software; you can redistribute it and/or
> +modify it under the terms of the GNU General Public License as published by
> +the Free Software Foundation, version 2.
> +
> +IcedTea 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
> +General Public License for more details.
> +
> +You should have received a copy of the GNU General Public License
> +along with IcedTea; see the file COPYING. If not, write to
> +the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
> +02110-1301 USA.
> +
> +Linking this library statically or dynamically with other modules is
> +making a combined work based on this library. Thus, the terms and
> +conditions of the GNU General Public License cover the whole
> +combination.
> +
> +As a special exception, the copyright holders of this library give you
> +permission to link this library with independent modules to produce an
> +executable, regardless of the license terms of these independent
> +modules, and to copy and distribute the resulting executable under
> +terms of your choice, provided that you also meet, for each linked
> +independent module, the terms and conditions of the license of that
> +module. An independent module is a module which is not derived from
> +or based on this library. If you modify this library, you may extend
> +this exception to your version of the library, but you are not
> +obligated to do so. If you do not wish to do so, delete this
> +exception statement from your version.
> + */
> +
> +package net.sourceforge.jnlp.util;
> +
> +import java.util.ArrayList;
> +import java.util.Arrays;
> +import java.util.List;
> +
> +//Parses options (generally from main(String[] args)
> +public class OptionParser {
> + private String[] args;
> + private List<String> acceptedArgs;
> +
> + public OptionParser(String[] args, String[] acceptedArgs) {
> + this.args = args;
> + this.acceptedArgs = new ArrayList<String>(Arrays.asList(acceptedArgs));
> + }
> +
> + /**
> + * Return all the values of the specified option, or an empty
> + * array if the option is not present. If the option is a
> + * is present with no parameters then the option name is
> + * returned once.
> + */
> + public String[] getOptionValues(String option) {
> + List<String> result = new ArrayList<String>();
> + int i = 0;
> + while(i < args.length) {
> + if (acceptedArgs.contains(args[i]) && option.equals(args[i])) {
> + i++;
> + while(i < args.length && !acceptedArgs.contains(args[i])) {
> + result.add(args[i]);
> + i++;
> + }
> + if (result.size() == 0) {
> + result.add(option);
> + }
> + } else {
> + i++;
> + }
> + }
> +
> + return result.toArray(new String[result.size()]);
> + }
> +
hm :) We already ahve this for Boot class :) If possible. Please reuse.
> +}
> diff --git a/tests/netx/unit/net/sourceforge/jnlp/resources/InfoGeneratorTest.java b/tests/netx/unit/net/sourceforge/jnlp/resources/InfoGeneratorTest.java
> new file mode 100644
> --- /dev/null
> +++ b/tests/netx/unit/net/sourceforge/jnlp/resources/InfoGeneratorTest.java
> @@ -0,0 +1,148 @@
> +/*Copyright (C) 2014 Red Hat, Inc.
> +
> +This file is part of IcedTea.
> +
> +IcedTea is free software; you can redistribute it and/or
> +modify it under the terms of the GNU General Public License as published by
> +the Free Software Foundation, version 2.
> +
> +IcedTea 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
> +General Public License for more details.
> +
> +You should have received a copy of the GNU General Public License
> +along with IcedTea; see the file COPYING. If not, write to
> +the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
> +02110-1301 USA.
> +
> +Linking this library statically or dynamically with other modules is
> +making a combined work based on this library. Thus, the terms and
> +conditions of the GNU General Public License cover the whole
> +combination.
> +
> +As a special exception, the copyright holders of this library give you
> +permission to link this library with independent modules to produce an
> +executable, regardless of the license terms of these independent
> +modules, and to copy and distribute the resulting executable under
> +terms of your choice, provided that you also meet, for each linked
> +independent module, the terms and conditions of the license of that
> +module. An independent module is a module which is not derived from
> +or based on this library. If you modify this library, you may extend
> +this exception to your version of the library, but you are not
> +obligated to do so. If you do not wish to do so, delete this
> +exception statement from your version.
> + */
> +
> +package net.sourceforge.jnlp.resources;
> +
> +import static org.junit.Assert.assertEquals;
> +import static org.junit.Assert.assertTrue;
> +
> +import java.io.File;
> +import java.io.FileOutputStream;
> +import java.io.IOException;
> +import java.io.PrintStream;
> +import java.nio.file.Files;
> +import java.nio.file.Path;
> +import java.nio.file.Paths;
> +import java.util.Locale;
> +
> +import org.junit.Before;
> +import org.junit.Test;
> +
> +public class InfoGeneratorTest {
> + private FileOutputStream faos;
> + private String bundleName = "net.sourceforge.jnlp.resources.testbundle";
> +
> + @Before
> + public void setup() throws IOException {
> + faos = new FileOutputStream("test");
> + PrintStream tempErr = new PrintStream(faos);
> + System.setErr(tempErr);
> +
> + }
> + private String runMainAndReadFromErr(String[] args) throws IOException {
> + InfoGenerator.main(args);
> + File fios = new File("test");
> +
> + String err = new String(Files.readAllBytes(fios.toPath()));
> +
> + return err;
> +
> + }
> +
> + @Test
> + public void testIncorrectArgs() throws IOException {
> + String[] args = {};
> +
> + String expected = "Error: Expects at least four arguments. No work has been done.";
> + String actual = runMainAndReadFromErr(args);;
> +
> + assertTrue(actual.contains(expected));
> + }
> +
> + @Test
> + public void testUnsupportedLanguage() throws IOException {
> + String language = "blob";
> + String name = "policyeditor";
> + String[] args = {"-lang", language, "-name", name};
> +
> + String expected = "Error: Locale: " + language + " not supported. No work has been done.";
> + String actual = runMainAndReadFromErr(args);;
> +
> + assertTrue(actual.contains(expected));
> + }
> +
> + @Test
> + public void testUnsupportedManual() throws IOException{
> + String language = "en";
> + String name = "blob";
> + String[] args = {"-lang", language, "-name", name};
> +
> + String expected = "Warning: " + name + " not supported. No work has been done for this manual.";
> + String actual = runMainAndReadFromErr(args);
> +
> + assertTrue(actual.contains(expected));
> + }
> +
> + @Test
> + public void testGenerateEnManPage() throws IOException {
> + String language = "en";
> + String name = "policyeditor";
> +
> +
> + String template = "MANPEA\nMANPEB";
> + String expected = "halp\nmeow";
> +
> + String out = generateManPage(name, template, language, bundleName);
> +
> + assertEquals(expected, out);
> + }
> +
> + @Test
> + public void testGeneratePlManPage() throws IOException {
> + String language = "pl";
> + String name = "policyeditor";
> +
> +
> + String template = "MANPEA\nMANPEB";
> + String expected = "blob\now";
> +
> + String out = generateManPage(name, template, language, bundleName);
> + assertEquals(expected, out);
> + }
> +
> + private String generateManPage(String name, String template, String language, String bundleName) throws IOException {
> + Path templatePath = Paths.get(name + "_" + "template");
> + Files.write(templatePath, template.getBytes());
> +
> + Path outputPath = Paths.get(name + "_" + language);
> +
> + Locale locale = new Locale(language);
> + InfoGenerator.createManPage(templatePath, outputPath, locale, name, bundleName);
> +
> + String out = new String(Files.readAllBytes(outputPath));
> + return out;
> + }
> +}
> diff --git a/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle.properties b/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle.properties
> new file mode 100644
> --- /dev/null
> +++ b/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle.properties
> @@ -0,0 +1,2 @@
> +MANPEA=halp
> +MANPEB=meow
> \ No newline at end of file
> diff --git a/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle_pl.properties b/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle_pl.properties
> new file mode 100644
> --- /dev/null
> +++ b/tests/netx/unit/net/sourceforge/jnlp/resources/testbundle_pl.properties
> @@ -0,0 +1,2 @@
> +MANPEA=blob
> +MANPEB=ow
> \ No newline at end of file
> diff --git a/tests/netx/unit/net/sourceforge/jnlp/util/OptionParserTest.java b/tests/netx/unit/net/sourceforge/jnlp/util/OptionParserTest.java
> new file mode 100644
> --- /dev/null
> +++ b/tests/netx/unit/net/sourceforge/jnlp/util/OptionParserTest.java
> @@ -0,0 +1,127 @@
> +/*Copyright (C) 2014 Red Hat, Inc.
> +
> +This file is part of IcedTea.
> +
> +IcedTea is free software; you can redistribute it and/or
> +modify it under the terms of the GNU General Public License as published by
> +the Free Software Foundation, version 2.
> +
> +IcedTea 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
> +General Public License for more details.
> +
> +You should have received a copy of the GNU General Public License
> +along with IcedTea; see the file COPYING. If not, write to
> +the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
> +02110-1301 USA.
> +
> +Linking this library statically or dynamically with other modules is
> +making a combined work based on this library. Thus, the terms and
> +conditions of the GNU General Public License cover the whole
> +combination.
> +
> +As a special exception, the copyright holders of this library give you
> +permission to link this library with independent modules to produce an
> +executable, regardless of the license terms of these independent
> +modules, and to copy and distribute the resulting executable under
> +terms of your choice, provided that you also meet, for each linked
> +independent module, the terms and conditions of the license of that
> +module. An independent module is a module which is not derived from
> +or based on this library. If you modify this library, you may extend
> +this exception to your version of the library, but you are not
> +obligated to do so. If you do not wish to do so, delete this
> +exception statement from your version.
> + */
> +
> +package net.sourceforge.jnlp.util;
> +
> +import static org.junit.Assert.assertEquals;
> +
> +import org.junit.Test;
> +
> +public class OptionParserTest {
> +
> + @Test
> + public void testGetSingleOptionValue() {
> + String[] args = {"-basedir", "blob"};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals("blob", values[0]);
> + assertEquals(1, values.length);
> + }
> +
> + @Test
> + public void testGetSingleOptionMultipleValues() {
> + String[] args = {"-basedir", "blob", "meow"};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals("blob", values[0]);
> + assertEquals("meow", values[1]);
> + assertEquals(2, values.length);
> + }
> +
> + @Test
> + public void testGetDifferentOptionValues() {
> + String[] args = {"-basedir", "blob", "-arg", "help"};
> + String[] acceptedArgs = {"-basedir", "-arg"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals("blob", values[0]);
> + assertEquals(1, values.length);
> +
> + values = parser.getOptionValues("-arg");
> + assertEquals("help", values[0]);
> + assertEquals(1, values.length);
> +
> + }
> +
> + @Test
> + public void testUnsupportedOptionValue() {
> + String[] args = {"-basedir", "blob"};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-unsupported");
> + assertEquals(0, values.length);
> + }
> +
> + @Test
> + public void testSupportedOptionValueWithNoUse() {
> + String[] args = {};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals(0, values.length);
> + }
> +
> + @Test
> + public void testOptionValueWithNoArgument() {
> + String[] args = {"-basedir"};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals("-basedir", values[0]);
> + assertEquals(1, values.length);
> + }
> +
> + @Test
> + public void testMultipleOptionsMultipleValues() {
> + String[] args = {"-basedir", "poke", "blob", "-basedir", "meep"};
> + String[] acceptedArgs = {"-basedir"};
> + OptionParser parser = new OptionParser(args, acceptedArgs);
> +
> + String[] values = parser.getOptionValues("-basedir");
> + assertEquals(3, values.length);
> + assertEquals("poke", values[0]);
> + assertEquals("blob", values[1]);
> + assertEquals("meep", values[2]);
> + }
> +}
>
Thank you for the tests. Please don't give up on them when you will rework most of this. Tests are
good for gathering ideas around actual coding.
J.
More information about the distro-pkg-dev
mailing list