All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.sfj.DumbCLIParse Maven / Gradle / Ivy

/*
 * Copyright 2020 C. Schanck
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.sfj;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * 

This is in honor of a past boss of mine, a crackerjack dev, who struggled * (one time! but sadly on a projected screen... :-) ) with regular expressions. * Here we have a tiny but useful command line arg parser based entirely on regular * expressions. This one is for you, Louis. * *

The way you use this is by first call {@link #args} with the main() args. * Just gives you a list back, but feels nice for examples. * *

There are two ways to use it. You can use the per arg parsing methods, * {@link #scanForArgWithParm(List, String)} or {@link #scanForFlag(List, String, boolean)} * passing the arg list in. Each of the scan methods will scan the entire list, * removing the first match and returning appropriately. * *

When you are done scanning for args, the remainder args are whatever is in * the list. * *

Alternatively, you can pass the list to {link {@link #scanForAllFlags(List)} and * then {@link #scanForAllParamArgs(List)} in turn, and each will return a list of * matching flag or string args. * *

Personally, I find the incremental ones easier and more natural to code * against, but some may prefer the "all" variants. See unit test for sample usage. * *

No one will mistake it for JCommander or any of the other full up CLI parsing * packages, but very sufficient for rapid prototyping. * * @author cschanck **/ public final class DumbCLIParse { private DumbCLIParse() { } /** * Just turn main() args into a mutable list. Nothing exotic. * * @param args array of args * @return args as mutable list. */ public static List args(String[] args) { return new LinkedList<>(Arrays.asList(args)); } private static String argWithParamPattern(String argname) { return "--?" + argname + "=(.+)$"; } private static String argWithParamPattern() { return "--?([^=]+)=(.+)$"; } private static String flagPattern(String argname) { return "--?((not)-(-)?)?" + argname + "$"; } private static String flagPattern() { return "--?((not)-(-)?)?([^=]+)$"; } private static Matcher match(Pattern pattern, String s) { Matcher match = pattern.matcher(s.trim()); if (match.matches()) { return match; } return null; } /** * Scan for the first argument with a param name that matches the argname. * Scanning for arg "foo" looks for either of "--foo=thing" or "-foo=thing". * Returns the arg as a string, further parsing is your problem. Note that if * you want a default value, since it returns an optional, you can just do * ret.orElse("default value") * * @param remaining list of remaining args to parse from * @param argname name of arg, case sensitive * @return Optional string value if it matched. */ public static Optional scanForArgWithParm(List remaining, String argname) { Pattern pattern = Pattern.compile(argWithParamPattern(argname)); return new ArrayList<>(remaining).stream().map(s -> { Matcher ret = match(pattern, s); if (ret != null) { remaining.remove(s); } return ret; }).filter(Objects::nonNull).limit(1).reduce((first, second) -> second).map((m) -> m.group(1)); } /** * Scan for the first argument with a param name that matches the argname. * Scanning for arg "foo" looks for any of "--foo", "--not--foo", "--not-foo", * "-foo", "-not--foo", "-not-foo". * * @param remaining list of remaining args to scan through * @param argname argname to scan for, case sensitive * @param foundVal value to use if it is found * @return If any of the variants without "not" are found, * the specific foundVal will be returned. If any of the "not" variants are found, * or if the flag is not seen, !foundVal will be returned. */ public static boolean scanForFlag(List remaining, String argname, boolean foundVal) { // match --not--arg, or single dash versions // case sensitive. Pattern pattern = Pattern.compile(flagPattern(argname)); Optional got = new ArrayList<>(remaining).stream().map((s) -> { Matcher ret = match(pattern, s); if (ret != null) { remaining.remove(s); } return ret; }).filter(Objects::nonNull).limit(1).reduce((first, second) -> second); if (got.isPresent()) { String not = got.get().group(2); if (not == null) { return foundVal; } else { return !foundVal; } } return !foundVal; } static class FlagArg { private final String arg; private final boolean isNot; FlagArg(String arg, boolean isNot) { this.arg = arg; this.isNot = isNot; } public String getArg() { return arg; } public boolean isNot() { return isNot; } @Override public String toString() { return "FlagArg{" + (isNot ? "not " : "") + arg + '}'; } } static class StringArg { private final String arg; private final String param; StringArg(String arg, String param) { this.arg = arg; this.param = param; } public String getArg() { return arg; } public String getParam() { return param; } @Override public String toString() { return "StringArg{" + arg + '=' + param + '}'; } } public static List scanForAllFlags(List all) { LinkedList found = new LinkedList<>(); Pattern patt = Pattern.compile(flagPattern()); new ArrayList<>(all).forEach((s) -> { Matcher m = match(patt, s.trim()); if (m != null) { String not = m.group(2); String arg = m.group(4); found.add(new FlagArg(arg, not != null)); all.remove(s); } }); return found; } public static List scanForAllParamArgs(List all) { LinkedList found = new LinkedList<>(); Pattern patt = Pattern.compile(argWithParamPattern()); new ArrayList<>(all).forEach((s) -> { Matcher m = match(patt, s.trim()); if (m != null) { String arg = m.group(1); String parm = m.group(2); found.add(new StringArg(arg, parm)); all.remove(s); } }); return found; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy