[rfc][icedtea-web] use Option Parser in itweb-settings
Lukasz Dracz
ldracz at redhat.com
Tue Dec 16 19:06:17 UTC 2014
Hello,
Here is the updated patch
>Yup! Realy cool work. I have only small nits, and the bigest ones I'm not sure how to deal
>
>minor nit (little bit unrelated, but should be included in patch):
>
>our docs says this:
>DESCRIPTION
> -check name - Checks that the current settings have valid values.(No argument expected)
> -get name - Shows the value of the named setting.(Expected one or more arguments)
> -headless - Disables download window, other UIs.(No argument expected)
> -help - Prints out information about supported command and basic usage.(No argument
>expected)
> -info name - Shows additional information about the named setting. Includes a description,
>the current value, the possible values, and the source of the setting.(Expected one or more arguments)
> -list - Shows a list of all settings.(No argument expected)
> -reset name - Resets the named setting to its original value.(Expected one or more arguments)
> -reset all - Resets all settings to their original values.(No argument expected)
> -set name value - Sets the setting to the new value, after checking that it is an appropriate
>value.(Expected even number of arguments with param=value as valid argument)
>
>You are affecting those, so try to align them with your real impl.
I need some clarification here, what do you want me to align or how ? Do you want me to edit the doc descriptions to be more accurate to how the real impl works or make the impl work more like how it is described ?
I have edited the message properties a bit.
>main issues are - described functionality is really excellent, but the impl have few flaws:
>
>[jvanek at jvanek bin]$ ./itweb-settings check dsf sd
>No issues found.
>
> - sounds strnage doesnt it? But I'mnot sure if worthy to fix...
That is strange indeed and I think worthy of a fix so now it will register dsf sd as unknown commands
>[jvanek at jvanek bin]$ ./itweb-settings get dsf sd
>Unknown property-name "dsf"
>[jvanek at jvanek bin]$ ./itweb-settings get deployment.user.security.policy
>deployment.user.security.policy: file:///home/jvanek/.config/icedtea-web/security/java.policy
>[jvanek at jvanek bin]$ ./itweb-settings get deployment.user.security.policy dsf sfd
>deployment.user.security.policy: file:///home/jvanek/.config/icedtea-web/security/java.policy
>Unknown property-name "dsf"
>
> - so it dies with first unrecognized property... Not sure what to do here, wheter to keep it as
>you have it, or invest time and dye iumidiately (not worthy probably) , or after Unknown
>property-name just continue with other one... Whether the first is definitly the most correct, not
>sure if it is worthy....
I think it should die immediatedly and inform the user of all the unrecognized properties. I have changed it to
this implementation.
>
>on contrary:
>[jvanek at jvanek bin]$ ./itweb-settings set dsf sfd
>WARNING: Unknown property name "dsf" - creating new property
>[jvanek at jvanek bin]$ ./itweb-settings set dsf sfd fdwf sfg
>Property "dsf" is unknown.
>WARNING: Unknown property name "fdwf" - creating new property
>[jvanek at jvanek bin]$ ./itweb-settings set dsf sfd fdw
>net.sourceforge.jnlp.util.optionparser.UnevenParameterException: For option -set expected an even
>number of params.
> at net.sourceforge.jnlp.util.optionparser.OptionParser.checkOptionHasEvenNumber(OptionParser.java:129)
>....
>
>- this loks correct.But the previous issue may be more spread...
I agree set works
>--details
>
>well this is strange. On one side, it do not seem to work. On second, it seems to be undefined n
>general lsit of swithces, and so *undocumented*.
>
>Unless I missed some functionality, I'm for to drop it. (but if it have some, please tell before
>removing
It does have functionality but for me atleast all it does is add Description: <Unknown> at the moment, I guess
we either need to add descriptions which could be useful or just drop --details if all its going to show is Unknown.
I think I am for having descriptions, What are your thoughts ?
I have updated it so that --details works.
>> diff --git a/netx/net/sourceforge/jnlp/OptionsDefinitions.java b/netx/net/sourceforge/jnlp/OptionsDefinitions.java
>> --- a/netx/net/sourceforge/jnlp/OptionsDefinitions.java
>> +++ b/netx/net/sourceforge/jnlp/OptionsDefinitions.java
>> @@ -76,7 +76,7 @@
>> LIST("-list", "IBOList"),
>> GET("-get", "name", "IBOGet", NumberOfArguments.ONE_OR_MORE),
>> INFO("-info", "name", "IBOInfo", NumberOfArguments.ONE_OR_MORE),
>> - SET("-set", "name value", "IBOSet", NumberOfArguments.ONE_OR_MORE),
>> + SET("-set", "name value", "IBOSet", NumberOfArguments.EVEN_NUMBER_SUPPORTS_EQUALS_CHAR),
>
>
>pelase fix the messages properties to be accurate.
I edited message properties to be more accurate by making them plural where it applied. Should they be changed more ?
>
>> RESETALL("-reset", "all", "IBOResetAll"),
>> RESET("-reset", "name", "IBOReset", NumberOfArguments.ONE_OR_MORE),
>> /**
>> * Handles the 'list' command
>> *
>> - * @param args the arguments to the list command
>> * @return result of handling the command. SUCCESS if no errors occurred.
>> */
>> - public int handleListCommand(List<String> args) {
>
>The advantage of this declaration ^ is that it is more t
>
>You can always have:
>public int handleListCommand() {
>handleListCommand(optionParser)
>}
>public int handleListCommand(OptionParser op) {...
>
>But does nto meter, its up to you.
I feel like you didn't finish typing out what you wanted to say or deleted a line by mistake, I am not sure what
advantage you are referring to ? What does more t mean ?
>> - if (args.contains("help")) {
>> + public int handleListCommand() {
>> + if (optionParser.hasOption(OptionsDefinitions.OPTIONS.HELP)) {
>> printListHelp();
>> return SUCCESS;
>> }
>>
>> boolean verbose = false;
>>
>> - if (args.contains("--details")) {
>> + if (optionParser.getMainArgs().contains("--details")) {
>
>
>hmhhm.. seesm nt to be working see top of email.
Good catch. Fixed.
>(update 10 minutes later
>
>Well, aybe it may work, if istead of this --details, the itwebsettings recognize general verbose
>switch... what do you think?
As in use "-verbose" from OptionsDefinitions instead ? I suppose it would make itweb-settings and javaws seem more together and common but it is only used with list and personally I like the --details more. I can do it with "-verbose" if that is wanted.
>> verbose = true;
>> - args.remove("--details");
>....
>> - old.getValidator().validate(value);
>> - } catch (IllegalArgumentException e) {
>> - OutputController.getLogger().log(OutputController.Level.WARNING_ALL, R("CLIncorrectValue", old.getName(), value, old.getValidator().getPossibleValues()));
>> - OutputController.getLogger().log(e);
>> - return ERROR;
>> + for (String arg : args) {
>> + if (args.indexOf(arg) % 2 == 0) {
>
>ok, I dont understand this. What this 17 lines should do? Can ti be more readable? More "use
>parser" like? Or at least commented?
Hmm your right, its not obvious what it does at first glance.
I have changed it to hide some of the details in private methods.
Does this make it better or is more work needed?
>> +
>> + key = arg;
>> + } else {
>> +
>> + value = arg;
>> + if (config.getRaw().containsKey(key)) {
>> + Setting<String> old = config.getRaw().get(key);
>> + if (old.getValidator() != null) {
>> + try {
>> + old.getValidator().validate(value);
>> + } catch (IllegalArgumentException e) {
>> + OutputController.getLogger().log(OutputController.Level.WARNING_ALL, R("CLIncorrectValue", old.getName(), value, old.getValidator().getPossibleValues()));
>> + OutputController.getLogger().log(e);
>> + return ERROR;
>
>yes.. return or not return? Handle in parser? See lower...
>
>
>
>> + }
>> + }
>> +
>> + config.setProperty(key, value);
>> + } else {
>> + OutputController.getLogger().printOutLn(R("CLWarningUnknownProperty", key));
>> + config.setProperty(key, value);
>> }
>> }
>> - config.setProperty(key, value);
>> - } else {
>> - OutputController.getLogger().printOutLn(R("CLWarningUnknownProperty", key));
>> - config.setProperty(key, value);
>> }
>>
>.. big snip!
>> -
>> + public int handle() {
>> int val;
>> - if (command.equals(OptionsDefinitions.OPTIONS.HELP.option)) {
>> + if (getNumberOfOptions() > 1) {
>> + OutputController.getLogger().log(OutputController.Level.MESSAGE_ALL, R("CLUnexpectedNumberOfCommands"));
>> val = handleHelpCommand();
>
>I think those exceptions may be handeld in parser, but i dont insists (as help may have different
>meaning in javaws / itwebsettings/policy editor
>
>But if it does so, then our documentation will be always wrong... Maybe to have two different
>"help" declarations? I think it is possible to do so. One willbe used in javaws/polici editor and
>second (HELP1, help, info1, no argument)in itweb-settings (HELP2, help, info2, no or more args)
>
>Thougths?
More clarification here would be helpful, I don't think I have a full understanding of what you are referring to or want. I am fairly sure two different helps is possible approach. As to the the exceptions which ones are you referring to ? CLUnexpectedNumberOfCommands should be an exception / have a throw exception there ? Or what exception are you referring to that you want/could be handled in the parser.
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.LIST.option)) {
>> - val = handleListCommand(arguments);
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.SET.option)) {
>> - val = handleSetCommand(arguments);
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.RESET.option)) {
>> - val = handleResetCommand(arguments);
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.GET.option)) {
>> - val = handleGetCommand(arguments);
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.INFO.option)) {
>> - val = handleInfoCommand(arguments);
>> - } else if (command.equals(OptionsDefinitions.OPTIONS.CHECK.option)) {
>> - val = handleCheckCommand(arguments);
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.LIST)) {
>> + val = handleListCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.SET)) {
>> + val = handleSetCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.RESET)) {
>> + val = handleResetCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.GET)) {
>> + val = handleGetCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.INFO)) {
>> + val = handleInfoCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.CHECK)) {
>> + val = handleCheckCommand();
>> + } else if (optionParser.hasOption(OptionsDefinitions.OPTIONS.HELP)) {
>> + val = handleHelpCommand();
>> } else {
>> - OutputController.getLogger().log(OutputController.Level.MESSAGE_ALL, R("CLUnknownCommand", command));
>> + for (String unknown : optionParser.getMainArgs()) {
>> + OutputController.getLogger().log(OutputController.Level.MESSAGE_ALL, R("CLUnknownCommand", unknown));
>> + }
>> handleHelpCommand();
>> val = ERROR;
>> }
>> -
>> return val;
>> }
>>
>> + private int getNumberOfOptions() {
>> + int number = optionParser.getNumberOfOptions();
>> + if (optionParser.hasOption(OptionsDefinitions.OPTIONS.HELP)) {
>> + number--;
>> + }
>> + if (optionParser.hasOption(OptionsDefinitions.OPTIONS.HEADLESS)) {
>> + number--;
>> + }
>> + return number;
>> + }
>> /**
>> * The starting point of the program
>> * @param args the command line arguments to this program
>> */
>> public static void main(String[] args) throws Exception {
>> - DeploymentConfiguration.move14AndOlderFilesTo15StructureCatched();
>> - if (args.length == 0) {
>> - ControlPanel.main(new String[] {});
>> - } else {
>> - CommandLine cli = new CommandLine();
>> - int result = cli.handle(args);
>> + try {
>> + optionParser = new OptionParser(args, OptionsDefinitions.getItwsettingsCommands());
>>
>> - // instead of returning, use JNLPRuntime.exit() so we can pass back
>> - // error codes indicating success or failure. Otherwise using
>> - // this program for scripting will become much more challenging
>> - JNLPRuntime.exit(result);
>> + if (optionParser.hasOption(OptionsDefinitions.OPTIONS.HEADLESS)) {
>> + JNLPRuntime.setHeadless(true);
>> + }
>> + DeploymentConfiguration.move14AndOlderFilesTo15StructureCatched();
>
>Not sure If I got this. It was missing here?? Crap.. It would be worthy for 1.5 too
Ah okay, do you want me to backport the headless change to 1.5 ?
>> + if (args.length == 0) {
>> + ControlPanel.main(new String[] {});
>> + } else {
>> + CommandLine cli = new CommandLine();
>> + int result = cli.handle();
>> +
>> + // instead of returning, use JNLPRuntime.exit() so we can pass back
>> + // error codes indicating success or failure. Otherwise using
>> + // this program for scripting will become much more challenging
>> + JNLPRuntime.exit(result);
>> + }
>> + } catch (Exception e) {
>> + OutputController.getLogger().log(OutputController.Level.WARNING_ALL, e);
>> + JNLPRuntime.exit(1);
>> }
>> }
>> }
>> diff --git a/netx/net/sourceforge/jnlp/resources/Messages.properties b/netx/net/sourceforge/jnlp/resources/Messages.properties
>> --- a/netx/net/sourceforge/jnlp/resources/Messages.properties
>> +++ b/netx/net/sourceforge/jnlp/resources/Messages.properties
>> @@ -338,6 +338,9 @@
>> be added and selected. Multiple URLs may also be given with a single -codebase flag by separating them with spaces. In this case, the last codebase given will be selected, and all will be \
>> added. If this flag is given more than once, only the first is used.
>>
>> +# Option Parser
>> +OPUnevenParams=For option {0} expected an even number of params.
>> +
>> # NumberOfArguments descriptions.
>> NOAnone=No argument expected
>> NOAone=Exactly one argument expected
>> @@ -909,6 +912,7 @@
>> CLResetDescription=Resets the value for property-name to it\'s default value.\nall resets all properties recognized by IcedTea-Web to their default value.
>> CLInfoDescription=Shows more information about the given property
>> CLCheckDescription=Shows any properties that have been defined but are not recognized by IcedTea-Web
>> +CLUnexpectedNumberOfCommands=Itweb-settings can only run one command at a time.
>>
>> # splash screen related
>> SPLASHerror = Click here for details. An exception has occurred.
>> diff --git a/netx/net/sourceforge/jnlp/runtime/Boot.java b/netx/net/sourceforge/jnlp/runtime/Boot.java
>> --- a/netx/net/sourceforge/jnlp/runtime/Boot.java
>> +++ b/netx/net/sourceforge/jnlp/runtime/Boot.java
>> @@ -46,6 +46,7 @@
>> import net.sourceforge.jnlp.util.logging.OutputController;
>> import net.sourceforge.jnlp.util.optionparser.InvalidArgumentException;
>> import net.sourceforge.jnlp.util.optionparser.OptionParser;
>> +import net.sourceforge.jnlp.util.optionparser.UnevenParameterException;
>> import sun.awt.AppContext;
>> import sun.awt.SunToolkit;
>>
>> @@ -96,7 +97,12 @@
>> * Launch the JNLP file specified by the command-line arguments.
>> */
>> public static void main(String[] argsIn) {
>> - optionParser = new OptionParser(argsIn, OptionsDefinitions.getJavaWsOptions());
>> + try {
>> + optionParser = new OptionParser(argsIn, OptionsDefinitions.getJavaWsOptions());
>> + } catch (UnevenParameterException upe) {
>> + OutputController.getLogger().log(OutputController.Level.WARNING_ALL, upe);
>> + JNLPRuntime.exit(1);
>
>Wby exit? Isnt rethrow better? Why rethrow at all.. this exception may continue up and up until
>uttermost top end death, or not?
Yeah, I wasn't too sure what would be best, I think making the exception continue is probably best.
>> + }
>>
>> if (optionParser.hasOption(OptionsDefinitions.OPTIONS.VERBOSE)) {
>> JNLPRuntime.setDebug(true);
>> diff --git a/netx/net/sourceforge/jnlp/util/optionparser/OptionParser.java b/netx/net/sourceforge/jnlp/util/optionparser/OptionParser.java
>> --- a/netx/net/sourceforge/jnlp/util/optionparser/OptionParser.java
>> +++ b/netx/net/sourceforge/jnlp/util/optionparser/OptionParser.java
>> @@ -41,6 +41,9 @@
>> import java.util.List;
>>
>> import net.sourceforge.jnlp.OptionsDefinitions;
>> +import net.sourceforge.jnlp.util.logging.OutputController;
>> +
>> +import static net.sourceforge.jnlp.runtime.Translator.R;
>>
>> /** Parses for options (passed in as arguments)
>> * To use add entries to OPTIONS enum in OptionsDefinitions
>> @@ -54,13 +57,18 @@
>> private final List<OptionsDefinitions.OPTIONS> possibleOptions;
>> //List of all possible main arguments
>> private final List<String> mainArgumentList = new ArrayList<>();
>> + private boolean evenNumberFound;
>>
>> - public OptionParser(String[] args, List<OptionsDefinitions.OPTIONS> options) {
>> + public OptionParser(String[] args, List<OptionsDefinitions.OPTIONS> options) throws UnevenParameterException {
>> this.args = Arrays.copyOf(args, args.length);
>> this.possibleOptions = options;
>>
>> + evenNumberFound = false;
>> parsedOptions = new ArrayList<>();
>> parseContents();
>> + if (evenNumberFound) {
>> + checkOptionHasEvenNumber();
>> + }
>>
>> }
>>
>> @@ -71,12 +79,23 @@
>> lastOption = addOptionToList(arg);
>> } else if (shouldAddParam(lastOption)) {
>> lastOption.addParam(arg);
>> + } else if (isEvenNumberSupportingEqualsChars(lastOption)) {
>> + evenNumberFound = true;
>> + handleEvenNumberSupportingEqualsChar(lastOption, arg);
>> } else {
>> mainArgumentList.add(arg);
>> }
>> }
>> }
>>
>> + private void handleEvenNumberSupportingEqualsChar(final ParsedOption lastOption, String arg) {
>> + if (arg.contains("=")) {
>> + lastOption.addParam(arg.split("=")[0]);
>> + lastOption.addParam(arg.split("=")[1]);
>> + } else {
>> + lastOption.addParam(arg);
>> + }
>> + }
>> private boolean shouldAddParam(final ParsedOption lastOption) {
>> return lastOption != null &&
>> (oneOrMoreArguments(lastOption) || isOneArgumentNotFull(lastOption));
>> @@ -90,6 +109,10 @@
>> return lastOption.getOption().hasOneOrMoreArguments();
>> }
>>
>> + private boolean isEvenNumberSupportingEqualsChars(final ParsedOption lastOption) {
>> + return lastOption.getOption().hasEvenNumberSupportingEqualsChar();
>> + }
>> +
>> private ParsedOption addOptionToList(final String arg) {
>> ParsedOption option = new ParsedOption(argumentToOption(arg));
>> if (arg.contains("=")) {
>> @@ -99,6 +122,16 @@
>> return option;
>> }
>>
>> + private void checkOptionHasEvenNumber() throws UnevenParameterException {
>> + for (ParsedOption option : parsedOptions) {
>> + if (isEvenNumberSupportingEqualsChars(option)) {
>> + if (option.getParams().size() % 2 != 0){
>> + throw new UnevenParameterException(R("OPUnevenParams", option.getOption().option));
>> + }
>> + }
>> + }
>> + }
>> +
>> private OptionsDefinitions.OPTIONS argumentToOption(final String arg) {
>> for (OptionsDefinitions.OPTIONS opt : possibleOptions) {
>> if (stringEqualsOption(arg, opt)) {
>> @@ -171,4 +204,8 @@
>> }
>> return result;
>> }
>> +
>> + public int getNumberOfOptions() {
>> + return parsedOptions.size();
>> + }
>
>
>
>Some tests of those new methods would be good.
Good Idea.
Added 4 new tests.
>> }
>> diff --git a/netx/net/sourceforge/jnlp/util/optionparser/UnevenParameterException.java b/netx/net/sourceforge/jnlp/util/optionparser/UnevenParameterException.java
>> new file mode 100644
>> --- /dev/null
>> +++ b/netx/net/sourceforge/jnlp/util/optionparser/UnevenParameterException.java
>> @@ -0,0 +1,43 @@
>> +/*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.optionparser;
>> +
>> +public class UnevenParameterException extends Exception {
>> + public UnevenParameterException(String argument) {
>> + super(argument);
>> + }
>> +}
>> \ No newline at end of file
>>
>
>
>Nioce job! Looking forward to have this in!
Thanks for the the review !
>
>J.
Thank you,
Lukasz Dracz
-------------- next part --------------
A non-text attachment was scrubbed...
Name: itweb-settings-OptionParser-4.patch
Type: text/x-patch
Size: 42611 bytes
Desc: not available
URL: <http://mail.openjdk.java.net/pipermail/distro-pkg-dev/attachments/20141216/666b303f/itweb-settings-OptionParser-4-0001.patch>
More information about the distro-pkg-dev
mailing list