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

net.morimekta.console.util.Parser Maven / Gradle / Ivy

Go to download

Utilities helping with various *nix console topics. Mostly geared toward expressive and interactive command line applications.

There is a newer version: 3.1.1
Show newest version
/*
 * Copyright (c) 2016, Stein Eldar Johnsen
 *
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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 net.morimekta.console.util;

import net.morimekta.console.args.Argument;
import net.morimekta.console.args.ArgumentException;
import net.morimekta.console.args.Option;
import net.morimekta.console.args.Property;

import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.function.Consumer;

/**
 * Value parser interface. It converts from a string value (usually the
 * CLI argument value) and converts it into a specific value type. Essentially
 * this interface is meant to bridge two {@link Consumer}s, the target consumer
 * (usually the value setter or adder), and the string consumer that the
 * {@link Option} or {@link Argument} needs.
 * 

* E.g., the following code will create an option '--timestamp' that parses the * argument value as a long (i64) and calls bean.setTimestamp(long) * with that value: * * new Option("--timestamp", null, "The timestamp", i64(bean::setTimestamp)); * */ @FunctionalInterface public interface Parser { @FunctionalInterface interface TypedPutter { /** * Put a typed value * @param key The property key. * @param value The property value. */ void put(String key, T value); } /** * Parse the value into a typed instance. * * @param value The string value. * @return The typed instance. * @throws ArgumentException If the parsing failed. */ T parse(String value); /** * Make a string consumer to typed value consumer out of the converter. * * @param consumer The consumer to wrap. * @return The string consumer. */ default Consumer andApply(Consumer consumer) { return s -> consumer.accept(parse(s)); } /** * Make a property putter that calls a typed putter with the parsed value. * * @param putter The typed putter. * @return The property putter. */ default Property.Putter andPut(TypedPutter putter) { return (k, v) -> putter.put(k, parse(v)); } /** * Make a consumer that puts a specific value with the typed putter. * * @param putter the typed putter. * @param key The property key. * @return The string consumer. */ default Consumer andPutAs(TypedPutter putter, String key) { return s -> putter.put(key, parse(s)); } /** * Convenience method to put a specific value into a putter. * * @param putter The putter. * @param key The key to put. * @return The string consumer. */ static Consumer putAs(Property.Putter putter, String key) { return s -> putter.put(key, s); } /** * Make a 32-bit integer parser. * * @return The parser. */ static Parser i32() { return new IntegerParser(); } /** * Make a 32-bit integer parsing consumer. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer i32(Consumer target) { return i32().andApply(target); } /** * Make a 64-bit integer parser. * * @return The parser. */ static Parser i64() { return new LongParser(); } /** * Make a 64-bit integer parsing consumer. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer i64(Consumer target) { return i64().andApply(target); } /** * Make a double parser. * * @return The parser. */ static Parser dbl() { return new DoubleParser(); } /** * Make a 64-bit integer parsing consumer. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer dbl(Consumer target) { return dbl().andApply(target); } /** * Make an enum value parsing consumer. * * @param klass The enum class. * @return The parser. * @param The enum type. */ static > Parser oneOf(Class klass) { return new EnumParser<>(klass); } /** * Make a file parsing consumer that refers to an existing file. * * @param klass The enum class. * @param target The target consumer. * @param The enum type. * @return The consumer wrapper. */ static > Consumer oneOf(Class klass, Consumer target) { return oneOf(klass).andApply(target); } /** * Make a file parser that refers to an existing file. * * @return The parser. */ static Parser file() { return new FileParser(); } /** * Make a file parsing consumer that refers to an existing file. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer file(Consumer target) { return file().andApply(target); } /** * Make a file parser that refers to an existing directory. * * @return The consumer wrapper. */ static Parser dir() { return new DirParser(); } /** * Make a file parsing consumer that refers to an existing directory. * * @param target The target consumer. * @return The parser. */ static Consumer dir(Consumer target) { return dir().andApply(target); } /** * Make a file parser that refers either to a non-existing entry or an * existing file, but not a directory or special device. * * @return The parser. */ static Parser outputFile() { return new OutputFileParser(); } /** * Make a file parsing consumer that refers either to a non-existing entry or an * existing file, but not a directory or special device. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer outputFile(Consumer target) { return outputFile().andApply(target); } /** * Make a parser that refers either to a non-existing entry or an * existing directory, but not a file or special device. * * @return The parser. */ static Parser outputDir() { return new OutputDirParser(); } /** * Make a parsing consumer that refers either to a non-existing entry or an * existing directory, but not a file or special device. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer outputDir(Consumer target) { return outputDir().andApply(target); } /** * Make a parser that parses a path. * * @return The parser. */ static Parser path() { return new PathParser(); } /** * Make a parsing consumer that parses a path. * * @param target The target consumer. * @return The consumer wrapper. */ static Consumer path(Consumer target) { return path().andApply(target); } /** * A converter to path values. */ class PathParser implements Parser { @Override public Path parse(String s) { return Paths.get(s); } } /** * A converter to long values. */ class LongParser implements Parser { @Override public Long parse(String s) { try { return Long.parseLong(s); } catch (NumberFormatException nfe) { throw new ArgumentException(nfe, "Invalid long value " + s); } } } /** * A converter to integer values. */ class IntegerParser implements Parser { @Override public Integer parse(String s) { try { return Integer.parseInt(s); } catch (NumberFormatException nfe) { throw new ArgumentException(nfe, "Invalid integer value " + s); } } } /** * A converter to file instances, with validator & error message. */ class FileParser implements Parser { @Override public File parse(String s) { File result = new File(s); if (!result.exists()) { throw new ArgumentException("No such file " + s); } if (!result.isFile()) { throw new ArgumentException(s + " is not a file"); } return result; } } /** * A converter to file instances, with validator & error message. */ class DirParser implements Parser { @Override public File parse(String s) { File result = new File(s); if (!result.exists()) { throw new ArgumentException("No such directory " + s); } if (!result.isDirectory()) { throw new ArgumentException(s + " is not a directory"); } return result; } } /** * A converter to file instances, with validator & error message. */ class OutputFileParser implements Parser { @Override public File parse(String s) { File result = new File(s); if (result.exists() && !result.isFile()) { throw new ArgumentException(s + " exists and is not a file"); } return result; } } /** * A converter to file instances, with validator & error message. */ class OutputDirParser implements Parser { @Override public File parse(String s) { File result = new File(s); if (result.exists() && !result.isDirectory()) { throw new ArgumentException(s + " exists and is not a directory"); } return result; } } /** * A converter to enum constant values. */ class EnumParser> implements Parser { private final Class klass; public EnumParser(Class klass) { this.klass = klass; } @Override @SuppressWarnings("unchecked") public E parse(String name) { try { return (E) klass.getDeclaredMethod("valueOf", String.class) .invoke(null, name); } catch (InvocationTargetException e) { throw new ArgumentException(e.getCause(), "Invalid " + klass.getSimpleName() + " value " + name); } catch (NoSuchMethodException | IllegalAccessException e) { // Should generally be impossible, since enums are declares via native syntax. throw new IllegalStateException(e.getMessage(), e); } } } /** * A converter to double values. */ class DoubleParser implements Parser { @Override public Double parse(String s) { try { return Double.parseDouble(s); } catch (NumberFormatException nfe) { throw new ArgumentException(nfe, "Invalid double value " + s); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy