summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--base/.classpath2
-rw-r--r--base/.settings/org.eclipse.jdt.core.prefs1
-rw-r--r--base/data/regexes.sprop3
-rw-r--r--base/src/examples/java/bjc/utils/examples/AbbrevMapTest.java3
-rw-r--r--base/src/examples/java/bjc/utils/examples/BinarySearchTest.java130
-rw-r--r--base/src/examples/java/bjc/utils/examples/FunctionalFizzBuzz.java71
-rw-r--r--base/src/examples/java/bjc/utils/examples/ShuntTest.java17
-rw-r--r--base/src/examples/java/bjc/utils/examples/gen/DiabloItemGen.java (renamed from base/src/examples/java/bjc/utils/examples/rangen/DiabloItemGen.java)8
-rw-r--r--base/src/examples/java/bjc/utils/examples/gen/RandomStringExamples.java (renamed from base/src/examples/java/bjc/utils/examples/rangen/RandomStringExamples.java)10
-rw-r--r--base/src/examples/java/bjc/utils/examples/gen/ZadronsPouch.java220
-rw-r--r--base/src/main/java/bjc/utils/cli/Command.java35
-rw-r--r--base/src/main/java/bjc/utils/cli/DelegatingCommand.java1
-rw-r--r--base/src/main/java/bjc/utils/cli/GenericCommand.java49
-rw-r--r--base/src/main/java/bjc/utils/cli/GenericCommandMode.java33
-rw-r--r--base/src/main/java/bjc/utils/cli/GenericHelp.java13
-rw-r--r--base/src/main/java/bjc/utils/cli/NullHelp.java9
-rw-r--r--base/src/main/java/bjc/utils/cli/objects/Command.java619
-rw-r--r--base/src/main/java/bjc/utils/cli/objects/DefineCLI.java8
-rw-r--r--base/src/main/java/bjc/utils/cli/objects/DelimSplitterCLI.java22
-rw-r--r--base/src/main/java/bjc/utils/components/ComponentDescription.java6
-rw-r--r--base/src/main/java/bjc/utils/components/ComponentRepository.java (renamed from base/src/main/java/bjc/utils/components/IComponentRepository.java)12
-rw-r--r--base/src/main/java/bjc/utils/components/DescribedComponent.java (renamed from base/src/main/java/bjc/utils/components/IDescribedComponent.java)4
-rw-r--r--base/src/main/java/bjc/utils/components/FileComponentRepository.java20
-rw-r--r--base/src/main/java/bjc/utils/components/MemoryComponentRepository.java16
-rw-r--r--base/src/main/java/bjc/utils/funcutils/Builder.java (renamed from base/src/main/java/bjc/utils/funcutils/IBuilder.java)2
-rw-r--r--base/src/main/java/bjc/utils/funcutils/Callables.java60
-rw-r--r--base/src/main/java/bjc/utils/funcutils/ChainIterator.java54
-rw-r--r--base/src/main/java/bjc/utils/funcutils/CollectorUtils.java8
-rw-r--r--base/src/main/java/bjc/utils/funcutils/CompoundCollector.java22
-rw-r--r--base/src/main/java/bjc/utils/funcutils/EnumUtils.java4
-rw-r--r--base/src/main/java/bjc/utils/funcutils/FuncUtils.java34
-rw-r--r--base/src/main/java/bjc/utils/funcutils/GroupPartIteration.java86
-rw-r--r--base/src/main/java/bjc/utils/funcutils/Isomorphism.java59
-rw-r--r--base/src/main/java/bjc/utils/funcutils/IteratorUtils.java102
-rw-r--r--base/src/main/java/bjc/utils/funcutils/LambdaLock.java12
-rw-r--r--base/src/main/java/bjc/utils/funcutils/ListUtils.java177
-rw-r--r--base/src/main/java/bjc/utils/funcutils/QueueBackedIterator.java36
-rw-r--r--base/src/main/java/bjc/utils/funcutils/SetUtils.java12
-rw-r--r--base/src/main/java/bjc/utils/funcutils/Strategy.java81
-rw-r--r--base/src/main/java/bjc/utils/funcutils/StringUtils.java249
-rw-r--r--base/src/main/java/bjc/utils/funcutils/TestUtils.java31
-rw-r--r--base/src/main/java/bjc/utils/funcutils/TreeUtils.java67
-rw-r--r--base/src/main/java/bjc/utils/gen/RandomGrammar.java12
-rw-r--r--base/src/main/java/bjc/utils/gen/WeightedGrammar.java86
-rw-r--r--base/src/main/java/bjc/utils/gen/WeightedRandom.java45
-rw-r--r--base/src/main/java/bjc/utils/graph/AdjacencyMap.java47
-rw-r--r--base/src/main/java/bjc/utils/graph/Edge.java33
-rw-r--r--base/src/main/java/bjc/utils/graph/Graph.java62
-rw-r--r--base/src/main/java/bjc/utils/gui/ExtensionFileFilter.java4
-rw-r--r--base/src/main/java/bjc/utils/gui/SimpleKeyedButton.java10
-rw-r--r--base/src/main/java/bjc/utils/gui/awt/ExtensionFileFilter.java4
-rw-r--r--base/src/main/java/bjc/utils/gui/panels/DropdownListPanel.java4
-rw-r--r--base/src/main/java/bjc/utils/gui/panels/HolderOutputPanel.java8
-rw-r--r--base/src/main/java/bjc/utils/gui/panels/ListParameterPanel.java4
-rw-r--r--base/src/main/java/bjc/utils/gui/panels/SimpleSpinnerPanel.java2
-rw-r--r--base/src/main/java/bjc/utils/gui/panels/SliderInputPanel.java92
-rw-r--r--base/src/main/java/bjc/utils/ioutils/LevelSplitter.java49
-rw-r--r--base/src/main/java/bjc/utils/ioutils/LineReader.java18
-rw-r--r--base/src/main/java/bjc/utils/ioutils/LogStream.java341
-rw-r--r--base/src/main/java/bjc/utils/ioutils/RegexStringEditor.java12
-rw-r--r--base/src/main/java/bjc/utils/ioutils/ReportWriter.java59
-rw-r--r--base/src/main/java/bjc/utils/ioutils/RuleBasedConfigReader.java22
-rw-r--r--base/src/main/java/bjc/utils/ioutils/SimpleProperties.java12
-rw-r--r--base/src/main/java/bjc/utils/ioutils/TextAreaOutputStream.java (renamed from base/src/main/java/bjc/utils/gui/TextAreaOutputStream.java)2
-rw-r--r--base/src/main/java/bjc/utils/misc/Direction.java6
-rw-r--r--base/src/main/java/bjc/utils/misc/RelativeDirection.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/Precedent.java (renamed from base/src/main/java/bjc/utils/parserutils/IPrecedent.java)4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/ShuntingYard.java43
-rw-r--r--base/src/main/java/bjc/utils/parserutils/StringDescaper.java2
-rw-r--r--base/src/main/java/bjc/utils/parserutils/TokenTransformer.java148
-rw-r--r--base/src/main/java/bjc/utils/parserutils/TokenUtils.java32
-rw-r--r--base/src/main/java/bjc/utils/parserutils/TreeConstructor.java176
-rw-r--r--base/src/main/java/bjc/utils/parserutils/delims/DelimiterGroup.java42
-rw-r--r--base/src/main/java/bjc/utils/parserutils/delims/RegexOpener.java10
-rw-r--r--base/src/main/java/bjc/utils/parserutils/delims/SequenceDelimiter.java20
-rw-r--r--base/src/main/java/bjc/utils/parserutils/delims/StringDelimiter.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/ChainTokenSplitter.java8
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/ConfigurableTokenSplitter.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/ExcludingTokenSplitter.java6
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/FilteredTokenSplitter.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/IdentityTokenSplitter.java16
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/SimpleTokenSplitter.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitter.java4
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitters.java9
-rw-r--r--base/src/main/java/bjc/utils/parserutils/splitter/TransformTokenSplitter.java4
-rw-r--r--base/src/main/java/bjc/utils/patterns/ComplexPattern.java223
-rw-r--r--base/src/main/java/bjc/utils/patterns/FunctionalPattern.java47
-rw-r--r--base/src/main/java/bjc/utils/patterns/FunctionalPatternMatcher.java31
-rw-r--r--base/src/main/java/bjc/utils/patterns/MutablePatternMatcher.java86
-rw-r--r--base/src/main/java/bjc/utils/patterns/NonExhaustiveMatch.java20
-rw-r--r--base/src/main/java/bjc/utils/patterns/Pattern.java14
-rw-r--r--base/src/main/java/bjc/utils/patterns/PatternMatcher.java85
-rw-r--r--base/src/main/java/bjc/utils/patterns/SimplePatternMatcher.java42
-rw-r--r--base/src/main/java/bjc/utils/patterns/SimplePatttern.java40
-rw-r--r--base/src/main/java/bjc/utils/patterns/package-info.java6
-rw-r--r--base/src/test/java/bjc/utils/test/ioutils/LevelSplitterTest.java12
-rw-r--r--clformat/TODO20
-rw-r--r--clformat/data/clformat.sprop11
-rw-r--r--clformat/docs/Makefile22
-rw-r--r--clformat/docs/clformat.ms231
-rw-r--r--clformat/docs/clformat.tmac63
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLFormatter.java324
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLModifiers.java18
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLParameters.java85
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLString.java10
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLTokenizer.java80
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/CLValue.java44
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/ClauseDecree.java54
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/Decree.java133
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/GroupDecree.java39
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/SimpleDecree.java129
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/AestheticDirective.java57
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/CaseDirective.java16
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/CharacterDirective.java17
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/CompileContext.java4
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/ConditionalDirective.java85
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/DecimalDirective.java89
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/EscapeDirective.java34
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/FormatParameters.java22
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/FreshlineDirective.java7
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/GeneralNumberDirective.java2
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/IndentDirective.java11
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/InflectDirective.java25
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/IterationDirective.java63
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/LiteralDirective.java4
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/NumberDirective.java5
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/RadixDirective.java18
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/RecursiveDirective.java12
-rw-r--r--clformat/src/main/java/bjc/utils/ioutils/format/directives/TabulateDirective.java24
-rw-r--r--clformat/src/test/java/bjc/utils/test/ioutils/CLFormatterTest.java16
-rw-r--r--clformat/src/test/java/bjc/utils/test/ioutils/CLTokenizerTest.java2
-rw-r--r--commander/.classpath28
-rw-r--r--commander/.gitignore1
-rw-r--r--commander/.project23
-rw-r--r--commander/.settings/org.eclipse.core.resources.prefs4
-rw-r--r--commander/.settings/org.eclipse.jdt.core.prefs8
-rw-r--r--commander/.settings/org.eclipse.m2e.core.prefs4
-rw-r--r--commander/pom.xml35
-rw-r--r--commander/src/example/java/bjc/commander/CommanderCLI.java27
-rw-r--r--commander/src/main/java/bjc/commander/Commander.java386
-rw-r--r--pom.xml5
141 files changed, 4384 insertions, 2405 deletions
diff --git a/base/.classpath b/base/.classpath
index 9a48b8b..673da3f 100644
--- a/base/.classpath
+++ b/base/.classpath
@@ -13,9 +13,9 @@
</classpathentry>
<classpathentry kind="src" output="target/test-classes" path="src/test/java">
<attributes>
+ <attribute name="test" value="true"/>
<attribute name="optional" value="true"/>
<attribute name="maven.pomderived" value="true"/>
- <attribute name="test" value="true"/>
</attributes>
</classpathentry>
<classpathentry kind="src" path="src/examples/java"/>
diff --git a/base/.settings/org.eclipse.jdt.core.prefs b/base/.settings/org.eclipse.jdt.core.prefs
index 51489b4..39e9a95 100644
--- a/base/.settings/org.eclipse.jdt.core.prefs
+++ b/base/.settings/org.eclipse.jdt.core.prefs
@@ -10,6 +10,7 @@ org.eclipse.jdt.core.compiler.debug.sourceFile=generate
org.eclipse.jdt.core.compiler.problem.assertIdentifier=error
org.eclipse.jdt.core.compiler.problem.enablePreviewFeatures=disabled
org.eclipse.jdt.core.compiler.problem.enumIdentifier=error
+org.eclipse.jdt.core.compiler.problem.forbiddenReference=warning
org.eclipse.jdt.core.compiler.problem.reportPreviewFeatures=ignore
org.eclipse.jdt.core.compiler.release=disabled
org.eclipse.jdt.core.compiler.source=1.8
diff --git a/base/data/regexes.sprop b/base/data/regexes.sprop
index f1e6787..53560c8 100644
--- a/base/data/regexes.sprop
+++ b/base/data/regexes.sprop
@@ -38,3 +38,6 @@ fpLeader [\x00-\x20]*[+-]?(?:NaN|Infinity|
# Miscellaneous validation regular expressions
##############################################
intLiteral \A[+\-]\d+\Z
+
+readerLineComments (?:#.*\R)*
+readerLineBlock (\s*)\S.+\R(?:\1\s+\S.*\R)* \ No newline at end of file
diff --git a/base/src/examples/java/bjc/utils/examples/AbbrevMapTest.java b/base/src/examples/java/bjc/utils/examples/AbbrevMapTest.java
index 9c539e9..a526408 100644
--- a/base/src/examples/java/bjc/utils/examples/AbbrevMapTest.java
+++ b/base/src/examples/java/bjc/utils/examples/AbbrevMapTest.java
@@ -6,6 +6,7 @@ import java.util.Scanner;
import bjc.esodata.AbbrevMap2;
import bjc.utils.funcutils.StringUtils;
+import bjc.utils.parserutils.*;
/**
* Test for abbreviation map.
@@ -29,7 +30,7 @@ public class AbbrevMapTest {
String ln = scn.nextLine().trim();
while (!ln.equals("")) {
- final List<String> commParts = StringUtils.processArguments(ln);
+ final List<String> commParts = TokenUtils.processArguments(ln);
switch (commParts.get(0)) {
case "add":
diff --git a/base/src/examples/java/bjc/utils/examples/BinarySearchTest.java b/base/src/examples/java/bjc/utils/examples/BinarySearchTest.java
deleted file mode 100644
index 5e8e372..0000000
--- a/base/src/examples/java/bjc/utils/examples/BinarySearchTest.java
+++ /dev/null
@@ -1,130 +0,0 @@
-package bjc.utils.examples;
-
-import java.util.Scanner;
-
-import bjc.funcdata.bst.BinarySearchTree;
-import bjc.funcdata.bst.TreeLinearizationMethod;
-
-/**
- * Example showing how to use the binary search tree.
- *
- * @author ben
- *
- */
-public class BinarySearchTest {
- private static void display(final BinarySearchTree<Character> tree,
- final Scanner input) {
- System.out.print(
- "What order would you like the tree to be printed in (m for options): ");
- char command;
-
- while (true) {
- command = input.nextLine().charAt(0);
- TreeLinearizationMethod method = null;
-
- switch (command) {
- case 'm':
- System.out.println("Possible tree printing methods: ");
- System.out.println(
- "\tp: Preorder printing (print parent first, then left & right).");
- System.out.println(
- "\ti: Inorder printing (print left first, then parent & right).");
- System.out.println(
- "\to: Postorder printing (print left first, then right & parent).");
- break;
- case 'p':
- method = TreeLinearizationMethod.PREORDER;
- break;
- case 'i':
- method = TreeLinearizationMethod.INORDER;
- break;
- case 'o':
- method = TreeLinearizationMethod.POSTORDER;
- break;
- default:
- System.out.println("ERROR: Unknown command.");
- }
-
- if (method != null) {
- tree.traverse(method, element -> {
- System.out.println("Node: " + element);
- return true;
- });
-
- return;
- }
-
- System.out.print(
- "What order would you like the tree to be printed in (m for options): ");
- }
- }
-
- /**
- * Main method of class
- *
- * @param args
- * Unused CLI args
- */
- public static void main(final String[] args) {
- final Scanner input = new Scanner(System.in);
- System.out.println("Binary Tree Constructor/Searcher");
- final BinarySearchTree<Character> tree
- = new BinarySearchTree<>((o1, o2) -> o1 - o2);
-
- char command = ' ';
- while (command != 'e') {
- System.out.print("Enter a command (m for help): ");
- command = input.nextLine().charAt(0);
-
- switch (command) {
- case 'm':
- System.out.println("Valid commands: ");
- System.out.println("\tm: Display this help message.");
- System.out.println("\te: Exit this program.");
- System.out.println("\ta: Add a node to the binary tree.");
- System.out.println("\td: Display the binary tree.");
- System.out.println("\tr: Remove a node from the binary tree.");
- System.out.println("\tf: Check if a given node is in the binary tree.");
- System.out.println("\tt: Trim all deleted nodes from the tree.");
- System.out.println("\tb: Balance the tree (also trims dead nodes)");
- break;
- case 'a':
- System.out.print("Enter the letter to add to the binary tree: ");
- command = input.nextLine().charAt(0);
-
- tree.addNode(command);
- break;
- case 'r':
- System.out.print("Enter the letter to add to the binary tree: ");
- command = input.nextLine().charAt(0);
-
- tree.deleteNode(command);
- break;
- case 'd':
- display(tree, input);
- break;
- case 'f':
- System.out.print("Enter the letter to add to the binary tree: ");
- command = input.nextLine().charAt(0);
-
- final boolean inTree = tree.isInTree(command);
- if (inTree) {
- System.out.printf("Node %s was found\n", command);
- } else {
- System.out.printf("Node %s was not found\n", command);
- }
- break;
- case 't':
- tree.trim();
- break;
- case 'b':
- tree.balance();
- break;
- default:
- System.out.println("ERROR: Unrecognized command.");
- }
- }
-
- input.close();
- }
-}
diff --git a/base/src/examples/java/bjc/utils/examples/FunctionalFizzBuzz.java b/base/src/examples/java/bjc/utils/examples/FunctionalFizzBuzz.java
new file mode 100644
index 0000000..08551c7
--- /dev/null
+++ b/base/src/examples/java/bjc/utils/examples/FunctionalFizzBuzz.java
@@ -0,0 +1,71 @@
+package bjc.utils.examples;
+
+import java.util.function.*;
+import bjc.data.*;
+
+import static bjc.functypes.Combinators.*;
+
+/**
+ * @author Ben Culkin
+ *
+ */
+public class FunctionalFizzBuzz {
+ interface FizzBuzzFunc
+ extends Function<UnaryOperator<String>, UnaryOperator<String>> {
+ // Alias type
+ }
+
+ /**
+ * Calculate fizz-buzz in an un-functional functional way.
+ *
+ * @param args Ignored CLI args
+ */
+ public static void main(String[] args) {
+ // Do something, at some point
+ times(
+ 100,
+ andThen(
+ invoke(
+ introducing(
+ () -> new IntHolder(),
+ (holder, arg) -> {
+ Consumer<Integer> numSetter2 = (num) -> holder.set(num);
+
+ return (num) -> beforeThis(
+ numSetter2,
+ compose(
+ input -> Integer.toString(input),
+ strang ->
+ fbMaker(3, "Fizz", "", holder)
+ .apply(
+ fbMaker(5, "Buzz", "", holder)
+ .apply(ignore -> ignore)
+ )
+ .apply(strang)
+ )
+ ).apply(num);
+ }
+ )
+ ),
+ System.out::println
+ )
+ );
+ }
+
+ private static FizzBuzzFunc fbMaker(
+ int cond, String initial, String interleave, IntHolder var) {
+ return func -> invoke(
+ iftt(
+ ignored1 -> var.get() % cond == 0,
+ arg -> initial + func.apply(interleave),
+ func
+ )
+ );
+ }
+
+ private static <Input, Output> Function<Input, Output> iftt(Predicate<Input> in,
+ Output ifTrue,
+ Output ifFalse) {
+ return (arg) -> in.test(arg) ? ifTrue : ifFalse;
+ }
+} \ No newline at end of file
diff --git a/base/src/examples/java/bjc/utils/examples/ShuntTest.java b/base/src/examples/java/bjc/utils/examples/ShuntTest.java
index b62a5a6..39d37ee 100644
--- a/base/src/examples/java/bjc/utils/examples/ShuntTest.java
+++ b/base/src/examples/java/bjc/utils/examples/ShuntTest.java
@@ -3,7 +3,7 @@ package bjc.utils.examples;
import java.util.Scanner;
import bjc.funcdata.FunctionalStringTokenizer;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.parserutils.ShuntingYard;
/**
@@ -12,7 +12,8 @@ import bjc.utils.parserutils.ShuntingYard;
* @author ben
*
*/
-public class ShuntTest {
+public class ShuntTest
+{
/**
* Main method
*
@@ -20,16 +21,16 @@ public class ShuntTest {
* Unused CLI args
*/
public static void main(final String[] args) {
- final Scanner inputSource = new Scanner(System.in);
+ Scanner inputSource = new Scanner(System.in);
System.out.print("Enter a expression to shunt: ");
- final String line = inputSource.nextLine();
+ String line = inputSource.nextLine();
- final ShuntingYard<String> yard = new ShuntingYard<>(true);
+ ShuntingYard<String> yard = new ShuntingYard<>(true);
- final IList<String> preTokens
- = new FunctionalStringTokenizer(line).toList(strang -> strang);
- final IList<String> shuntedTokens = yard.postfix(preTokens, strang -> strang);
+ FunctionalStringTokenizer tokenizer = new FunctionalStringTokenizer(line);
+ ListEx<String> preTokens = tokenizer.toList(strang -> strang);
+ ListEx<String> shuntedTokens = yard.postfix(preTokens, strang -> strang);
System.out.println(shuntedTokens.toString());
diff --git a/base/src/examples/java/bjc/utils/examples/rangen/DiabloItemGen.java b/base/src/examples/java/bjc/utils/examples/gen/DiabloItemGen.java
index 935a189..8699527 100644
--- a/base/src/examples/java/bjc/utils/examples/rangen/DiabloItemGen.java
+++ b/base/src/examples/java/bjc/utils/examples/gen/DiabloItemGen.java
@@ -1,7 +1,7 @@
-package bjc.utils.examples.rangen;
+package bjc.utils.examples.gen;
import bjc.funcdata.FunctionalStringTokenizer;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.gen.WeightedGrammar;
/**
@@ -15,7 +15,7 @@ public class DiabloItemGen {
private static void addCase(final String ruleName, final int probability,
final String ruleParts) {
- final IList<String> parts = FunctionalStringTokenizer.fromString(ruleParts)
+ final ListEx<String> parts = FunctionalStringTokenizer.fromString(ruleParts)
.toList(strang -> strang);
rules.addCase(ruleName, probability, parts);
@@ -85,7 +85,7 @@ public class DiabloItemGen {
addInfixRules();
for (int i = 0; i < 100; i++) {
- final IList<String> ls = rules.generateListValues("<item>", " ");
+ final ListEx<String> ls = rules.generateListValues("<item>", " ");
final StringBuilder sb = new StringBuilder();
ls.forEach(sb::append);
diff --git a/base/src/examples/java/bjc/utils/examples/rangen/RandomStringExamples.java b/base/src/examples/java/bjc/utils/examples/gen/RandomStringExamples.java
index bdc4f53..5e71ce0 100644
--- a/base/src/examples/java/bjc/utils/examples/rangen/RandomStringExamples.java
+++ b/base/src/examples/java/bjc/utils/examples/gen/RandomStringExamples.java
@@ -1,8 +1,8 @@
-package bjc.utils.examples.rangen;
+package bjc.utils.examples.gen;
import bjc.funcdata.FunctionalList;
import bjc.funcdata.FunctionalStringTokenizer;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.gen.RandomGrammar;
/**
@@ -15,10 +15,10 @@ public class RandomStringExamples {
private static RandomGrammar<String> rg;
private static void addRule(final String rule, final String... cases) {
- final IList<IList<String>> cses = new FunctionalList<>();
+ final ListEx<ListEx<String>> cses = new FunctionalList<>();
for (final String strang : cases) {
- final IList<String> lst
+ final ListEx<String> lst
= FunctionalStringTokenizer.fromString(strang).toList(s -> s);
cses.add(lst);
@@ -56,7 +56,7 @@ public class RandomStringExamples {
addRule("<something>", "<activity>", "<activity> with <person>", "<object>");
for (int i = 0; i < 10; i++) {
- final IList<String> ls = rg.generateListValues("<sentance>", " ");
+ final ListEx<String> ls = rg.generateListValues("<sentance>", " ");
final StringBuilder sb = new StringBuilder();
ls.forEach(sb::append);
diff --git a/base/src/examples/java/bjc/utils/examples/gen/ZadronsPouch.java b/base/src/examples/java/bjc/utils/examples/gen/ZadronsPouch.java
new file mode 100644
index 0000000..184d127
--- /dev/null
+++ b/base/src/examples/java/bjc/utils/examples/gen/ZadronsPouch.java
@@ -0,0 +1,220 @@
+package bjc.utils.examples.gen;
+
+import bjc.funcdata.FunctionalList;
+import bjc.funcdata.FunctionalStringTokenizer;
+import bjc.funcdata.ListEx;
+import bjc.utils.gen.RandomGrammar;
+
+/**
+ * Example showing code manipulate of random grammars
+ *
+ * @author ben
+ */
+public class ZadronsPouch {
+ /**
+ * Main method for running application
+ *
+ * @param args
+ * Unused CLI args
+ */
+ public static void main(String[] args) {
+ ZadronsPouch zp = new ZadronsPouch();
+
+ for (int i = 0; i < 100; i++) {
+ ListEx<String> ls = zp.grammar.generateListValues("[item]", " ");
+
+ StringBuilder sb = new StringBuilder();
+
+ ls.forEach(sp -> sb.append(sp));
+
+ System.out.println(sb.toString().replaceAll("\\s+", " "));
+ }
+ }
+
+ private RandomGrammar<String> grammar;
+
+ /** Create a new instance with a grammar */
+ public ZadronsPouch() {
+ grammar = new RandomGrammar<>();
+
+ /*
+ * @NOTE
+ * Should there be some sort of builder sort of interface?
+ */
+ addRule("[item]",
+ "[egg]", "[glove]", "[crys-sphere]", "[rock]",
+ "[figurine]", "[vial]", "[mini-weapon]", "[bag]",
+ "[card]", "[rope]", "[box]", "[wand]");
+
+ addEggRules();
+ addGloveRules();
+ addCrysSphereRules();
+ addRockRules();
+
+ addFigurineRules();
+ addVialRules();
+ addMiniWeaponRules();
+ addBagRules();
+
+ addCardRules();
+ addRopeRules();
+ addBoxRules();
+ addWandRules();
+ }
+
+ private void addBagRules() {
+ addRule("[bag]",
+ "bag of [bag-type]", "[sack-type] sack", "[purse-type] purse");
+ addRule("[bag-type]",
+ "holding", "tricks", "useful items",
+ "devouring", "dwarf-kind", "invisible cloth",
+ "monster summoning");
+ addRule("[sack-type]",
+ "lunch", "recursive");
+ addRule("[purse-type]",
+ "everfull");
+ }
+
+ private void addBoxRules() {
+ addRule("[box]",
+ "[box-type] box", "cube of [box-type]");
+ addRule("[box-type]",
+ "limited-force", "frost-resisting", "morphing",
+ "self-destructing", "pandora", "panicking");
+ }
+
+ private void addCardRules() {
+ addRule("[card]",
+ "card of [card-type]", "[card-type] card");
+ addRule("[card-type]",
+ "fate", "teleporting", "elusive treasure", "spell-storing",
+ "many-things", "imprisoning", "messaging", "bounty");
+ }
+
+ private void addCrysSphereRules() {
+ addRule("[crys-sphere]",
+ "[sphere-type] spheres", "[sphere-type] sphere",
+ "lens of [lens-type]", "[crystal-type] crystal",
+ "crystal of [crystal-type]", "crystal ball",
+ "crystal ball of [crys-suffix]");
+ addRule("[sphere-type]",
+ "microphonic", "seeing-eye");
+ addRule("[lens-type]",
+ "detection");
+ addRule("[crystal-type]",
+ "prison", "radar");
+ addRule("[crys-suffix]",
+ "jumping");
+ }
+
+ private void addEggRules() {
+ addRule("[egg]",
+ "[egg-type] egg");
+ addRule("[egg-type]",
+ "copper", "stone", "golden",
+ "white", "white/pink", "glass");
+ }
+
+ private void addFigurineRules() {
+ addRule("[figurine]",
+ "[fig-material] [fig-animal]");
+ addRule("[fig-material]",
+ "golden", "onyx", "serpentine", "ivory",
+ "marble", "bronze", "jade", "limestone");
+ addRule("[fig-animal]",
+ "lion", "dog", "owl", "goat",
+ "elephant", "warrior", "palace", "leprechaun");
+ }
+
+ private void addGloveRules() {
+ addRule("[glove]",
+ "gauntlets of [gauntlet-type]",
+ "gloves of [glove-type]",
+ "[glove-type] gloves");
+ addRule("[gauntlet-type]",
+ "dexterity", "power");
+ addRule("[glove-type]",
+ "pushing", "choking", "bigby", "stunning");
+ }
+
+ private void addMiniWeaponRules() {
+ addRule("[mini-weapon]",
+ "minature [weapon-type]", "small [weapon-type]",
+ "tiny [weapon-type]", "[sling-type] sling",
+ "[weapon-type]");
+ addRule("[weapon-type]",
+ "boomerang", "arrow", "net",
+ "catapult", "hammer", "sword", "club");
+ addRule("[sling-type]",
+ "seeking");
+ }
+
+ private void addRockRules() {
+ addRule("[rock]",
+ "[pebble-type] pebble", "stone of [stone-type]",
+ "[stone-type] stone", "brick of [brick-type]",
+ "[geode-type] geode");
+ addRule("[pebble-type]",
+ "inscribed", "elemental control");
+ addRule("[stone-type]",
+ "good-luck", "weight",
+ "blind-defense", "metal-clinging");
+ addRule("[brick-type]",
+ "flying");
+ addRule("[geode-type]",
+ "ioun");
+ }
+
+ private void addRopeRules() {
+ addRule("[rope]",
+ "[rope-type] rope", "rope of [rope-type]",
+ "ball of [string-type] [string-kind]");
+ addRule("[rope-type]",
+ "trick", "entangling", "climbing", "dancing",
+ "tripping", "snaring", "levitating", "self-entangling");
+ addRule("[string-type]",
+ "endless");
+ addRule("[string-kind]",
+ "string", "yarn");
+ }
+
+ private void addRule(String rule, String... cases) {
+ ListEx<ListEx<String>> cses = new FunctionalList<>();
+
+ for (String strang : cases) {
+ cses.add(FunctionalStringTokenizer.fromString(strang).toList(s -> s));
+ }
+
+ grammar.makeRule(rule, cses);
+ }
+
+ private void addVialRules() {
+ addRule("[vial]",
+ "vial of [vial-type]", "[vial-type] vial",
+ "[bottle-type] bottle", "[flask-type] flask");
+ addRule("[vial-type]",
+ "holding", "trapping",
+ "experience", "unnatural regeneration");
+ addRule("[bottle-type]",
+ "ever-smoking", "wheezing",
+ "blank potion");
+ addRule("[flask-type]",
+ "iron");
+ }
+
+ private void addWandRules() {
+ addRule("[wand]",
+ "[wand-type] wand", "wand of [wand-type]",
+ "canceling [wand-type] wand");
+ addRule("[wand-type]",
+ "magic missile", "[spell-1]", "[spell-2]",
+ "gusting", "life-detecting", "zadron");
+ addRule("[spell-1]",
+ "frost", "fire", "lightning", "fear",
+ "illumination", "polymorphing", "conjuration", "paralyzing");
+ addRule("[spell-2]",
+ "[spell2-type] detecting");
+ addRule("[spell2-type]",
+ "magic", "enemy", "secret door/trap");
+ }
+}
diff --git a/base/src/main/java/bjc/utils/cli/Command.java b/base/src/main/java/bjc/utils/cli/Command.java
index 7451cf7..68351bf 100644
--- a/base/src/main/java/bjc/utils/cli/Command.java
+++ b/base/src/main/java/bjc/utils/cli/Command.java
@@ -7,13 +7,6 @@ package bjc.utils.cli;
*/
public interface Command {
/**
- * Create a command that serves as an alias to this one
- *
- * @return A command that serves as an alias to this one
- */
- Command aliased();
-
- /**
* Get the handler that executes this command
*
* @return The handler that executes this command
@@ -26,7 +19,16 @@ public interface Command {
* @return The help entry for this command
*/
CommandHelp getHelp();
-
+
+ /**
+ * Create a command that serves as an alias to this one
+ *
+ * @return A command that serves as an alias to this one
+ */
+ default Command aliased() {
+ return new DelegatingCommand(this);
+ };
+
/**
* Check if this command is an alias of another command
*
@@ -35,4 +37,21 @@ public interface Command {
default boolean isAlias() {
return false;
}
+
+ /**
+ * Create a new basic command.
+ *
+ * @param summary The summary of the command. This is used as a short help
+ * message displayed when listing commands.
+ * @param description The description of the command. This is what is shown
+ * when the detailed help for a command is asked for.
+ * @param handler The implementation for the command.
+ *
+ * @return A command with the given implementation.
+ */
+ static Command from(
+ String summary, String description, CommandHandler handler)
+ {
+ return new GenericCommand(handler, summary, description);
+ }
}
diff --git a/base/src/main/java/bjc/utils/cli/DelegatingCommand.java b/base/src/main/java/bjc/utils/cli/DelegatingCommand.java
index f17b6b5..fa2d1db 100644
--- a/base/src/main/java/bjc/utils/cli/DelegatingCommand.java
+++ b/base/src/main/java/bjc/utils/cli/DelegatingCommand.java
@@ -21,6 +21,7 @@ class DelegatingCommand implements Command {
@Override
public Command aliased() {
+ // Prevent double-indirections
return new DelegatingCommand(delegate);
}
diff --git a/base/src/main/java/bjc/utils/cli/GenericCommand.java b/base/src/main/java/bjc/utils/cli/GenericCommand.java
index a847bea..6903152 100644
--- a/base/src/main/java/bjc/utils/cli/GenericCommand.java
+++ b/base/src/main/java/bjc/utils/cli/GenericCommand.java
@@ -12,40 +12,25 @@ public class GenericCommand implements Command {
/* The help for the command. */
private CommandHelp help;
- /**
- * Create a new generic command.
+ /** Create a new generic command.
*
- * @param handler
- * The handler to use for the command.
- *
- * @param description
- * The description of the command. May be null, in which case
- * a default is provided.
- *
- * @param help
- * The detailed help message for the command. May be null, in
- * which case the description is repeated for the detailed
- * help.
- */
- public GenericCommand(final CommandHandler handler, final String description,
- final String help) {
+ * @param handler The handler to use for the command.
+ * @param summary The summary of the command. May be null, in which case a
+ * default is provided.
+ * @param description The detailed help message for the command. May be null,
+ * in which case the summary is repeated for the
+ * detailed help. */
+ public GenericCommand(final CommandHandler handler, final String summary,
+ final String description) {
if (handler == null)
throw new NullPointerException("Command handler must not be null");
this.handler = handler;
- if (description == null) {
- this.help = new NullHelp();
- } else {
- this.help = new GenericHelp(description, help);
- }
- }
-
- @Override
- public Command aliased() {
- return new DelegatingCommand(this);
+ if (summary == null) this.help = new NullHelp();
+ else this.help = new GenericHelp(summary, description);
}
-
+
@Override
public CommandHandler getHandler() {
return handler;
@@ -57,16 +42,6 @@ public class GenericCommand implements Command {
}
@Override
- public boolean isAlias() {
- return false;
- }
-
- /*
- * (non-Javadoc)
- *
- * @see java.lang.Object#toString()
- */
- @Override
public String toString() {
return String.format("GenericCommand [help=%s]", help);
}
diff --git a/base/src/main/java/bjc/utils/cli/GenericCommandMode.java b/base/src/main/java/bjc/utils/cli/GenericCommandMode.java
index 328bbee..3642137 100644
--- a/base/src/main/java/bjc/utils/cli/GenericCommandMode.java
+++ b/base/src/main/java/bjc/utils/cli/GenericCommandMode.java
@@ -5,7 +5,7 @@ import java.util.function.BiConsumer;
import java.util.function.Consumer;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IMap;
+import bjc.funcdata.MapEx;
/**
* A general command mode, with a customizable set of commands.
@@ -18,12 +18,12 @@ import bjc.funcdata.IMap;
*/
public class GenericCommandMode implements CommandMode {
/* Contains the commands this mode handles. */
- private final IMap<String, Command> commandHandlers;
+ private final MapEx<String, Command> commandHandlers;
/* Commands to execute in every mode. */
- private final IMap<String, Command> defaultHandlers;
+ private final MapEx<String, Command> defaultHandlers;
/* Contains help topics without an associated command */
- private final IMap<String, CommandHelp> helpTopics;
+ private final MapEx<String, CommandHelp> helpTopics;
/* The action to execute upon encountering an unknown command */
private BiConsumer<String, String[]> unknownCommandHandler;
@@ -55,12 +55,12 @@ public class GenericCommandMode implements CommandMode {
throw new NullPointerException("Error output source must be non-null");
this.normalOutput = normalOutput;
- this.errorOutput = errorOutput;
+ this.errorOutput = errorOutput;
/* Initialize maps so that they sort in alphabetical order. */
commandHandlers = new FunctionalMap<>(new TreeMap<>());
defaultHandlers = new FunctionalMap<>(new TreeMap<>());
- helpTopics = new FunctionalMap<>(new TreeMap<>());
+ helpTopics = new FunctionalMap<>(new TreeMap<>());
/* Setup default commands. */
setupDefaultCommands();
@@ -84,17 +84,16 @@ public class GenericCommandMode implements CommandMode {
if (commandName == null) {
throw new NullPointerException("Command name must not be null");
} else if (aliasName == null) {
- String msg = "Alias name must not be null";
- throw new NullPointerException(msg);
+ throw new NullPointerException("Alias name must not be null");
} else if (!commandHandlers.containsKey(commandName)
&& !defaultHandlers.containsKey(commandName)) {
- String msg = String.format("Cannot alias non-existant command '%s'",
+ String msg = String.format("Cannot alias non-existing command '%s'",
commandName);
throw new IllegalArgumentException(msg);
} else if (commandHandlers.containsKey(aliasName)
|| defaultHandlers.containsKey(aliasName)) {
String msg = String.format(
- "Cannot bind alias '%s' to an already bound command.", aliasName);
+ "Cannot bind alias '%s' to an already used name.", aliasName);
throw new IllegalArgumentException(msg);
} else {
/* The command that will be aliased. */
@@ -102,9 +101,9 @@ public class GenericCommandMode implements CommandMode {
/* Get the alias. */
if (defaultHandlers.containsKey(commandName)) {
- aliasedCommand = defaultHandlers.get(commandName).aliased();
+ aliasedCommand = defaultHandlers.get(commandName).get().aliased();
} else {
- aliasedCommand = commandHandlers.get(commandName).aliased();
+ aliasedCommand = commandHandlers.get(commandName).get().aliased();
}
commandHandlers.put(aliasName, aliasedCommand);
@@ -262,16 +261,16 @@ public class GenericCommandMode implements CommandMode {
private void doHelpCommand(final String commandName) {
if (commandHandlers.containsKey(commandName)) {
final String desc
- = commandHandlers.get(commandName).getHelp().getDescription();
+ = commandHandlers.get(commandName).get().getHelp().getDescription();
normalOutput.accept("\n" + desc);
} else if (defaultHandlers.containsKey(commandName)) {
final String desc
- = defaultHandlers.get(commandName).getHelp().getDescription();
+ = defaultHandlers.get(commandName).get().getHelp().getDescription();
normalOutput.accept("\n" + desc);
} else if (helpTopics.containsKey(commandName)) {
- normalOutput.accept("\n" + helpTopics.get(commandName).getDescription());
+ normalOutput.accept("\n" + helpTopics.get(commandName).get().getDescription());
} else {
String msg = String.format("ERROR: No help available for '%s'", commandName);
@@ -371,9 +370,9 @@ public class GenericCommandMode implements CommandMode {
normalOutput.accept("\n");
if (defaultHandlers.containsKey(command))
- return defaultHandlers.get(command).getHandler().handle(args);
+ return defaultHandlers.get(command).get().getHandler().handle(args);
else if (commandHandlers.containsKey(command))
- return commandHandlers.get(command).getHandler().handle(args);
+ return commandHandlers.get(command).get().getHandler().handle(args);
else {
if (args != null) {
String argString = String.join(", ", args);
diff --git a/base/src/main/java/bjc/utils/cli/GenericHelp.java b/base/src/main/java/bjc/utils/cli/GenericHelp.java
index 45d3a88..0157d4f 100644
--- a/base/src/main/java/bjc/utils/cli/GenericHelp.java
+++ b/base/src/main/java/bjc/utils/cli/GenericHelp.java
@@ -1,17 +1,14 @@
package bjc.utils.cli;
-/**
- * Generic implementation of a help topic.
+/** Generic implementation of a help topic.
*
- * @author ben
- */
+ * @author ben */
public class GenericHelp implements CommandHelp {
/* The strings for this help topic. */
private final String summary;
private final String description;
- /**
- * Create a new help topic.
+ /** Create a new help topic.
*
* @param summary
* The summary of this help topic.
@@ -31,9 +28,7 @@ public class GenericHelp implements CommandHelp {
@Override
public String getDescription() {
- if (description == null) {
- return summary;
- }
+ if (description == null) return summary;
return description;
}
diff --git a/base/src/main/java/bjc/utils/cli/NullHelp.java b/base/src/main/java/bjc/utils/cli/NullHelp.java
index 5e5fb67..048b6e7 100644
--- a/base/src/main/java/bjc/utils/cli/NullHelp.java
+++ b/base/src/main/java/bjc/utils/cli/NullHelp.java
@@ -1,10 +1,8 @@
package bjc.utils.cli;
-/**
- * Implementation of a help topic that doesn't exist.
+/** Implementation of a help topic that doesn't exist or isn't provided.
*
- * @author ben
- */
+ * @author ben */
public class NullHelp implements CommandHelp {
@Override
public String getDescription() {
@@ -15,5 +13,4 @@ public class NullHelp implements CommandHelp {
public String getSummary() {
return "No summary provided";
}
-
-}
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/cli/objects/Command.java b/base/src/main/java/bjc/utils/cli/objects/Command.java
index af22643..3b287ea 100644
--- a/base/src/main/java/bjc/utils/cli/objects/Command.java
+++ b/base/src/main/java/bjc/utils/cli/objects/Command.java
@@ -3,341 +3,288 @@ package bjc.utils.cli.objects;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
-/**
- * A single-line command read from the user.
+/** A single-line command read from the user.
*
- * @author Ben Culkin
- */
+ * @author Ben Culkin */
public class Command {
- /**
- * Command status values.
- *
- * @author Ben Culkin
- */
- public static enum CommandStatus {
- /**
- * The command succeeded.
- */
- SUCCESS,
- /**
- * The command failed non-fatally.
- */
- FAIL,
- /**
- * The command failed fatally.
- */
- ERROR,
- /**
- * The command was the last one.
- */
- FINISH,
- }
-
- /**
- * The line number of this command.
- */
- public final int lno;
-
- /**
- * The full text of this command.
- */
- public final String full;
- /**
- * The text of this command without its name.
- */
- public String remn;
- /**
- * The name of this command.
- */
- public final String name;
-
- /**
- * The name of the I/O source this command was read from.
- */
- public final String src;
-
- /**
- * Create a new command.
- *
- * @param ln
- * The string to get the command from.
- *
- * @param lno
- * The number of the line the command came from.
- *
- * @param ioSrc
- * The name of where the I/O came from.
- */
- public Command(String ln, int lno, String ioSrc) {
- int idx = ln.indexOf(' ');
-
- if (idx == -1)
- idx = ln.length();
-
- /* Grab command parts. */
- full = ln;
- name = ln.substring(0, idx).trim();
- remn = ln.substring(idx).trim();
-
- this.lno = lno;
-
- src = ioSrc;
- }
-
- /**
- * Removes up until the first occurrence of a particular string for the
- * remaining command, and returns the removed string.
- *
- * By default, both the substring and the remaining text are trimmed
- * (leading/trailing spaces removed).
- *
- * @param delm
- * The delimiter to stop substringing at.
- *
- * @return The substring, or null if there is no occurrence of the delimiter.
- */
- public String trimTo(String delm) {
- return trimTo(delm, true);
- }
-
- /**
- * Removes up until the first occurrence of a particular string for the
- * remaining command, and returns the removed string.
- *
- * @param delm
- * The delimiter to stop substringing at.
- * @param doTrim
- * Whether or not to trim the substring and remaining command
- * (Remove leading/trailing spaces).
- *
- * @return The substring, or null if there is no occurrence of the delimiter.
- */
- public String trimTo(String delm, boolean doTrim) {
- int idx = remn.indexOf(delm);
- if (idx == -1)
- return null;
-
- String tmp = remn.substring(0, idx);
- remn = remn.substring(idx);
-
- if (doTrim) {
- tmp = tmp.trim();
- remn = remn.trim();
- }
-
- return tmp;
- }
-
- /**
- * Removes up until the first occurrence of a particular regex for the remaining
- * command, and returns the removed string.
- *
- * By default, both the substring and the remaining text are trimmed
- * (leading/trailing spaces removed).
- *
- * @param rDelm
- * The regex to stop substringing at.
- *
- * @return The string, up to the matched pattern.
- */
- public String trimToRX(String rDelm) {
- return trimToRX(Pattern.compile(rDelm), true);
- }
-
- /**
- * Removes up until the first occurrence of a particular regex for the remaining
- * command, and returns the removed string.
- *
- * By default, both the substring and the remaining text are trimmed
- * (leading/trailing spaces removed).
- *
- * @param delm
- * The regex to stop substringing at.
- *
- * @return The string, up to the matched pattern.
- */
- public String trimToRX(Pattern delm) {
- return trimToRX(delm, true);
- }
-
- /**
- * Removes up until the first occurrence of a particular regex for the remaining
- * command, and returns the removed string.
- *
- * @param rDelm
- * The regex to stop substringing at.
- * @param doTrim
- * Whether or not to trim the substring and remaining command
- * (Remove leading/trailing spaces).
- *
- * @return The string, up to the matched pattern.
- */
- public String trimToRX(String rDelm, boolean doTrim) {
- return trimToRX(Pattern.compile(rDelm), doTrim);
- }
-
- /**
- * Removes up until the first occurrence of a particular regex for the remaining
- * command, and returns the removed string.
- *
- * @param delm
- * The regex to stop substringing at.
- * @param doTrim
- * Whether or not to trim the substring and remaining command
- * (Remove leading/trailing spaces).
- *
- * @return The string, up to the matched pattern.
- */
- public String trimToRX(Pattern delm, boolean doTrim) {
- Matcher mat = delm.matcher(remn);
- if (!mat.find())
- return null;
-
- String tmp = remn.substring(0, mat.start());
- remn = remn.substring(mat.end());
-
- if (doTrim) {
- tmp = tmp.trim();
- remn = remn.trim();
- }
-
- return tmp;
- }
-
- /**
- * Removes up until the first occurrence of a particular string for the
- * remaining command, and returns the removed string.
- *
- * By default, both the substring and the remaining text are trimmed
- * (leading/trailing spaces removed).
- *
- * @param delm
- * The delimiter to stop substringing at.
- *
- * @return The substring, or null if there is no occurrence of the delimiter.
- */
- public String trimTo(char delm) {
- return trimTo(delm, true);
- }
-
- /**
- * Removes up until the first occurrence of a particular string for the
- * remaining command, and returns the removed string.
- *
- * @param delm
- * The delimiter to stop substringing at.
- * @param doTrim
- * Whether or not to trim the substring and remaining command
- * (Remove leading/trailing spaces).
- *
- * @return The substring, or null if there is no occurrence of the delimiter.
- */
- public String trimTo(char delm, boolean doTrim) {
- int idx = remn.indexOf(delm);
- if (idx == -1)
- return null;
-
- String tmp = remn.substring(0, idx);
- remn = remn.substring(idx);
-
- if (doTrim) {
- tmp = tmp.trim();
- remn = remn.trim();
- }
-
- return tmp;
- }
-
- /**
- * Check if this command has text after its name.
- *
- * @return Whether or not this command has text after its name.
- */
- public boolean hasRemaining() {
- return !remn.equals("");
- }
-
- /**
- * Parse a command from a string.
- *
- * The main thing this does is ignore blank lines, as well as comments marked by
- * #'s either at the start of the line or part of the way through the line.
- *
- * @param lne
- * The string to get the command from.
- *
- * @param lno
- * The line number of the command.
- *
- * @param srcName
- * The name of where the I/O came from.
- * @return The parsed command
- */
- public static Command fromString(String lne, int lno, String srcName) {
- String ln = lne;
-
- /* Ignore blank lines and comments. */
- if (ln.equals(""))
- return null;
- if (ln.startsWith("#"))
- return null;
-
- /* Trim off comments part-way through the line. */
- int idxHash = ln.indexOf('#');
- if (idxHash != -1) {
- ln = ln.substring(0, idxHash).trim();
- }
-
- return new Command(ln, lno, srcName);
- }
-
- /**
- * Give an informational message about something in relation to this command.
- *
- * @param info
- * The informational message.
- *
- * @param parms
- * The parameters for the informational message.
- * @return The information message.
- */
- public String info(String info, Object... parms) {
- String msg = String.format(info, parms);
-
- return String.format("INFO (%s:%d): %s", src, lno, msg);
- }
-
- /**
- * Warn about something in relation to this command.
- *
- * @param warning
- * The warning message.
- *
- * @param parms
- * The parameters for the warning message.
- *
- * @return The formatted warning.
- */
- public String warn(String warning, Object... parms) {
- String msg = String.format(warning, parms);
-
- return String.format("WARNING (%s:%d): %s", src, lno, msg);
- }
-
- /**
- * Give an error about something in relation to this command.
- *
- * @param err
- * The error message.
- *
- * @param parms
- * The parameters for the error message.
- *
- * @return The formatted error
- */
- public String error(String err, Object... parms) {
- String msg = String.format(err, parms);
-
- return String.format("ERROR (%s:%d): %s", src, lno, msg);
- }
-}
+ /** Command status values.
+ *
+ * @author Ben Culkin */
+ public static enum CommandStatus {
+ /** The command succeeded. */
+ SUCCESS,
+ /** The command failed non-fatally. */
+ FAIL,
+ /** The command failed fatally. */
+ ERROR,
+ /** The command was the last one. */
+ FINISH,
+ }
+
+ /** The line number of this command. */
+ public final int lno;
+
+ /** The full text of this command. */
+ public final String full;
+ /** The text of this command without its name. */
+ public String remn;
+ /** The name of this command. */
+ public final String name;
+
+ /** The name of the I/O source this command was read from. */
+ public final String src;
+
+ /** Create a new command.
+ *
+ * @param ln
+ * The string to get the command from.
+ *
+ * @param lno
+ * The number of the line the command came from.
+ *
+ * @param ioSrc
+ * The name of where the I/O came from. */
+ public Command(String ln, int lno, String ioSrc) {
+ int idx = ln.indexOf(' ');
+ if (idx == -1) idx = ln.length();
+
+ /* Grab command parts. */
+ full = ln;
+ name = ln.substring(0, idx).trim();
+ remn = ln.substring(idx).trim();
+
+ this.lno = lno;
+
+ src = ioSrc;
+ }
+
+ /** Removes up until the first occurrence of a particular string for the
+ * remaining command, and returns the removed string.
+ *
+ * By default, both the substring and the remaining text are trimmed
+ * (leading/trailing spaces removed).
+ *
+ * @param delm
+ * The delimiter to stop substringing at.
+ *
+ * @return The substring, or null if there is no occurrence of the delimiter. */
+ public String trimTo(String delm) {
+ return trimTo(delm, true);
+ }
+
+ /** Removes up until the first occurrence of a particular string for the
+ * remaining command, and returns the removed string.
+ *
+ * @param delm
+ * The delimiter to stop substringing at.
+ * @param doTrim
+ * Whether or not to trim the substring and remaining command
+ * (Remove leading/trailing spaces).
+ *
+ * @return The substring, or null if there is no occurrence of the delimiter. */
+ public String trimTo(String delm, boolean doTrim) {
+ int idx = remn.indexOf(delm);
+ if (idx == -1) return null;
+
+ String tmp = remn.substring(0, idx);
+ remn = remn.substring(idx);
+
+ if (doTrim) {
+ tmp = tmp.trim();
+ remn = remn.trim();
+ }
+
+ return tmp;
+ }
+
+ /** Removes up until the first occurrence of a particular regex for the
+ * remaining
+ * command, and returns the removed string.
+ *
+ * By default, both the substring and the remaining text are trimmed
+ * (leading/trailing spaces removed).
+ *
+ * @param rDelm
+ * The regex to stop substringing at.
+ *
+ * @return The string, up to the matched pattern. */
+ public String trimToRX(String rDelm) {
+ return trimToRX(Pattern.compile(rDelm), true);
+ }
+
+ /** Removes up until the first occurrence of a particular regex for the
+ * remaining
+ * command, and returns the removed string.
+ *
+ * By default, both the substring and the remaining text are trimmed
+ * (leading/trailing spaces removed).
+ *
+ * @param delm
+ * The regex to stop substringing at.
+ *
+ * @return The string, up to the matched pattern. */
+ public String trimToRX(Pattern delm) {
+ return trimToRX(delm, true);
+ }
+
+ /** Removes up until the first occurrence of a particular regex for the
+ * remaining
+ * command, and returns the removed string.
+ *
+ * @param rDelm
+ * The regex to stop substringing at.
+ * @param doTrim
+ * Whether or not to trim the substring and remaining command
+ * (Remove leading/trailing spaces).
+ *
+ * @return The string, up to the matched pattern. */
+ public String trimToRX(String rDelm, boolean doTrim) {
+ return trimToRX(Pattern.compile(rDelm), doTrim);
+ }
+
+ /** Removes up until the first occurrence of a particular regex for the
+ * remaining
+ * command, and returns the removed string.
+ *
+ * @param delm
+ * The regex to stop substringing at.
+ * @param doTrim
+ * Whether or not to trim the substring and remaining command
+ * (Remove leading/trailing spaces).
+ *
+ * @return The string, up to the matched pattern. */
+ public String trimToRX(Pattern delm, boolean doTrim) {
+ Matcher mat = delm.matcher(remn);
+ if (!mat.find()) return null;
+
+ String tmp = remn.substring(0, mat.start());
+ remn = remn.substring(mat.end());
+
+ if (doTrim) {
+ tmp = tmp.trim();
+ remn = remn.trim();
+ }
+
+ return tmp;
+ }
+
+ /** Removes up until the first occurrence of a particular string for the
+ * remaining command, and returns the removed string.
+ *
+ * By default, both the substring and the remaining text are trimmed
+ * (leading/trailing spaces removed).
+ *
+ * @param delm
+ * The delimiter to stop substringing at.
+ *
+ * @return The substring, or null if there is no occurrence of the delimiter. */
+ public String trimTo(char delm) {
+ return trimTo(delm, true);
+ }
+
+ /** Removes up until the first occurrence of a particular string for the
+ * remaining command, and returns the removed string.
+ *
+ * @param delm
+ * The delimiter to stop substringing at.
+ * @param doTrim
+ * Whether or not to trim the substring and remaining command
+ * (Remove leading/trailing spaces).
+ *
+ * @return The substring, or null if there is no occurrence of the delimiter. */
+ public String trimTo(char delm, boolean doTrim) {
+ int idx = remn.indexOf(delm);
+ if (idx == -1) return null;
+
+ String tmp = remn.substring(0, idx);
+ remn = remn.substring(idx);
+
+ if (doTrim) {
+ tmp = tmp.trim();
+ remn = remn.trim();
+ }
+
+ return tmp;
+ }
+
+ /** Check if this command has text after its name.
+ *
+ * @return Whether or not this command has text after its name. */
+ public boolean hasRemaining() {
+ return !remn.equals("");
+ }
+
+ /** Parse a command from a string.
+ *
+ * The main thing this does is ignore blank lines, as well as comments marked by
+ * #'s either at the start of the line or part of the way through the line.
+ *
+ * @param lne
+ * The string to get the command from.
+ *
+ * @param lno
+ * The line number of the command.
+ *
+ * @param srcName
+ * The name of where the I/O came from.
+ *
+ * @return The parsed command */
+ public static Command fromString(String lne, int lno, String srcName) {
+ String ln = lne;
+
+ /* Ignore blank lines and comments. */
+ if (ln.equals("")) return null;
+ if (ln.startsWith("#")) return null;
+
+ /* Trim off comments part-way through the line. */
+ int idxHash = ln.indexOf('#');
+ if (idxHash != -1) ln = ln.substring(0, idxHash).trim();
+
+ return new Command(ln, lno, srcName);
+ }
+
+ /** Give an informational message about something in relation to this command.
+ *
+ * @param info
+ * The informational message.
+ *
+ * @param parms
+ * The parameters for the informational message.
+ *
+ * @return The information message. */
+ public String info(String info, Object... parms) {
+ String msg = String.format(info, parms);
+
+ return String.format("INFO (%s:%d): %s", src, lno, msg);
+ }
+
+ /** Warn about something in relation to this command.
+ *
+ * @param warning
+ * The warning message.
+ *
+ * @param parms
+ * The parameters for the warning message.
+ *
+ * @return The formatted warning. */
+ public String warn(String warning, Object... parms) {
+ String msg = String.format(warning, parms);
+
+ return String.format("WARNING (%s:%d): %s", src, lno, msg);
+ }
+
+ /** Give an error about something in relation to this command.
+ *
+ * @param err
+ * The error message.
+ *
+ * @param parms
+ * The parameters for the error message.
+ *
+ * @return The formatted error */
+ public String error(String err, Object... parms) {
+ String msg = String.format(err, parms);
+
+ return String.format("ERROR (%s:%d): %s", src, lno, msg);
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/cli/objects/DefineCLI.java b/base/src/main/java/bjc/utils/cli/objects/DefineCLI.java
index d2595d6..613b223 100644
--- a/base/src/main/java/bjc/utils/cli/objects/DefineCLI.java
+++ b/base/src/main/java/bjc/utils/cli/objects/DefineCLI.java
@@ -14,7 +14,7 @@ import java.util.logging.Logger;
import java.util.regex.Pattern;
import bjc.utils.cli.objects.Command.CommandStatus;
-import bjc.utils.funcutils.StringUtils;
+import bjc.utils.parserutils.*;
/*
* @TODO 10/09/17 :DefineCLIFinish
@@ -130,7 +130,7 @@ public class DefineCLI {
}
private CommandStatus defString(Command com) {
- List<String> arguments = StringUtils.processArguments(com.remn);
+ List<String> arguments = TokenUtils.processArguments(com.remn);
if (arguments.size() < 1) {
LOGGER.severe(com.error(
@@ -158,7 +158,7 @@ public class DefineCLI {
}
private CommandStatus defFormat(Command com) {
- List<String> arguments = StringUtils.processArguments(com.remn);
+ List<String> arguments = TokenUtils.processArguments(com.remn);
if (arguments.size() < 1) {
LOGGER.severe(com.error(
@@ -186,7 +186,7 @@ public class DefineCLI {
}
private CommandStatus bindFormat(Command com) {
- List<String> strings = StringUtils.processArguments(com.remn);
+ List<String> strings = TokenUtils.processArguments(com.remn);
if (strings.size() < 2) {
LOGGER.severe(com.error(
diff --git a/base/src/main/java/bjc/utils/cli/objects/DelimSplitterCLI.java b/base/src/main/java/bjc/utils/cli/objects/DelimSplitterCLI.java
index a6820f2..53d6d1e 100644
--- a/base/src/main/java/bjc/utils/cli/objects/DelimSplitterCLI.java
+++ b/base/src/main/java/bjc/utils/cli/objects/DelimSplitterCLI.java
@@ -10,9 +10,9 @@ import java.util.List;
import java.util.Map;
import java.util.Scanner;
-import bjc.data.ITree;
+import bjc.data.Tree;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.funcutils.StringUtils;
import bjc.utils.ioutils.MirrorDB;
import bjc.utils.parserutils.delims.DelimiterException;
@@ -359,7 +359,7 @@ public class DelimSplitterCLI {
private void handleDelim(final String args) {
try {
- final ITree<String> res = dlm.delimitSequence(args.split(" "));
+ final Tree<String> res = dlm.delimitSequence(args.split(" "));
printDelimSeq(res);
} catch (final DelimiterException dex) {
@@ -372,14 +372,14 @@ public class DelimSplitterCLI {
for (int i = 0; i < argArray.length; i++) {
final String arg = argArray[i];
- final IList<String> strangs = split.split(arg);
+ final ListEx<String> strangs = split.split(arg);
System.out.printf("%d '%s' %s\n", i, arg, strangs);
}
}
private void handleTest(final String inp, final boolean splitWS) {
- IList<String> strings;
+ ListEx<String> strings;
try {
strings = split.split(inp);
@@ -400,7 +400,7 @@ public class DelimSplitterCLI {
strings = new FunctionalList<>(tks);
}
try {
- final ITree<String> delim
+ final Tree<String> delim
= dlm.delimitSequence(strings.toArray(new String[0]));
printDelimSeq(delim);
@@ -410,7 +410,7 @@ public class DelimSplitterCLI {
}
}
- private void printDelimSeq(final ITree<String> delim) {
+ private void printDelimSeq(final Tree<String> delim) {
System.out.println("Delimited tokens:\n" + delim.getChild(1).toString());
System.out.print("Delimited expr: ");
printDelimTree(delim);
@@ -420,7 +420,7 @@ public class DelimSplitterCLI {
System.out.println();
}
- private void printDelimTree(final ITree<String> tree) {
+ private void printDelimTree(final Tree<String> tree) {
final StringBuilder sb = new StringBuilder();
intPrintDelimTree(tree.getChild(1), sb);
@@ -428,13 +428,13 @@ public class DelimSplitterCLI {
System.out.println(sb.toString().replaceAll("\\s+", " "));
}
- private void intPrintDelimTree(final ITree<String> tree, final StringBuilder sb) {
+ private void intPrintDelimTree(final Tree<String> tree, final StringBuilder sb) {
tree.doForChildren(child -> {
intPrintDelimNode(child, sb);
});
}
- private void intPrintDelimNode(final ITree<String> tree, final StringBuilder sb) {
+ private void intPrintDelimNode(final Tree<String> tree, final StringBuilder sb) {
if (tree.getHead().equals("contents")) {
intPrintDelimTree(tree, sb);
return;
@@ -458,7 +458,7 @@ public class DelimSplitterCLI {
case 3:
intPrintDelimNode(tree.getChild(0), sb);
- final ITree<String> contents = tree.getChild(1);
+ final Tree<String> contents = tree.getChild(1);
intPrintDelimTree(contents.getChild(0), sb);
intPrintDelimNode(tree.getChild(2), sb);
diff --git a/base/src/main/java/bjc/utils/components/ComponentDescription.java b/base/src/main/java/bjc/utils/components/ComponentDescription.java
index 189ef90..2feed8d 100644
--- a/base/src/main/java/bjc/utils/components/ComponentDescription.java
+++ b/base/src/main/java/bjc/utils/components/ComponentDescription.java
@@ -5,7 +5,7 @@ package bjc.utils.components;
*
* @author ben
*/
-public class ComponentDescription implements IDescribedComponent {
+public class ComponentDescription implements DescribedComponent {
/* Check arguments are good. */
@SuppressWarnings("unused")
private static void sanityCheckArgs(final String name, final String author,
@@ -58,7 +58,7 @@ public class ComponentDescription implements IDescribedComponent {
@Override
public String getAuthor() {
if (author == null) {
- return IDescribedComponent.super.getAuthor();
+ return DescribedComponent.super.getAuthor();
}
return author;
@@ -67,7 +67,7 @@ public class ComponentDescription implements IDescribedComponent {
@Override
public String getDescription() {
if (description == null) {
- return IDescribedComponent.super.getDescription();
+ return DescribedComponent.super.getDescription();
}
return description;
diff --git a/base/src/main/java/bjc/utils/components/IComponentRepository.java b/base/src/main/java/bjc/utils/components/ComponentRepository.java
index 5ebb1de..120edc8 100644
--- a/base/src/main/java/bjc/utils/components/IComponentRepository.java
+++ b/base/src/main/java/bjc/utils/components/ComponentRepository.java
@@ -1,25 +1,25 @@
package bjc.utils.components;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
/**
* A collection of implementations of a particular type of
- * {@link IDescribedComponent}.
+ * {@link DescribedComponent}.
*
* @author ben
*
* @param <ComponentType>
* The type of components contained in this repository.
*/
-public interface IComponentRepository<ComponentType extends IDescribedComponent> {
+public interface ComponentRepository<ComponentType extends DescribedComponent> {
/**
* Get all of the components this repository knows about.
*
* @return A map from component name to component, containing all of the
* components in the repositories.
*/
- public IMap<String, ComponentType> getAll();
+ public MapEx<String, ComponentType> getAll();
/**
* Get a component with a specific name.
@@ -36,7 +36,7 @@ public interface IComponentRepository<ComponentType extends IDescribedComponent>
*
* @return A list of all the registered components.
*/
- public default IList<ComponentType> getList() {
+ public default ListEx<ComponentType> getList() {
return getAll().valueList();
}
diff --git a/base/src/main/java/bjc/utils/components/IDescribedComponent.java b/base/src/main/java/bjc/utils/components/DescribedComponent.java
index ae3e06c..dcbaf59 100644
--- a/base/src/main/java/bjc/utils/components/IDescribedComponent.java
+++ b/base/src/main/java/bjc/utils/components/DescribedComponent.java
@@ -7,7 +7,7 @@ package bjc.utils.components;
* @author ben
*
*/
-public interface IDescribedComponent extends Comparable<IDescribedComponent> {
+public interface DescribedComponent extends Comparable<DescribedComponent> {
/**
* Get the author of this component.
*
@@ -52,7 +52,7 @@ public interface IDescribedComponent extends Comparable<IDescribedComponent> {
}
@Override
- default int compareTo(final IDescribedComponent o) {
+ default int compareTo(final DescribedComponent o) {
int res = getName().compareTo(o.getName());
if (res == 0) {
diff --git a/base/src/main/java/bjc/utils/components/FileComponentRepository.java b/base/src/main/java/bjc/utils/components/FileComponentRepository.java
index 82d2770..e0e929f 100644
--- a/base/src/main/java/bjc/utils/components/FileComponentRepository.java
+++ b/base/src/main/java/bjc/utils/components/FileComponentRepository.java
@@ -9,11 +9,11 @@ import java.util.function.Function;
import java.util.logging.Level;
import java.util.logging.Logger;
-import bjc.data.IHolder;
+import bjc.data.Holder;
import bjc.data.Identity;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
import bjc.utils.funcutils.FileUtils;
/**
@@ -24,14 +24,14 @@ import bjc.utils.funcutils.FileUtils;
* @param <ComponentType>
* The type of component being read in.
*/
-public class FileComponentRepository<ComponentType extends IDescribedComponent>
- implements IComponentRepository<ComponentType> {
+public class FileComponentRepository<ComponentType extends DescribedComponent>
+ implements ComponentRepository<ComponentType> {
/* The logger to use for storing data about this class. */
private static final Logger CLASS_LOGGER
= Logger.getLogger("FileComponentRepository");
/* The internal storage of components. */
- private IMap<String, ComponentType> components;
+ private MapEx<String, ComponentType> components;
/* The path that all the components came from. */
private Path sourceDirectory;
@@ -69,7 +69,7 @@ public class FileComponentRepository<ComponentType extends IDescribedComponent>
sourceDirectory = directory.toPath().toAbsolutePath();
/* Marker for making sure we don't skip the parent. */
- final IHolder<Boolean> isFirstDir = new Identity<>(true);
+ final Holder<Boolean> isFirstDir = new Identity<>(true);
/*
* Predicate to use to traverse all the files in a directory, but not recurse
@@ -110,17 +110,17 @@ public class FileComponentRepository<ComponentType extends IDescribedComponent>
}
@Override
- public IMap<String, ComponentType> getAll() {
+ public MapEx<String, ComponentType> getAll() {
return components;
}
@Override
public ComponentType getByName(final String name) {
- return components.get(name);
+ return components.get(name).get();
}
@Override
- public IList<ComponentType> getList() {
+ public ListEx<ComponentType> getList() {
return components.valueList();
}
diff --git a/base/src/main/java/bjc/utils/components/MemoryComponentRepository.java b/base/src/main/java/bjc/utils/components/MemoryComponentRepository.java
index ec3911e..f83c293 100644
--- a/base/src/main/java/bjc/utils/components/MemoryComponentRepository.java
+++ b/base/src/main/java/bjc/utils/components/MemoryComponentRepository.java
@@ -1,6 +1,6 @@
package bjc.utils.components;
-import bjc.funcdata.IMap;
+import bjc.funcdata.MapEx;
/**
* A repository of components stored in memory.
@@ -10,9 +10,9 @@ import bjc.funcdata.IMap;
* @param <ComponentType>
* The type of component stored in the repository.
*/
-public class MemoryComponentRepository<ComponentType extends IDescribedComponent>
- implements IComponentRepository<ComponentType> {
- private final IMap<String, ComponentType> repo;
+public class MemoryComponentRepository<ComponentType extends DescribedComponent>
+ implements ComponentRepository<ComponentType> {
+ private final MapEx<String, ComponentType> repo;
private final String source;
@@ -22,7 +22,7 @@ public class MemoryComponentRepository<ComponentType extends IDescribedComponent
* @param repo
* The set of components to use.
*/
- public MemoryComponentRepository(IMap<String, ComponentType> repo) {
+ public MemoryComponentRepository(MapEx<String, ComponentType> repo) {
this(repo, "memory");
}
@@ -34,20 +34,20 @@ public class MemoryComponentRepository<ComponentType extends IDescribedComponent
* @param source
* Where the components came from.
*/
- public MemoryComponentRepository(IMap<String, ComponentType> repo, String source) {
+ public MemoryComponentRepository(MapEx<String, ComponentType> repo, String source) {
this.repo = repo;
this.source = source;
}
@Override
- public IMap<String, ComponentType> getAll() {
+ public MapEx<String, ComponentType> getAll() {
return repo;
}
@Override
public ComponentType getByName(String name) {
- return repo.get(name);
+ return repo.get(name).get();
}
@Override
diff --git a/base/src/main/java/bjc/utils/funcutils/IBuilder.java b/base/src/main/java/bjc/utils/funcutils/Builder.java
index b1a2020..72c045d 100644
--- a/base/src/main/java/bjc/utils/funcutils/IBuilder.java
+++ b/base/src/main/java/bjc/utils/funcutils/Builder.java
@@ -8,7 +8,7 @@ package bjc.utils.funcutils;
* @param <E>
* The type of object being built.
*/
-public interface IBuilder<E> {
+public interface Builder<E> {
/**
* Build the object this builder is building.
*
diff --git a/base/src/main/java/bjc/utils/funcutils/Callables.java b/base/src/main/java/bjc/utils/funcutils/Callables.java
new file mode 100644
index 0000000..5895347
--- /dev/null
+++ b/base/src/main/java/bjc/utils/funcutils/Callables.java
@@ -0,0 +1,60 @@
+package bjc.utils.funcutils;
+
+import java.util.concurrent.*;
+import java.util.function.*;
+
+/**
+ * Utility function for dealing with callables and other things.
+ *
+ * @author Ben Culkin
+ *
+ */
+public class Callables
+{
+ /**
+ * Perform a 'bind' that appends a function to a callable.
+ *
+ * @param <Input> The type originally returned by the callable.
+ * @param <Output> The type returned by the function.
+ *
+ * @param call The original callable.
+ * @param func The function to use to transform the result.
+ *
+ * @return A callable which applies the given function to the result of them.
+ */
+ public static <Input, Output> Callable<Output> bind(
+ Callable<Input> call, Function<Input, Callable<Output>> func)
+ {
+ return () -> func.apply(call.call()).call();
+ }
+
+ /**
+ * Convert a normal function to a function on callables.
+ *
+ * @param <Input> The input to the function.
+ * @param <Output> The output from the function.
+ *
+ * @param func The function to convert.
+ *
+ * @return The function, made to work over callables.
+ */
+ public static <Input, Output> Function<Callable<Input>, Callable<Output>>
+ fmap(Function<Input, Output> func)
+ {
+ return (inp) -> () -> func.apply(inp.call());
+ }
+
+ /**
+ * Convert a future into a callable.
+ *
+ * @param <Output> The type returned by the future.
+ *
+ * @param fut The future to convert.
+ *
+ * @return A future which yields that value.
+ */
+ public static <Output> Callable<Output> obtain(Future<Output> fut)
+ {
+ return () -> fut.get();
+ }
+}
diff --git a/base/src/main/java/bjc/utils/funcutils/ChainIterator.java b/base/src/main/java/bjc/utils/funcutils/ChainIterator.java
new file mode 100644
index 0000000..36f94e5
--- /dev/null
+++ b/base/src/main/java/bjc/utils/funcutils/ChainIterator.java
@@ -0,0 +1,54 @@
+package bjc.utils.funcutils;
+
+import java.util.*;
+import java.util.function.*;
+
+/**
+ * A chain iterator. This is essentially flatMap in iterator form.
+ *
+ * @author bjculkin
+ *
+ * @param <T1>
+ * The type of the input values.
+ *
+ * @param <T2>
+ * The type of the output values.
+ */
+public class ChainIterator<T1, T2> implements Iterator<T2> {
+ private Iterator<T1> mainItr;
+ private Function<T1, Iterator<T2>> trans;
+
+ private Iterator<T2> curItr;
+
+ /**
+ * Create a new chain iterator.
+ *
+ * @param mainItr
+ * The main iterator for input.
+ *
+ * @param trans
+ * The transformation to use to produce the outputs.
+ */
+ public ChainIterator(Iterator<T1> mainItr, Function<T1, Iterator<T2>> trans) {
+ this.mainItr = mainItr;
+ this.trans = trans;
+ }
+
+ @Override
+ public boolean hasNext() {
+ if (curItr != null) {
+ return curItr.hasNext() ? true : mainItr.hasNext();
+ }
+
+ return mainItr.hasNext();
+ }
+
+ @Override
+ public T2 next() {
+ if (curItr == null || !curItr.hasNext()) {
+ curItr = trans.apply(mainItr.next());
+ }
+
+ return curItr.next();
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/funcutils/CollectorUtils.java b/base/src/main/java/bjc/utils/funcutils/CollectorUtils.java
index 81313c8..a92c2d1 100644
--- a/base/src/main/java/bjc/utils/funcutils/CollectorUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/CollectorUtils.java
@@ -2,8 +2,8 @@ package bjc.utils.funcutils;
import java.util.stream.Collector;
-import bjc.data.IHolder;
-import bjc.data.IPair;
+import bjc.data.Holder;
+import bjc.data.Pair;
/**
* Utilities for producing implementations of {@link Collector}
@@ -38,8 +38,8 @@ public class CollectorUtils {
* @return A collector that functions as mentioned above.
*/
public static <InitialType, AuxType1, AuxType2, FinalType1, FinalType2>
- Collector<InitialType, IHolder<IPair<AuxType1, AuxType2>>,
- IPair<FinalType1, FinalType2>>
+ Collector<InitialType, Holder<Pair<AuxType1, AuxType2>>,
+ Pair<FinalType1, FinalType2>>
compoundCollect(final Collector<InitialType, AuxType1, FinalType1> first,
final Collector<InitialType, AuxType2, FinalType2> second) {
return new CompoundCollector<>(first, second);
diff --git a/base/src/main/java/bjc/utils/funcutils/CompoundCollector.java b/base/src/main/java/bjc/utils/funcutils/CompoundCollector.java
index 5e51c20..5aa266e 100644
--- a/base/src/main/java/bjc/utils/funcutils/CompoundCollector.java
+++ b/base/src/main/java/bjc/utils/funcutils/CompoundCollector.java
@@ -7,10 +7,10 @@ import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Collector;
-import bjc.data.IHolder;
-import bjc.data.IPair;
-import bjc.data.Identity;
+import bjc.data.Holder;
import bjc.data.Pair;
+import bjc.data.Identity;
+import bjc.data.SimplePair;
/**
* Implementation of a collecter that uses two collectors.
@@ -18,8 +18,8 @@ import bjc.data.Pair;
* @author Ben Culkin
*/
final class CompoundCollector<InitialType, AuxType1, AuxType2, FinalType1, FinalType2>
- implements Collector<InitialType, IHolder<IPair<AuxType1, AuxType2>>,
- IPair<FinalType1, FinalType2>> {
+ implements Collector<InitialType, Holder<Pair<AuxType1, AuxType2>>,
+ Pair<FinalType1, FinalType2>> {
/* Our characteristics. */
private final Set<java.util.stream.Collector.Characteristics> characteristicSet;
@@ -48,7 +48,7 @@ final class CompoundCollector<InitialType, AuxType1, AuxType2, FinalType1, Final
}
@Override
- public BiConsumer<IHolder<IPair<AuxType1, AuxType2>>, InitialType> accumulator() {
+ public BiConsumer<Holder<Pair<AuxType1, AuxType2>>, InitialType> accumulator() {
final BiConsumer<AuxType1, InitialType> firstAccumulator = first.accumulator();
final BiConsumer<AuxType2, InitialType> secondAccumulator = second.accumulator();
@@ -68,7 +68,7 @@ final class CompoundCollector<InitialType, AuxType1, AuxType2, FinalType1, Final
}
@Override
- public BinaryOperator<IHolder<IPair<AuxType1, AuxType2>>> combiner() {
+ public BinaryOperator<Holder<Pair<AuxType1, AuxType2>>> combiner() {
final BinaryOperator<AuxType1> firstCombiner = first.combiner();
final BinaryOperator<AuxType2> secondCombiner = second.combiner();
@@ -80,25 +80,25 @@ final class CompoundCollector<InitialType, AuxType1, AuxType2, FinalType1, Final
}
@Override
- public Function<IHolder<IPair<AuxType1, AuxType2>>, IPair<FinalType1, FinalType2>>
+ public Function<Holder<Pair<AuxType1, AuxType2>>, Pair<FinalType1, FinalType2>>
finisher() {
return state -> state.unwrap(pair -> {
return pair.bind((left, right) -> {
final FinalType1 finalLeft = first.finisher().apply(left);
final FinalType2 finalRight = second.finisher().apply(right);
- return new Pair<>(finalLeft, finalRight);
+ return new SimplePair<>(finalLeft, finalRight);
});
});
}
@Override
- public Supplier<IHolder<IPair<AuxType1, AuxType2>>> supplier() {
+ public Supplier<Holder<Pair<AuxType1, AuxType2>>> supplier() {
return () -> {
final AuxType1 initialLeft = first.supplier().get();
final AuxType2 initialRight = second.supplier().get();
- return new Identity<>(new Pair<>(initialLeft, initialRight));
+ return new Identity<>(new SimplePair<>(initialLeft, initialRight));
};
}
}
diff --git a/base/src/main/java/bjc/utils/funcutils/EnumUtils.java b/base/src/main/java/bjc/utils/funcutils/EnumUtils.java
index e8898ca..6d53952 100644
--- a/base/src/main/java/bjc/utils/funcutils/EnumUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/EnumUtils.java
@@ -4,7 +4,7 @@ import java.util.Random;
import java.util.function.Consumer;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Utility methods on enums.
@@ -34,7 +34,7 @@ public class EnumUtils {
final int nValues, final Consumer<E> action, final Random rnd) {
final E[] enumValues = clasz.getEnumConstants();
- final IList<E> valueList = new FunctionalList<>(enumValues);
+ final ListEx<E> valueList = new FunctionalList<>(enumValues);
final int randomValueCount = enumValues.length - nValues;
diff --git a/base/src/main/java/bjc/utils/funcutils/FuncUtils.java b/base/src/main/java/bjc/utils/funcutils/FuncUtils.java
index 70e521a..2c65876 100644
--- a/base/src/main/java/bjc/utils/funcutils/FuncUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/FuncUtils.java
@@ -29,9 +29,12 @@ public class FuncUtils {
*
* @return The function transformed into a unary function returning a function.
*/
- public static <A, B, C> Function<A, Function<B, C>>
- curry2(final BiFunction<A, B, C> func) {
- return arg1 -> arg2 -> func.apply(arg1, arg2);
+ public static <A, B, C> Function<A, Function<B, C>> curry2(
+ final BiFunction<A, B, C> func)
+ {
+ return arg1 ->
+ arg2 ->
+ func.apply(arg1, arg2);
}
/**
@@ -43,14 +46,17 @@ public class FuncUtils {
* @param cons
* The action to perform.
*/
- public static void doTimes(final int nTimes, final Consumer<Integer> cons) {
- for (int i = 0; i < nTimes; i++) {
- cons.accept(i);
- }
+ public static void doTimes(
+ final int nTimes,
+ final Consumer<Integer> cons)
+ {
+ for (int i = 0; i < nTimes; i++) cons.accept(i);
}
/**
* Return an operator that executes until it converges.
+ *
+ * @param <T> The type the operator is on.
*
* @param op
* The operator to execute.
@@ -62,12 +68,15 @@ public class FuncUtils {
* @return The requested operator.
*/
public static <T> UnaryOperator<T> converge(final UnaryOperator<T> op,
- final int maxTries) {
+ final int maxTries)
+ {
return converge(op, Object::equals, maxTries);
}
/**
* Return an operator that executes until it converges.
+ *
+ * @param <T> The type the operator is on.
*
* @param op
* The operator to execute.
@@ -81,11 +90,14 @@ public class FuncUtils {
*
* @return The requested operator.
*/
- public static <T> UnaryOperator<T> converge(final UnaryOperator<T> op,
- final BiPredicate<T, T> converged, final int maxTries) {
+ public static <T> UnaryOperator<T> converge(
+ final UnaryOperator<T> op,
+ final BiPredicate<T, T> converged,
+ final int maxTries)
+ {
return val -> {
T newVal = op.apply(val);
- T oldVal;
+ T oldVal = newVal;
int tries = 0;
diff --git a/base/src/main/java/bjc/utils/funcutils/GroupPartIteration.java b/base/src/main/java/bjc/utils/funcutils/GroupPartIteration.java
deleted file mode 100644
index 681d707..0000000
--- a/base/src/main/java/bjc/utils/funcutils/GroupPartIteration.java
+++ /dev/null
@@ -1,86 +0,0 @@
-package bjc.utils.funcutils;
-
-import java.util.function.Consumer;
-import java.util.function.Function;
-
-import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
-
-/**
- * Implements a single group partitioning pass on a list.
- *
- * @author ben
- *
- * @param <E>
- * The type of element in the list being partitioned
- */
-final class GroupPartIteration<E> implements Consumer<E> {
- /* The list we're returning. */
- private final IList<IList<E>> returnedList;
-
- /* The current partition of the list. */
- public IList<E> currentPartition;
- /* The items rejected from the current partition. */
- private final IList<E> rejectedItems;
-
- /* The number of items in the current partition. */
- private int numberInCurrentPartition;
- /* The number of items in each partition. */
- private final int numberPerPartition;
-
- /* The function to use to count an item. */
- private final Function<E, Integer> elementCounter;
-
- /**
- * Create a new group partitioning iteration.
- *
- * @param returned
- * The list containing all of the existing partitions.
- *
- * @param rejects
- * The items that have been rejected from a partition for being
- * too large.
- *
- * @param nPerPart
- * The combined value of items that should go into each
- * partition.
- *
- * @param eleCount
- * The function to use to determine the value of an item.
- */
- public GroupPartIteration(final IList<IList<E>> returned, final IList<E> rejects,
- final int nPerPart, final Function<E, Integer> eleCount) {
- this.returnedList = returned;
- this.rejectedItems = rejects;
- this.numberPerPartition = nPerPart;
- this.elementCounter = eleCount;
-
- this.currentPartition = new FunctionalList<>();
- this.numberInCurrentPartition = 0;
- }
-
- @Override
- public void accept(final E value) {
- final boolean shouldStartPartition
- = numberInCurrentPartition >= numberPerPartition;
-
- if (shouldStartPartition) {
- returnedList.add(currentPartition);
-
- currentPartition = new FunctionalList<>();
- numberInCurrentPartition = 0;
- } else {
- final int currentElementCount = elementCounter.apply(value);
-
- final boolean shouldReject = (numberInCurrentPartition + currentElementCount)
- >= numberPerPartition;
-
- if (shouldReject) {
- rejectedItems.add(value);
- } else {
- currentPartition.add(value);
- numberInCurrentPartition += currentElementCount;
- }
- }
- }
-}
diff --git a/base/src/main/java/bjc/utils/funcutils/Isomorphism.java b/base/src/main/java/bjc/utils/funcutils/Isomorphism.java
deleted file mode 100644
index c219d7f..0000000
--- a/base/src/main/java/bjc/utils/funcutils/Isomorphism.java
+++ /dev/null
@@ -1,59 +0,0 @@
-package bjc.utils.funcutils;
-
-import java.util.function.Function;
-
-/**
- * A pair of functions to transform between a pair of types.
- *
- * @author bjculkin
- *
- * @param <S>
- * The source type of the isomorphism.
- *
- * @param <D>
- * The destination type of isomorphism.
- */
-public class Isomorphism<S, D> {
- /* The function to the destination type. */
- private Function<S, D> toFunc;
- /* The function to the source type. */
- private Function<D, S> fromFunc;
-
- /**
- * Create a new isomorphism.
- *
- * @param to
- * The 'forward' function, from the source to the definition.
- *
- * @param from
- * The 'backward' function, from the definition to the source.
- */
- public Isomorphism(Function<S, D> to, Function<D, S> from) {
- toFunc = to;
- fromFunc = from;
- }
-
- /**
- * Apply the isomorphism forward.
- *
- * @param val
- * The source value.
- *
- * @return The destination value.
- */
- public D to(S val) {
- return toFunc.apply(val);
- }
-
- /**
- * Apply the isomorphism backward.
- *
- * @param val
- * The destination value.
- *
- * @return The source value.
- */
- public S from(D val) {
- return fromFunc.apply(val);
- }
-}
diff --git a/base/src/main/java/bjc/utils/funcutils/IteratorUtils.java b/base/src/main/java/bjc/utils/funcutils/IteratorUtils.java
index 8d51996..662b1bf 100644
--- a/base/src/main/java/bjc/utils/funcutils/IteratorUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/IteratorUtils.java
@@ -3,7 +3,7 @@ package bjc.utils.funcutils;
import java.util.*;
import java.util.function.*;
-import bjc.data.ArrayIterator;
+import bjc.data.*;
/**
* Utility methods for dealing with iterators.
@@ -13,57 +13,9 @@ import bjc.data.ArrayIterator;
*/
public class IteratorUtils {
/**
- * A chain iterator. This is essentially flatMap in iterator form.
- *
- * @author bjculkin
- *
- * @param <T1>
- * The type of the input values.
- *
- * @param <T2>
- * The type of the output values.
- */
- public static class ChainIterator<T1, T2> implements Iterator<T2> {
- private Iterator<T1> mainItr;
- private Function<T1, Iterator<T2>> trans;
-
- private Iterator<T2> curItr;
-
- /**
- * Create a new chain iterator.
- *
- * @param mainItr
- * The main iterator for input.
- *
- * @param trans
- * The transformation to use to produce the outputs.
- */
- public ChainIterator(Iterator<T1> mainItr, Function<T1, Iterator<T2>> trans) {
- this.mainItr = mainItr;
- this.trans = trans;
- }
-
- @Override
- public boolean hasNext() {
- if (curItr != null) {
- return curItr.hasNext() ? true : mainItr.hasNext();
- }
-
- return mainItr.hasNext();
- }
-
- @Override
- public T2 next() {
- if (curItr == null || !curItr.hasNext()) {
- curItr = trans.apply(mainItr.next());
- }
-
- return curItr.next();
- }
- }
-
- /**
* Convert an iterator to an iterable.
+ *
+ * @param <E> The type being iterated over.
*
* @param itr
* The iterator to convert.
@@ -77,6 +29,8 @@ public class IteratorUtils {
/**
* Convert an iterable to an iterator.
*
+ * @param <E> The type being iterated over.
+ *
* @param itr
* The iterable to convert.
*
@@ -89,18 +43,23 @@ public class IteratorUtils {
/**
* Convert an array to an iterator.
*
+ * @param <E> The type being iterated over.
+ *
* @param parms
* The array to iterate over.
*
* @return An iterator over the provided array.
*/
@SafeVarargs
- public static <E> Iterator<E> AI(E... parms) {
+ public static <E> Iterator<E> I(E... parms) {
return new ArrayIterator<>(parms);
}
/**
* Create a chain iterator.
+ *
+ * @param <A> The initial type being iterated over.
+ * @param <B> The resulting type being iterated over.
*
* @param itrA
* The iterator for input values.
@@ -114,4 +73,43 @@ public class IteratorUtils {
Function<A, Iterator<B>> itrB) {
return new ChainIterator<>(itrA, itrB);
}
+
+ /**
+ * Perform a left-fold over an iterator.
+ *
+ * @param <ElementType> The type of elements in the iterator.
+ * @param <ResultType> The result from the fold.
+ *
+ * @param itr The items to iterate over.
+ * @param zero The initial element for the fold.
+ * @param folder The function that does the folding.
+ *
+ * @return The result of folding over the iterator.
+ */
+ public static <ElementType, ResultType> ResultType foldLeft(
+ Iterable<ElementType> itr,
+ ResultType zero,
+ BiFunction<ElementType, ResultType, ResultType> folder)
+ {
+ ResultType state = zero;
+ for (ElementType elem : itr) {
+ state = folder.apply(elem, state);
+ }
+ return state;
+ }
+
+ /**
+ * Creates an 'entangled' pair of a consumer and an iterator.
+ *
+ * @param <ElementType> The type of value involved.
+ *
+ * @return A pair consisting of a consumer of values, and an iterator that
+ * yields the consumed values.
+ */
+ public static <ElementType>
+ Pair<Consumer<ElementType>, Iterator<ElementType>> entangle()
+ {
+ Queue<ElementType> backer = new ArrayDeque<>();
+ return Pair.pair(backer::add, new QueueBackedIterator<>(backer));
+ }
}
diff --git a/base/src/main/java/bjc/utils/funcutils/LambdaLock.java b/base/src/main/java/bjc/utils/funcutils/LambdaLock.java
index f3637f9..c7ba09a 100644
--- a/base/src/main/java/bjc/utils/funcutils/LambdaLock.java
+++ b/base/src/main/java/bjc/utils/funcutils/LambdaLock.java
@@ -36,8 +36,9 @@ public class LambdaLock {
/**
* Execute an action with the read lock taken.
*
- * @param supp
- * The action to call.
+ * @param <T> The type returned by the action.
+ *
+ * @param supp The action to call.
*
* @return The result of the action.
*/
@@ -53,9 +54,10 @@ public class LambdaLock {
/**
* Execute an action with the write lock taken.
- *
- * @param supp
- * The action to call.
+ *
+ * @param <T> The type returned by the action.
+ *
+ * @param supp The action to call.
*
* @return The result of the action.
*/
diff --git a/base/src/main/java/bjc/utils/funcutils/ListUtils.java b/base/src/main/java/bjc/utils/funcutils/ListUtils.java
index f689d6c..ab32ccc 100644
--- a/base/src/main/java/bjc/utils/funcutils/ListUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/ListUtils.java
@@ -1,14 +1,10 @@
package bjc.utils.funcutils;
-import java.util.ArrayList;
-import java.util.Iterator;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.function.Function;
-import java.util.function.Supplier;
+import java.util.*;
+import java.util.function.*;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Utilities for manipulating FunctionalLists and regular Collections lists that
@@ -28,9 +24,8 @@ public class ListUtils {
*
* @return The collapsed string of tokens.
*/
- public static String collapseTokens(final IList<String> input) {
- if (input == null)
- throw new NullPointerException("Input must not be null");
+ public static String collapseTokens(final ListEx<String> input) {
+ if (input == null) throw new NullPointerException("Input must not be null");
return collapseTokens(input, "");
}
@@ -47,13 +42,10 @@ public class ListUtils {
*
* @return The collapsed string of tokens.
*/
- public static String collapseTokens(final IList<String> input,
+ public static String collapseTokens(final ListEx<String> input,
final String seperator) {
- if (input == null) {
- throw new NullPointerException("Input must not be null");
- } else if (seperator == null) {
- throw new NullPointerException("Seperator must not be null");
- }
+ if (input == null) throw new NullPointerException("Input must not be null");
+ else if (seperator == null) throw new NullPointerException("Seperator must not be null");
if (input.getSize() < 1) {
return "";
@@ -66,9 +58,7 @@ public class ListUtils {
for (final String itm : input.toIterable()) {
state.append(itm);
- if (i != input.getSize()) {
- state.append(seperator);
- }
+ if (i != input.getSize()) state.append(seperator);
i += 1;
}
@@ -96,9 +86,9 @@ public class ListUtils {
* @return A new list containing the desired number of items randomly selected
* from the specified list without replacement.
*/
- public static <E> IList<E> drawWithoutReplacement(final IList<E> list,
+ public static <E> ListEx<E> drawWithoutReplacement(final ListEx<E> list,
final int number, final Function<Integer, Integer> rng) {
- final IList<E> selected = new FunctionalList<>(new ArrayList<>(number));
+ final ListEx<E> selected = new FunctionalList<>(new ArrayList<>(number));
final int total = list.getSize();
@@ -140,13 +130,11 @@ public class ListUtils {
* @return A new list containing the desired number of items randomly selected
* from the specified list.
*/
- public static <E> IList<E> drawWithReplacement(final IList<E> list, final int number,
+ public static <E> ListEx<E> drawWithReplacement(final ListEx<E> list, final int number,
final Function<Integer, Integer> rng) {
- final IList<E> selected = new FunctionalList<>(new ArrayList<>(number));
+ final ListEx<E> selected = new FunctionalList<>(new ArrayList<>(number));
- for (int i = 0; i < number; i++) {
- selected.add(list.randItem(rng));
- }
+ for (int i = 0; i < number; i++) selected.add(list.randItem(rng));
return selected;
}
@@ -170,7 +158,7 @@ public class ListUtils {
*
* @return A list partitioned according to the above rules.
*/
- public static <E> IList<IList<E>> groupPartition(final IList<E> input,
+ public static <E> ListEx<ListEx<E>> groupPartition(final ListEx<E> input,
final Function<E, Integer> counter, final int partitionSize) {
if (input == null) {
throw new NullPointerException("Input list must not be null");
@@ -185,10 +173,10 @@ public class ListUtils {
}
/* List that holds our results. */
- final IList<IList<E>> returned = new FunctionalList<>();
+ final ListEx<ListEx<E>> returned = new FunctionalList<>();
/* List that holds elements rejected during current pass. */
- final IList<E> rejected = new FunctionalList<>();
+ final ListEx<E> rejected = new FunctionalList<>();
final GroupPartIteration<E> it
= new GroupPartIteration<>(returned, rejected, partitionSize, counter);
@@ -199,10 +187,8 @@ public class ListUtils {
numberOfIterations++) {
input.forEach(it);
- if (rejected.isEmpty()) {
- /* Nothing was rejected, so we're done. */
- return returned;
- }
+ /* Nothing was rejected, so we're done. */
+ if (rejected.isEmpty()) return returned;
}
final String fmt
@@ -226,13 +212,11 @@ public class ListUtils {
* @return A list containing all the elements of the lists.
*/
@SafeVarargs
- public static <E> IList<E> mergeLists(final IList<E>... lists) {
- final IList<E> returned = new FunctionalList<>();
+ public static <E> ListEx<E> mergeLists(final ListEx<E>... lists) {
+ final ListEx<E> returned = new FunctionalList<>();
- for (final IList<E> list : lists) {
- for (final E itm : list.toIterable()) {
- returned.add(itm);
- }
+ for (final ListEx<E> list : lists) {
+ for (final E itm : list.toIterable()) returned.add(itm);
}
return returned;
@@ -262,12 +246,12 @@ public class ListUtils {
* If the list couldn't be padded to the
* desired size.
*/
- public static <E> IList<E> padList(final IList<E> list,
+ public static <E> ListEx<E> padList(final ListEx<E> list,
final Function<E, Integer> counter, final int size,
final Supplier<E> padder) {
int count = 0;
- final IList<E> returned = new FunctionalList<>();
+ final ListEx<E> returned = new FunctionalList<>();
for (final E itm : list.toIterable()) {
count += counter.apply(itm);
@@ -334,8 +318,9 @@ public class ListUtils {
*
* This is a version of Algorith P (Plain Changes) from Knuth (vol 4A, pg 322)
*
- * @param list
- * The list to generate permutations from.
+ * @param <T> The type of elements in the list.
+ *
+ * @param list The list to generate permutations from.
* @return The list of permutations of the list.
*/
public static <T> List<List<T>> permuteList(List<T> list) {
@@ -344,9 +329,7 @@ public class ListUtils {
/*
* Special-case small usages.
*/
- if (list.size() == 0) {
- return permutes;
- }
+ if (list.size() == 0) return permutes;
if (list.size() == 1) {
permutes.add(list);
@@ -358,19 +341,8 @@ public class ListUtils {
T elm1 = list.get(0);
T elm2 = list.get(1);
- List<T> currPerm = new ArrayList<>(2);
-
- currPerm.add(elm1);
- currPerm.add(elm2);
-
- permutes.add(currPerm);
-
- currPerm = new ArrayList<>(2);
-
- currPerm.add(elm2);
- currPerm.add(elm1);
-
- permutes.add(currPerm);
+ permutes.add(Arrays.asList(elm1, elm2));
+ permutes.add(Arrays.asList(elm2, elm1));
return permutes;
}
@@ -384,9 +356,7 @@ public class ListUtils {
}
List<T> currentPermute = new ArrayList<>(list.size());
- for (T elm : list) {
- currentPermute.add(elm);
- }
+ for (T elm : list) currentPermute.add(elm);
permutes.add(currentPermute);
int j = list.size() - 1;
@@ -422,9 +392,7 @@ public class ListUtils {
auxC[j] = q;
currentPermute = new ArrayList<>(list.size());
- for (T elm : list) {
- currentPermute.add(elm);
- }
+ for (T elm : list) currentPermute.add(elm);
permutes.add(currentPermute);
j = list.size() - 1;
@@ -441,3 +409,82 @@ public class ListUtils {
list.set(b, tmp);
}
}
+
+/**
+ * Implements a single group partitioning pass on a list.
+ *
+ * @author ben
+ *
+ * @param <E>
+ * The type of element in the list being partitioned
+ */
+class GroupPartIteration<E> implements Consumer<E> {
+ /* The list we're returning. */
+ private final ListEx<ListEx<E>> returnedList;
+
+ /* The current partition of the list. */
+ public ListEx<E> currentPartition;
+ /* The items rejected from the current partition. */
+ private final ListEx<E> rejectedItems;
+
+ /* The number of items in the current partition. */
+ private int numberInCurrentPartition;
+ /* The number of items in each partition. */
+ private final int numberPerPartition;
+
+ /* The function to use to count an item. */
+ private final Function<E, Integer> elementCounter;
+
+ /**
+ * Create a new group partitioning iteration.
+ *
+ * @param returned
+ * The list containing all of the existing partitions.
+ *
+ * @param rejects
+ * The items that have been rejected from a partition for being
+ * too large.
+ *
+ * @param nPerPart
+ * The combined value of items that should go into each
+ * partition.
+ *
+ * @param eleCount
+ * The function to use to determine the value of an item.
+ */
+ public GroupPartIteration(final ListEx<ListEx<E>> returned, final ListEx<E> rejects,
+ final int nPerPart, final Function<E, Integer> eleCount) {
+ this.returnedList = returned;
+ this.rejectedItems = rejects;
+ this.numberPerPartition = nPerPart;
+ this.elementCounter = eleCount;
+
+ this.currentPartition = new FunctionalList<>();
+ this.numberInCurrentPartition = 0;
+ }
+
+ @Override
+ public void accept(final E value) {
+ final boolean shouldStartPartition
+ = numberInCurrentPartition >= numberPerPartition;
+
+ if (shouldStartPartition) {
+ returnedList.add(currentPartition);
+
+ currentPartition = new FunctionalList<>();
+ numberInCurrentPartition = 0;
+ } else {
+ final int currentElementCount = elementCounter.apply(value);
+
+ final boolean shouldReject = (numberInCurrentPartition + currentElementCount)
+ >= numberPerPartition;
+
+ if (shouldReject) {
+ rejectedItems.add(value);
+ } else {
+ currentPartition.add(value);
+ numberInCurrentPartition += currentElementCount;
+ }
+ }
+ }
+}
diff --git a/base/src/main/java/bjc/utils/funcutils/QueueBackedIterator.java b/base/src/main/java/bjc/utils/funcutils/QueueBackedIterator.java
new file mode 100644
index 0000000..8b9f401
--- /dev/null
+++ b/base/src/main/java/bjc/utils/funcutils/QueueBackedIterator.java
@@ -0,0 +1,36 @@
+package bjc.utils.funcutils;
+
+import java.util.*;
+
+/**
+ * An iterator backed by a queue.
+ *
+ * @author Ben Culkin
+ *
+ * @param <ElementType> The type of element
+ */
+public class QueueBackedIterator<ElementType>
+ implements Iterator<ElementType>
+{
+ private final Queue<ElementType> backer;
+
+ /**
+ * Create a new queue-backed iterator.
+ *
+ * @param backer The queue which backs this iterator.
+ */
+ public QueueBackedIterator(Queue<ElementType> backer)
+ {
+ this.backer = backer;
+ }
+
+ @Override
+ public boolean hasNext() {
+ return !backer.isEmpty();
+ }
+
+ @Override
+ public ElementType next() {
+ return backer.remove();
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/funcutils/SetUtils.java b/base/src/main/java/bjc/utils/funcutils/SetUtils.java
index 83c191b..b721d10 100644
--- a/base/src/main/java/bjc/utils/funcutils/SetUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/SetUtils.java
@@ -15,8 +15,8 @@ public class SetUtils {
/**
* Create a power-set (set of all subsets) of a given set.
*
- * @param originalSet
- * The set to create a power-set of.
+ * @param <T> The type of elements contained in the set.
+ * @param originalSet The set to create a power-set of.
* @return The power-set of the set.
*/
public static <T> Set<Set<T>> powerSet(Set<T> originalSet) {
@@ -55,17 +55,15 @@ public class SetUtils {
/**
* Utility method for set construction.
*
- * @param elms
- * The elements to stick in the set.
+ * @param <T> The type of elements in the set.
+ * @param elms The elements to stick in the set.
* @return A set containing the specified elements.
*/
@SafeVarargs
public static <T> Set<T> toSet(T... elms) {
Set<T> set = new HashSet<>();
- for (T elm : elms) {
- set.add(elm);
- }
+ for (T elm : elms) set.add(elm);
return set;
}
diff --git a/base/src/main/java/bjc/utils/funcutils/Strategy.java b/base/src/main/java/bjc/utils/funcutils/Strategy.java
new file mode 100644
index 0000000..316879f
--- /dev/null
+++ b/base/src/main/java/bjc/utils/funcutils/Strategy.java
@@ -0,0 +1,81 @@
+package bjc.utils.funcutils;
+
+import java.util.concurrent.*;
+import java.util.function.*;
+
+/**
+ * Strategy for dealing with parallel execution.
+ *
+ * @author Ben Culkin
+ *
+ * @param <Output> The type returned by the tasks.
+ *
+ */
+public interface Strategy<Output> extends Function<Callable<Output>, Future<Output>>
+{
+ /**
+ * Convert a function into one which operates concurrently, using this strategy.
+ *
+ * @param <Input> The type of the function argument.
+ *
+ * @param func The type of the function.
+ *
+ * @return A function which executes concurrently.
+ */
+ public default <Input> Function<Input, Future<Output>> using(Function<Input, Output> func)
+ {
+ return (input) -> this.apply(() -> func.apply(input));
+ }
+
+ /**
+ * A strategy which will run tasks in serial.
+ *
+ * @param <Output> The type returned by the task.
+ *
+ * @return A strategy which executes things serially.
+ */
+ public static <Output> Strategy<Output> serial()
+ {
+ return (call) -> {
+ FutureTask<Output> task = new FutureTask<>(call);
+ task.run();
+ return task;
+ };
+ }
+ /**
+ * A strategy which creates a fresh thread to execute a task on.
+ *
+ * @param <Output> The type returned by the task.
+ *
+ * @return A strategy which uses threads to create tasks.
+ */
+ public static <Output> Strategy<Output> simpleThread()
+ {
+ // I leave this as an example as of what is possible with combinators.
+ // return (call) -> invoke(introducing(
+ // () -> new FutureTask<>(call),
+ // (task, input) -> doWith(
+ // (FutureTask<Output> tsk) ->
+ // new Thread(task).start()).apply(task)
+ // ));
+ return (call) -> {
+ FutureTask<Output> task = new FutureTask<>(call);
+ new Thread(task).start();
+ return task;
+ };
+ }
+
+ /**
+ * A strategy that uses an executor service.
+ *
+ * @param <Output> The type returned by the task.
+ *
+ * @param svc The executor service to use.
+ *
+ * @return A strategy which uses the provided executor.
+ */
+ public static <Output> Strategy<Output> executorService(ExecutorService svc)
+ {
+ return svc::submit;
+ }
+}
diff --git a/base/src/main/java/bjc/utils/funcutils/StringUtils.java b/base/src/main/java/bjc/utils/funcutils/StringUtils.java
index b7a6835..0b57e18 100644
--- a/base/src/main/java/bjc/utils/funcutils/StringUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/StringUtils.java
@@ -1,79 +1,53 @@
package bjc.utils.funcutils;
-import java.util.ArrayList;
-import java.util.Deque;
-import java.util.Iterator;
-import java.util.List;
-import java.util.Scanner;
-
+import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import com.ibm.icu.text.BreakIterator;
-import bjc.data.BooleanToggle;
import bjc.utils.ioutils.LevelSplitter;
-import bjc.utils.parserutils.TokenUtils;
-/**
- * Utility methods for operations on strings.
+/** Utility methods for operations on strings.
*
- * @author ben
- */
+ * @author ben */
public class StringUtils {
- /**
- * Check if a string consists only of one or more matches of a regular
+ /**Check if a string consists only of one or more matches of a regular
* expression.
*
- * @param input
- * The string to check.
- *
- * @param rRegex
- * The regex to see if the string only contains matches of.
+ * @param input The string to check.
+ * @param rRegex The regex to see if the string only contains matches of.
*
* @return Whether or not the string consists only of multiple matches of the
* provided regex.
*/
public static boolean containsOnly(final String input, final String rRegex) {
- if (input == null)
- throw new NullPointerException("Input must not be null");
- else if (rRegex == null)
- throw new NullPointerException("Regex must not be null");
+ if (input == null) throw new NullPointerException("Input must not be null");
+ else if (rRegex == null) throw new NullPointerException("Regex must not be null");
- /*
- * This regular expression is fairly simple.
+ /* This regular expression is fairly simple.
*
* First, we match the beginning of the string. Then, we start a non-capturing
* group whose contents are the passed in regex. That group is then matched one
- * or more times and the pattern matches to the end of the string.
- */
+ * or more times and the pattern matches to the end of the string. */
return input.matches("\\A(?:" + rRegex + ")+\\Z");
}
- /**
- * Indent the string being built in a StringBuilder n levels.
- *
- * @param builder
- * The builder to indent in.
+ /** Indent the string being built in a StringBuilder n levels.
*
- * @param levels
- * The number of levels to indent.
+ * @param builder The builder to indent in.
+ * @param levels The number of levels to indent.
*/
public static void indentNLevels(final StringBuilder builder, final int levels) {
- for (int i = 0; i < levels; i++) {
- builder.append("\t");
- }
+ for (int i = 0; i < levels; i++) builder.append("\t");
}
- /**
- * Print out a deque with a special case for easily showing a deque is empty.
- *
- * @param <ContainedType>
- * The type in the deque.
+ /** Print out a deque with a special case for easily showing a deque is empty.
*
- * @param queue
- * The deque to print.
+ * @param <ContainedType> The type in the deque.
*
+ * @param queue The deque to print.
+ *
* @return A string version of the deque, with allowance for an empty deque.
*/
public static <ContainedType> String printDeque(final Deque<ContainedType> queue) {
@@ -83,15 +57,10 @@ public class StringUtils {
/**
* Converts a sequence to an English list.
*
- * @param objects
- * The sequence to convert to an English list.
- *
+ * @param objects The sequence to convert to an English list.
* @param join
- * The string to use for separating the last element from the
- * rest.
- *
- * @param comma
- * The string to use as a comma
+ * The string to use for separating the last element from the rest.
+ * @param comma The string to use as a comma
*
* @return The sequence as an English list.
*/
@@ -140,15 +109,11 @@ public class StringUtils {
return sb.toString();
}
- /**
- * Converts a sequence to an English list.
- *
- * @param objects
- * The sequence to convert to an English list.
+ /** Converts a sequence to an English list.
*
+ * @param objects The sequence to convert to an English list.
* @param join
- * The string to use for separating the last element from the
- * rest.
+ * The string to use for separating the last element from the rest.
*
* @return The sequence as an English list.
*/
@@ -156,30 +121,21 @@ public class StringUtils {
return toEnglishList(objects, join, ",");
}
- /**
- * Converts a sequence to an English list.
- *
- * @param objects
- * The sequence to convert to an English list.
+ /** Converts a sequence to an English list.
*
- * @param and
- * Whether to use 'and' or 'or'.
+ * @param objects The sequence to convert to an English list.
+ * @param and Whether to use 'and' or 'or'.
*
* @return The sequence as an English list.
*/
public static String toEnglishList(final Object[] objects, final boolean and) {
- if (and) {
- return toEnglishList(objects, "and");
- }
-
- return toEnglishList(objects, "or");
+ if (and) return toEnglishList(objects, "and");
+ else return toEnglishList(objects, "or");
}
- /**
- * Count the number of graphemes in a string.
+ /** Count the number of graphemes in a string.
*
- * @param value
- * The string to check.
+ * @param value The string to check.
*
* @return The number of graphemes in the string.
*/
@@ -188,97 +144,57 @@ public class StringUtils {
it.setText(value);
int count = 0;
- while (it.next() != BreakIterator.DONE) {
- count++;
- }
+ while (it.next() != BreakIterator.DONE) count++;
return count;
}
- /**
- * Count the number of times a pattern matches in a given string.
- *
- * @param value
- * The string to count occurances in.
+ /** Count the number of times a pattern matches in a given string.
*
- * @param pattern
- * The pattern to count occurances of.
+ * @param value The string to count occurrences in.
+ * @param pattern The pattern to count occurrences of.
+ *
* @return The number of times the pattern matches.
*/
public static int countMatches(final String value, final String pattern) {
Matcher mat = Pattern.compile(pattern).matcher(value);
int num = 0;
- while (mat.find())
- num += 1;
+ while (mat.find()) num += 1;
return num;
}
- /**
- * Get a substring until a specified string.
+ /** Get a substring until a specified string.
*
- * @param strang
- * The string to substring.
- * @param vx
- * The place to substring until.
+ * @param strang The string to substring.
+ * @param until The place to substring until.
*
* @return The specified substring.
*/
- public static String substringTo(String strang, String vx) {
- return substringTo(strang, vx, true);
+ public static String substringTo(String strang, String until) {
+ return substringTo(strang, until, true);
}
/**
* Get a substring until a specified string.
*
- * @param strang
- * The string to substring.
- * @param vx
- * The place to substring until.
- * @param allowFail
- * Whether or not to allow failure.
+ * @param strang The string to substring.
+ * @param until The place to substring until.
+ * @param allowFail Whether or not to allow failure.
*
* @return The specified substring, or null if the specified place to substring
* to was not found, and we were not allowed to fail.
*/
- public static String substringTo(String strang, String vx, boolean allowFail) {
- int idx = strang.indexOf(vx);
+ public static String substringTo(String strang, String until, boolean allowFail) {
+ int idx = strang.indexOf(until);
if (idx == -1) {
- if (allowFail)
- return strang;
-
- return null;
- }
-
- return strang.substring(0, strang.indexOf(vx));
- }
-
- /**
- * Split a line into a series of space-separated arguments, including string
- * literals.
- *
- * @param com
- * The command to split from
- * @return The split arguments.
- */
- public static List<String> processArguments(String com) {
- List<String> strings = new ArrayList<>();
-
- BooleanToggle togg = new BooleanToggle();
-
- for (String strang : TokenUtils.removeDQuotedStrings(com)) {
- if (togg.get()) {
- strings.add(TokenUtils.descapeString(strang));
- } else {
- for (String strung : strang.split("\\s+")) {
- strings.add(strung);
- }
- }
+ if (allowFail) return strang;
+ else return null;
}
- return strings;
+ return strang.substring(0, strang.indexOf(until));
}
private static class LineIterator implements Iterator<String> {
@@ -306,16 +222,13 @@ public class StringUtils {
boolean doLoop = true;
do {
- if (!scn.hasNextLine())
- break;
+ if (!scn.hasNextLine()) break;
tmp = scn.nextLine().trim();
// Skip blank lines
- if (skipBlanks && tmp.equals(""))
- continue;
- if (processComments && tmp.startsWith(commentInd))
- continue;
+ if (skipBlanks && tmp.equals("")) continue;
+ if (processComments && tmp.startsWith(commentInd)) continue;
doLoop = tmp.endsWith("\\") && !tmp.endsWith("\\\\");
@@ -330,11 +243,9 @@ public class StringUtils {
}
}
- /**
- * Read a series of lines from an input source.
+ /** Read a series of lines from an input source.
*
- * @param scn
- * The source to read the lines from.
+ * @param scn The source to read the lines from.
*
* @return An iterator over the lines from the input source.
*/
@@ -345,17 +256,10 @@ public class StringUtils {
/**
* Read a series of lines from an input source.
*
- * @param scn
- * The source to read the lines from.
- *
- * @param processComments
- * Whether or not to skip comment lines.
- *
- * @param commentInd
- * Indicator for starting comment lines.
- *
- * @param skipBlanks
- * Whether or not to skip blank lines.
+ * @param scn The source to read the lines from.
+ * @param processComments Whether or not to skip comment lines.
+ * @param commentInd Indicator for starting comment lines.
+ * @param skipBlanks Whether or not to skip blank lines.
*
* @return An iterator over the lines from the input source.
*/
@@ -371,27 +275,23 @@ public class StringUtils {
return itr;
}
- /**
- * Check if a string contains any one of a specified number of things,
+ /** Check if a string contains any one of a specified number of things,
* respecting groups.
*
- * @param haystack
- * The string to look in.
- * @param needles
- * The strings to look for.
+ * @param haystack The string to look in.
+ * @param needles The strings to look for.
+ *
* @return Whether or not any of the strings were contained outside of groups.
*/
public static boolean levelContains(String haystack, String... needles) {
return LevelSplitter.def.levelContains(haystack, needles);
}
- /**
- * Split a string, respecting groups.
+ /** Split a string, respecting groups.
*
- * @param phrase
- * The string to split.
- * @param splits
- * The strings to split on.
+ * @param phrase The string to split.
+ * @param splits The strings to split on.
+ *
* @return A list of split strings. If keepDelims is true, it also includes the
* delimiters in between the split strings.
*/
@@ -399,15 +299,12 @@ public class StringUtils {
return LevelSplitter.def.levelSplit(phrase, false, splits);
}
- /**
- * Split a string, respecting groups.
+ /** Split a string, respecting groups.
*
- * @param phrase
- * The string to split.
- * @param keepDelims
- * Whether or not to include the delimiters in the results.
- * @param splits
- * The strings to split on.
+ * @param phrase The string to split.
+ * @param keepDelims Whether or not to include the delimiters in the results.
+ * @param splits The strings to split on.
+ *
* @return A list of split strings. If keepDelims is true, it also includes the
* delimiters in between the split strings.
*/
@@ -415,4 +312,4 @@ public class StringUtils {
String... splits) {
return LevelSplitter.def.levelSplit(phrase, keepDelims, splits);
}
-}
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/funcutils/TestUtils.java b/base/src/main/java/bjc/utils/funcutils/TestUtils.java
index 3aa20a2..bf38cbe 100644
--- a/base/src/main/java/bjc/utils/funcutils/TestUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/TestUtils.java
@@ -15,27 +15,22 @@ public class TestUtils {
/**
* Assert an iterator provides a particular sequence of values.
*
- * @param src
- * The iterator to pull values from.
- * @param vals
- * The values to expect from the iterator.
+ * @param <T> The type of the values.
+ * @param src The iterator to pull values from.
+ * @param vals The values to expect from the iterator.
*/
@SafeVarargs
public static <T> void assertIteratorEquals(Iterator<T> src, T... vals) {
- for (T val : vals) {
- assertEquals(val, src.next());
- }
+ for (T val : vals) assertEquals(val, src.next());
}
/**
* Assert an iterator provides a particular sequence of values.
*
- * @param src
- * The iterator to pull values from.
- * @param hasMore
- * The expected value of hasNext for the iterator.
- * @param vals
- * The values to expect from the iterator.
+ * @param <T> The type of the values.
+ * @param src The iterator to pull values from.
+ * @param hasMore The expected value of hasNext for the iterator.
+ * @param vals The values to expect from the iterator.
*/
@SafeVarargs
public static <T> void assertIteratorEquals(boolean hasMore, Iterator<T> src,
@@ -46,6 +41,8 @@ public class TestUtils {
* Even though it's awkward, the boolean has to come first. Otherwise, there are
* cases where the compiler will get confused as to what the right value for T
* is, and be unable to pick an overload.
+ *
+ * This is a case where named parameter would be rather useful.
*/
assertIteratorEquals(src, vals);
@@ -55,11 +52,9 @@ public class TestUtils {
/**
* Assert that a list has a given set of contents.
*
- * @param src
- * The list of actual elements.
- *
- * @param exps
- * The list of expected elements.
+ * @param <T> The type of the values.
+ * @param src The list of actual elements.
+ * @param exps The list of expected elements.
*/
@SafeVarargs
public static <T> void assertListEquals(List<T> src, T... exps) {
diff --git a/base/src/main/java/bjc/utils/funcutils/TreeUtils.java b/base/src/main/java/bjc/utils/funcutils/TreeUtils.java
index d525773..5a1d9c8 100644
--- a/base/src/main/java/bjc/utils/funcutils/TreeUtils.java
+++ b/base/src/main/java/bjc/utils/funcutils/TreeUtils.java
@@ -1,11 +1,10 @@
package bjc.utils.funcutils;
import java.util.LinkedList;
-import java.util.function.Predicate;
+import java.util.function.*;
-import bjc.data.ITree;
-import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.data.*;
+import bjc.funcdata.*;
/**
* Implements various utilities for trees.
@@ -16,14 +15,13 @@ public class TreeUtils {
/**
* Convert a tree into a list of outline nodes that match a certain path.
*
- * @param tre
- * The tree to outline.
- * @param leafMarker
- * The path to mark nodes with.
+ * @param <T> The type of the values.
+ * @param tre The tree to outline.
+ * @param leafMarker The path to mark nodes with.
* @return The list of marked paths.
*/
- public static <T> IList<IList<T>> outlineTree(ITree<T> tre, Predicate<T> leafMarker) {
- IList<IList<T>> paths = new FunctionalList<>();
+ public static <T> ListEx<ListEx<T>> outlineTree(Tree<T> tre, Predicate<T> leafMarker) {
+ ListEx<ListEx<T>> paths = new FunctionalList<>();
LinkedList<T> path = new LinkedList<>();
path.add(tre.getHead());
@@ -34,15 +32,13 @@ public class TreeUtils {
}
/* Find a path in a tree. */
- private static <T> void findPath(ITree<T> subtree, LinkedList<T> path,
- Predicate<T> leafMarker, IList<IList<T>> paths) {
+ private static <T> void findPath(Tree<T> subtree, LinkedList<T> path,
+ Predicate<T> leafMarker, ListEx<ListEx<T>> paths) {
if (subtree.getChildrenCount() == 0 && leafMarker.test(subtree.getHead())) {
/* We're at a matching leaf node. Add it. */
- IList<T> finalPath = new FunctionalList<>();
+ ListEx<T> finalPath = new FunctionalList<>();
- for (T ePath : path) {
- finalPath.add(ePath);
- }
+ for (T ePath : path) finalPath.add(ePath);
finalPath.add(subtree.getHead());
@@ -56,4 +52,43 @@ public class TreeUtils {
path.removeLast();
}
}
+
+ /**
+ * Performs 'variable substitution' or something along those lines on a tree.
+ *
+ * @param <ContainedType> The type of element contained in the tree.
+ * @param tree The tree to do expansion in.
+ * @param marker The function to mark which nodes should be expanded.
+ * @param expander The function to expand nodes.
+ * @return A transformed copy of the tree.
+ */
+ public static <ContainedType> Tree<ContainedType> substitute(
+ Tree<ContainedType> tree,
+ Predicate<ContainedType> marker,
+ Function<ContainedType, Tree<ContainedType>> expander) {
+ tree.topDownTransform((contents) -> {
+ if (marker.test(contents)) return TopDownTransformResult.TRANSFORM;
+ else return TopDownTransformResult.PASSTHROUGH;
+ }, (node) -> {
+ return expander.apply(node.getHead());
+ });
+ return tree;
+ }
+
+ /**
+ * Performs 'variable substitution' or something along those lines on a tree.
+ *
+ * @param <ContainedType> The type of element contained in the tree.
+ * @param tree The tree to do expansion in.
+ * @param environment A map which contains the variables to substitute.
+ * @return A transformed copy of the tree.
+ */
+ public static <ContainedType> Tree<ContainedType> substitute(
+ Tree<ContainedType> tree,
+ MapEx<ContainedType, Tree<ContainedType>> environment) {
+ return substitute(
+ tree,
+ environment::containsKey,
+ (element) -> environment.get(element).get());
+ }
}
diff --git a/base/src/main/java/bjc/utils/gen/RandomGrammar.java b/base/src/main/java/bjc/utils/gen/RandomGrammar.java
index 050165b..620cc5b 100644
--- a/base/src/main/java/bjc/utils/gen/RandomGrammar.java
+++ b/base/src/main/java/bjc/utils/gen/RandomGrammar.java
@@ -1,7 +1,7 @@
package bjc.utils.gen;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A weighted grammar where all the rules have a equal chance of occuring.
@@ -27,8 +27,8 @@ public class RandomGrammar<E> extends WeightedGrammar<E> {
* The cases to add for this rule.
*/
@SafeVarargs
- public final void addCases(final E rule, final IList<E>... cases) {
- for (final IList<E> currentCase : cases) {
+ public final void addCases(final E rule, final ListEx<E>... cases) {
+ for (final ListEx<E> currentCase : cases) {
super.addCase(rule, 1, currentCase);
}
}
@@ -43,10 +43,10 @@ public class RandomGrammar<E> extends WeightedGrammar<E> {
* The cases to add for this rule.
*/
@SafeVarargs
- public final void makeRule(final E rule, final IList<E>... cases) {
+ public final void makeRule(final E rule, final ListEx<E>... cases) {
super.addRule(rule);
- for (final IList<E> currentCase : cases) {
+ for (final ListEx<E> currentCase : cases) {
super.addCase(rule, 1, currentCase);
}
}
@@ -60,7 +60,7 @@ public class RandomGrammar<E> extends WeightedGrammar<E> {
* @param cases
* The cases to add for this rule.
*/
- public void makeRule(final E rule, final IList<IList<E>> cases) {
+ public void makeRule(final E rule, final ListEx<ListEx<E>> cases) {
if (cases == null)
throw new NullPointerException("Cases must not be null");
diff --git a/base/src/main/java/bjc/utils/gen/WeightedGrammar.java b/base/src/main/java/bjc/utils/gen/WeightedGrammar.java
index 324a80c..48cc658 100644
--- a/base/src/main/java/bjc/utils/gen/WeightedGrammar.java
+++ b/base/src/main/java/bjc/utils/gen/WeightedGrammar.java
@@ -6,12 +6,12 @@ import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
-import bjc.data.IPair;
import bjc.data.Pair;
+import bjc.data.SimplePair;
import bjc.funcdata.FunctionalList;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
/**
* A random grammar, where certain rules will come up more often than others.
@@ -26,22 +26,22 @@ public class WeightedGrammar<E> {
protected String initialRule;
/** The rules currently in this grammar */
- protected IMap<E, WeightedRandom<IList<E>>> rules;
+ protected MapEx<E, WeightedRandom<ListEx<E>>> rules;
/** The random number generator used for random numbers */
private Random rng;
/** All of the subgrammars of this grammar */
- protected IMap<E, WeightedGrammar<E>> subgrammars;
+ protected MapEx<E, WeightedGrammar<E>> subgrammars;
/** Rules that require special handling */
- private IMap<E, Supplier<IList<E>>> specialRules;
+ private MapEx<E, Supplier<ListEx<E>>> specialRules;
/** Predicate for marking special tokens */
private Predicate<E> specialMarker;
/** Action for special tokens */
- private BiFunction<E, WeightedGrammar<E>, IList<E>> specialAction;
+ private BiFunction<E, WeightedGrammar<E>, ListEx<E>> specialAction;
/** Create a new weighted grammar. */
public WeightedGrammar() {
@@ -75,7 +75,7 @@ public class WeightedGrammar<E> {
* The action to take on those tokens.
*/
public void configureSpecial(final Predicate<E> marker,
- final BiFunction<E, WeightedGrammar<E>, IList<E>> action) {
+ final BiFunction<E, WeightedGrammar<E>, ListEx<E>> action) {
specialMarker = marker;
specialAction = action;
}
@@ -89,7 +89,7 @@ public class WeightedGrammar<E> {
* @param cse
* The case for the rule.
*/
- public void addSpecialRule(final E ruleName, final Supplier<IList<E>> cse) {
+ public void addSpecialRule(final E ruleName, final Supplier<ListEx<E>> cse) {
if (ruleName == null) {
throw new NullPointerException("Rule name must not be null");
} else if (cse == null) {
@@ -111,14 +111,14 @@ public class WeightedGrammar<E> {
* @param cse
* The case being added.
*/
- public void addCase(final E ruleName, final int probability, final IList<E> cse) {
+ public void addCase(final E ruleName, final int probability, final ListEx<E> cse) {
if (ruleName == null) {
throw new NullPointerException("Rule name must be not null");
} else if (cse == null) {
throw new NullPointerException("Case body must not be null");
}
- rules.get(ruleName).addProbability(probability, cse);
+ rules.get(ruleName).get().addProbability(probability, cse);
}
/**
@@ -143,7 +143,7 @@ public class WeightedGrammar<E> {
return false;
if (subgrammars.containsKey(name)) {
- subgrammars.put(alias, subgrammars.get(name));
+ subgrammars.put(alias, subgrammars.get(name).get());
return true;
}
@@ -180,7 +180,7 @@ public class WeightedGrammar<E> {
*
* @return Whether or not the rule was succesfully added.
*/
- public boolean addRule(final E name, final WeightedRandom<IList<E>> cases) {
+ public boolean addRule(final E name, final WeightedRandom<ListEx<E>> cases) {
if (name == null) {
throw new NullPointerException("Name must not be null");
} else if (cases == null) {
@@ -255,13 +255,13 @@ public class WeightedGrammar<E> {
*
* @return A set of sentences generated by the specified rule.
*/
- public IList<IList<E>> generateDebugValues(final E ruleName) {
+ public ListEx<ListEx<E>> generateDebugValues(final E ruleName) {
if (ruleName == null)
throw new NullPointerException("Rule name must not be null");
- final IList<IList<E>> returnedList = new FunctionalList<>();
+ final ListEx<ListEx<E>> returnedList = new FunctionalList<>();
- final WeightedRandom<IList<E>> ruleGenerator = rules.get(ruleName);
+ final WeightedRandom<ListEx<E>> ruleGenerator = rules.get(ruleName).get();
for (int i = 0; i < 10; i++) {
returnedList.add(ruleGenerator.generateValue());
@@ -288,7 +288,7 @@ public class WeightedGrammar<E> {
*
* @return A randomly generated sentence from the specified initial rule.
*/
- public <T> IList<T> generateGenericValues(final E initRules,
+ public <T> ListEx<T> generateGenericValues(final E initRules,
final Function<E, T> tokenTransformer, final T spacer) {
if (initRules == null) {
throw new NullPointerException("Initial rule must not be null");
@@ -298,9 +298,9 @@ public class WeightedGrammar<E> {
throw new NullPointerException("Spacer must not be null");
}
- final IList<T> returnedList = new FunctionalList<>();
+ final ListEx<T> returnedList = new FunctionalList<>();
- IList<E> genRules = new FunctionalList<>(initRules);
+ ListEx<E> genRules = new FunctionalList<>(initRules);
if (specialMarker != null) {
if (specialMarker.test(initRules)) {
@@ -313,7 +313,7 @@ public class WeightedGrammar<E> {
*/
for (final E initRule : genRules.toIterable()) {
if (specialRules.containsKey(initRule)) {
- for (final E rulePart : specialRules.get(initRule).get().toIterable()) {
+ for (final E rulePart : specialRules.get(initRule).get().get().toIterable()) {
final Iterable<T> generatedRuleParts
= generateGenericValues(rulePart, tokenTransformer, spacer)
.toIterable();
@@ -324,7 +324,7 @@ public class WeightedGrammar<E> {
}
}
} else if (subgrammars.containsKey(initRule)) {
- final Iterable<T> ruleParts = subgrammars.get(initRule)
+ final Iterable<T> ruleParts = subgrammars.get(initRule).get()
.generateGenericValues(initRule, tokenTransformer, spacer)
.toIterable();
@@ -334,7 +334,7 @@ public class WeightedGrammar<E> {
}
} else if (rules.containsKey(initRule)) {
final Iterable<E> ruleParts
- = rules.get(initRule).generateValue().toIterable();
+ = rules.get(initRule).get().generateValue().toIterable();
for (final E rulePart : ruleParts) {
final Iterable<T> generatedRuleParts
@@ -371,8 +371,8 @@ public class WeightedGrammar<E> {
*
* @return A list of random grammar elements generated by the specified rule.
*/
- public IList<E> generateListValues(final E initRule, final E spacer) {
- final IList<E> retList
+ public ListEx<E> generateListValues(final E initRule, final E spacer) {
+ final ListEx<E> retList
= generateGenericValues(initRule, strang -> strang, spacer);
return retList;
@@ -401,8 +401,8 @@ public class WeightedGrammar<E> {
*
* @return The set of all rule names in this grammar.
*/
- public IList<E> getRuleNames() {
- final IList<E> ruleNames = new FunctionalList<>();
+ public ListEx<E> getRuleNames() {
+ final ListEx<E> ruleNames = new FunctionalList<>();
ruleNames.addAll(rules.keyList());
ruleNames.addAll(specialRules.keyList());
@@ -422,7 +422,7 @@ public class WeightedGrammar<E> {
if (name == null)
throw new NullPointerException("Subgrammar name must not be null");
- return subgrammars.get(name);
+ return subgrammars.get(name).get();
}
/**
@@ -471,19 +471,19 @@ public class WeightedGrammar<E> {
throw new IllegalArgumentException(
"Number of times to prefix must be positive.");
- final WeightedRandom<IList<E>> rule = rules.get(ruleName);
+ final WeightedRandom<ListEx<E>> rule = rules.get(ruleName).get();
- final IList<IPair<Integer, IList<E>>> newResults = new FunctionalList<>();
+ final ListEx<Pair<Integer, ListEx<E>>> newResults = new FunctionalList<>();
/*
* @NOTE Can this be simplified?
*/
rule.getValues().forEach(pair -> {
- final IList<IList<E>> newRule = new FunctionalList<>();
+ final ListEx<ListEx<E>> newRule = new FunctionalList<>();
for (int i = 1; i <= numberOfTimes; i++) {
- final IList<E> newCase = pair.merge((left, right) -> {
- final IList<E> returnVal = new FunctionalList<>();
+ final ListEx<E> newCase = pair.merge((left, right) -> {
+ final ListEx<E> returnVal = new FunctionalList<>();
for (final E val : right.toIterable()) {
returnVal.add(val);
@@ -502,7 +502,7 @@ public class WeightedGrammar<E> {
newRule.forEach((list) -> {
final Integer currentProb = pair.merge((left, right) -> left);
- newResults.add(new Pair<>(currentProb + additionalProbability, list));
+ newResults.add(new SimplePair<>(currentProb + additionalProbability, list));
});
});
@@ -534,13 +534,13 @@ public class WeightedGrammar<E> {
throw new NullPointerException("Prefix token must not be null");
}
- final WeightedRandom<IList<E>> rule = rules.get(ruleName);
+ final WeightedRandom<ListEx<E>> rule = rules.get(ruleName).get();
- final IList<IPair<Integer, IList<E>>> newResults = new FunctionalList<>();
+ final ListEx<Pair<Integer, ListEx<E>>> newResults = new FunctionalList<>();
rule.getValues().forEach(pair -> {
- final IList<E> newCase = pair.merge((left, right) -> {
- final IList<E> returnVal = new FunctionalList<>();
+ final ListEx<E> newCase = pair.merge((left, right) -> {
+ final ListEx<E> returnVal = new FunctionalList<>();
for (final E val : right.toIterable()) {
returnVal.add(val);
@@ -551,7 +551,7 @@ public class WeightedGrammar<E> {
newCase.prepend(prefixToken);
- newResults.add(new Pair<>(
+ newResults.add(new SimplePair<>(
pair.merge((left, right) -> left) + additionalProbability, newCase));
});
@@ -589,13 +589,13 @@ public class WeightedGrammar<E> {
throw new NullPointerException("Prefix token must not be null");
}
- final WeightedRandom<IList<E>> rule = rules.get(ruleName);
+ final WeightedRandom<ListEx<E>> rule = rules.get(ruleName).get();
- final IList<IPair<Integer, IList<E>>> newResults = new FunctionalList<>();
+ final ListEx<Pair<Integer, ListEx<E>>> newResults = new FunctionalList<>();
rule.getValues().forEach(par -> {
- final IList<E> newCase = par.merge((left, right) -> {
- final IList<E> returnVal = new FunctionalList<>();
+ final ListEx<E> newCase = par.merge((left, right) -> {
+ final ListEx<E> returnVal = new FunctionalList<>();
for (final E val : right.toIterable()) {
returnVal.add(val);
@@ -606,7 +606,7 @@ public class WeightedGrammar<E> {
newCase.add(suffixToken);
- newResults.add(new Pair<>(
+ newResults.add(new SimplePair<>(
par.merge((left, right) -> left) + additionalProbability, newCase));
});
diff --git a/base/src/main/java/bjc/utils/gen/WeightedRandom.java b/base/src/main/java/bjc/utils/gen/WeightedRandom.java
index b17017b..01a961f 100644
--- a/base/src/main/java/bjc/utils/gen/WeightedRandom.java
+++ b/base/src/main/java/bjc/utils/gen/WeightedRandom.java
@@ -2,10 +2,10 @@ package bjc.utils.gen;
import java.util.Random;
-import bjc.data.IPair;
import bjc.data.Pair;
+import bjc.data.SimplePair;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Represents a random number generator where certain results are weighted more
@@ -17,7 +17,7 @@ import bjc.funcdata.IList;
* The type of values that are randomly selected.
*/
public class WeightedRandom<E> {
- private final IList<IPair<Integer, E>> values;
+ private final ListEx<Pair<Integer, E>> values;
/* The source for any needed random numbers */
private Random source;
@@ -38,8 +38,7 @@ public class WeightedRandom<E> {
public WeightedRandom(Random src) {
values = new FunctionalList<>();
- if (src == null)
- throw new NullPointerException("Source of randomness must not be null");
+ if (src == null) throw new NullPointerException("Source of randomness must not be null");
source = src;
}
@@ -51,7 +50,7 @@ public class WeightedRandom<E> {
this(BASE);
}
- private WeightedRandom(Random src, IList<IPair<Integer, E>> vals, int chance) {
+ private WeightedRandom(Random src, ListEx<Pair<Integer, E>> vals, int chance) {
source = src;
values = vals;
@@ -69,7 +68,7 @@ public class WeightedRandom<E> {
* The result to get when the chance comes up.
*/
public void addProbability(final int chance, final E result) {
- values.add(new Pair<>(chance, result));
+ values.add(new SimplePair<>(chance, result));
totalChance += chance;
}
@@ -92,7 +91,7 @@ public class WeightedRandom<E> {
*/
public E generateValue(Random rn) {
int target = rn.nextInt(totalChance);
- for (IPair<Integer, E> val : values) {
+ for (Pair<Integer, E> val : values) {
int prob = val.getLeft();
if (target < prob) {
@@ -115,8 +114,8 @@ public class WeightedRandom<E> {
*
* @return A list of all the values that can be generated
*/
- public IList<E> getResults() {
- return values.map(IPair::getRight);
+ public ListEx<E> getResults() {
+ return values.map(Pair::getRight);
}
/**
@@ -125,7 +124,7 @@ public class WeightedRandom<E> {
*
* @return A list of pairs of values and value probabilities
*/
- public IList<IPair<Integer, E>> getValues() {
+ public ListEx<Pair<Integer, E>> getValues() {
return values;
}
@@ -156,12 +155,10 @@ public class WeightedRandom<E> {
* @return A random value.
*/
public E getDescent(int factor, Random rn) {
- if (values.getSize() == 0)
- return null;
+ if (values.getSize() == 0) return null;
- for (IPair<Integer, E> val : values) {
- if (rn.nextInt(factor) == 0)
- continue;
+ for (Pair<Integer, E> val : values) {
+ if (rn.nextInt(factor) == 0) continue;
if (exhaust) {
totalChance -= val.getLeft();
@@ -172,9 +169,8 @@ public class WeightedRandom<E> {
return val.getRight();
}
- IPair<Integer, E> val = values.getByIndex(values.getSize() - 1);
- if (exhaust)
- values.removeMatching(val);
+ Pair<Integer, E> val = values.getByIndex(values.getSize() - 1);
+ if (exhaust) values.removeMatching(val);
return val.getRight();
}
@@ -214,8 +210,7 @@ public class WeightedRandom<E> {
* @return The value at the index corresponding to the number of successes.
*/
public E getBinomial(int target, int bound, int trials, Random rn) {
- if (values.getSize() == 0)
- return null;
+ if (values.getSize() == 0) return null;
int numSuc = 0;
@@ -236,7 +231,7 @@ public class WeightedRandom<E> {
// System.err.printf("\tTRACE: got %d success for binomial trials (%d <= 1d%d,
// %d times)\n", numSuc, target, bound, trials);
- IPair<Integer, E> val = values.getByIndex(Math.min(numSuc, values.getSize() - 1));
+ Pair<Integer, E> val = values.getByIndex(Math.min(numSuc, values.getSize() - 1));
if (exhaust) {
totalChance -= val.getLeft();
@@ -253,10 +248,8 @@ public class WeightedRandom<E> {
* @return A new WeightedRandom that is exhaustible.
*/
public WeightedRandom<E> exhaustible() {
- IList<IPair<Integer, E>> lst = new FunctionalList<>();
- for (IPair<Integer, E> val : values) {
- lst.add(val);
- }
+ ListEx<Pair<Integer, E>> lst = new FunctionalList<>();
+ for (Pair<Integer, E> val : values) lst.add(val);
WeightedRandom<E> res = new WeightedRandom<>(source, lst, totalChance);
diff --git a/base/src/main/java/bjc/utils/graph/AdjacencyMap.java b/base/src/main/java/bjc/utils/graph/AdjacencyMap.java
index e046fb5..978b21d 100644
--- a/base/src/main/java/bjc/utils/graph/AdjacencyMap.java
+++ b/base/src/main/java/bjc/utils/graph/AdjacencyMap.java
@@ -6,12 +6,12 @@ import java.io.PrintStream;
import java.util.InputMismatchException;
import java.util.Scanner;
-import bjc.data.IHolder;
+import bjc.data.Holder;
import bjc.data.Identity;
import bjc.funcdata.FunctionalList;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
import bjc.utils.funcutils.FuncUtils;
/**
@@ -32,8 +32,7 @@ public class AdjacencyMap<T> {
* @return An adjacency map defined by the text
*/
public static AdjacencyMap<Integer> fromStream(final InputStream stream) {
- if (stream == null)
- throw new NullPointerException("Input source must not be null");
+ if (stream == null) throw new NullPointerException("Input source must not be null");
/* Create the adjacency map. */
AdjacencyMap<Integer> adjacency;
@@ -60,17 +59,16 @@ public class AdjacencyMap<T> {
throw imex;
}
- if (vertexCount <= 0)
- throw new InputMismatchException(
+ if (vertexCount <= 0) throw new InputMismatchException(
"The number of vertices must be greater than 0");
- final IList<Integer> vertices = new FunctionalList<>();
+ final ListEx<Integer> vertices = new FunctionalList<>();
FuncUtils.doTimes(vertexCount, vertexNo -> vertices.add(vertexNo));
adjacency = new AdjacencyMap<>(vertices);
- final IHolder<Integer> row = new Identity<>(0);
+ final Holder<Integer> row = new Identity<>(0);
input.forEachRemaining(strang -> {
readRow(adjacency, vertexCount, row, strang);
@@ -82,7 +80,7 @@ public class AdjacencyMap<T> {
/* Read a row of edges. */
private static void readRow(final AdjacencyMap<Integer> adjacency,
- final int vertexCount, final IHolder<Integer> row, final String strang) {
+ final int vertexCount, final Holder<Integer> row, final String strang) {
final String[] parts = strang.split(" ");
if (parts.length != vertexCount) {
@@ -117,7 +115,7 @@ public class AdjacencyMap<T> {
}
/** The backing storage of the map */
- private final IMap<T, IMap<T, Integer>> adjacency = new FunctionalMap<>();
+ private final MapEx<T, MapEx<T, Integer>> adjacency = new FunctionalMap<>();
/**
* Create a new map from a set of vertices
@@ -125,12 +123,11 @@ public class AdjacencyMap<T> {
* @param vertices
* The set of vertices to create a map from
*/
- public AdjacencyMap(final IList<T> vertices) {
- if (vertices == null)
- throw new NullPointerException("Vertices must not be null");
+ public AdjacencyMap(final ListEx<T> vertices) {
+ if (vertices == null) throw new NullPointerException("Vertices must not be null");
vertices.forEach(vertex -> {
- final IMap<T, Integer> row = new FunctionalMap<>();
+ final MapEx<T, Integer> row = new FunctionalMap<>();
vertices.forEach(target -> {
row.put(target, 0);
@@ -146,15 +143,13 @@ public class AdjacencyMap<T> {
* @return Whether or not the graph is directed
*/
public boolean isDirected() {
- final IHolder<Boolean> result = new Identity<>(true);
+ final Holder<Boolean> result = new Identity<>(true);
adjacency.forEach((sourceKey, sourceValue) -> {
sourceValue.forEach((targetKey, targetValue) -> {
- final int inverseValue = adjacency.get(targetKey).get(sourceKey);
+ final int inverseValue = adjacency.get(targetKey).get().get(sourceKey).get();
- if (targetValue != inverseValue) {
- result.replace(false);
- }
+ if (targetValue != inverseValue) result.replace(false);
});
});
@@ -172,11 +167,8 @@ public class AdjacencyMap<T> {
* The weight of the edge.
*/
public void setWeight(final T source, final T target, final int weight) {
- if (source == null) {
- throw new NullPointerException("Source vertex must not be null");
- } else if (target == null) {
- throw new NullPointerException("Target vertex must not be null");
- }
+ if (source == null) throw new NullPointerException("Source vertex must not be null");
+ else if (target == null) throw new NullPointerException("Target vertex must not be null");
if (!adjacency.containsKey(source)) {
String msg = String.format("Source vertex %s isn't present in map", source);
@@ -188,7 +180,7 @@ public class AdjacencyMap<T> {
throw new IllegalArgumentException(msg);
}
- adjacency.get(source).put(target, weight);
+ adjacency.get(source).get().put(target, weight);
}
/**
@@ -215,8 +207,7 @@ public class AdjacencyMap<T> {
* The stream to convert to.
*/
public void toStream(final OutputStream sink) {
- if (sink == null)
- throw new NullPointerException("Output source must not be null");
+ if (sink == null) throw new NullPointerException("Output source must not be null");
final PrintStream outputPrinter = new PrintStream(sink);
diff --git a/base/src/main/java/bjc/utils/graph/Edge.java b/base/src/main/java/bjc/utils/graph/Edge.java
index fe3d891..b48fcd0 100644
--- a/base/src/main/java/bjc/utils/graph/Edge.java
+++ b/base/src/main/java/bjc/utils/graph/Edge.java
@@ -28,11 +28,8 @@ public class Edge<T> {
* The distance between initial and terminal edge.
*/
public Edge(final T initial, final T terminal, final int distance) {
- if (initial == null) {
- throw new NullPointerException("Initial node must not be null");
- } else if (terminal == null) {
- throw new NullPointerException("Terminal node must not be null");
- }
+ if (initial == null) throw new NullPointerException("Initial node must not be null");
+ else if (terminal == null) throw new NullPointerException("Terminal node must not be null");
this.source = initial;
this.target = terminal;
@@ -41,27 +38,23 @@ public class Edge<T> {
@Override
public boolean equals(final Object obj) {
- if (this == obj)
- return true;
- else if (obj == null)
- return false;
- else if (getClass() != obj.getClass())
- return false;
+ if (this == obj) return true;
+ else if (obj == null) return false;
+ else if (getClass() != obj.getClass()) return false;
else {
final Edge<?> other = (Edge<?>) obj;
- if (distance != other.distance)
+ if (distance != other.distance) {
return false;
- else if (source == null) {
- if (other.source != null)
- return false;
- } else if (!source.equals(other.source))
+ } else if (source == null) {
+ if (other.source != null) return false;
+ } else if (!source.equals(other.source)) {
return false;
- else if (target == null) {
- if (other.target != null)
- return false;
- } else if (!target.equals(other.target))
+ } else if (target == null) {
+ if (other.target != null) return false;
+ } else if (!target.equals(other.target)) {
return false;
+ }
return true;
}
diff --git a/base/src/main/java/bjc/utils/graph/Graph.java b/base/src/main/java/bjc/utils/graph/Graph.java
index 8ff5647..f32e07f 100644
--- a/base/src/main/java/bjc/utils/graph/Graph.java
+++ b/base/src/main/java/bjc/utils/graph/Graph.java
@@ -10,11 +10,11 @@ import java.util.Set;
import java.util.function.BiConsumer;
import java.util.function.BiPredicate;
-import bjc.data.IHolder;
+import bjc.data.Holder;
import bjc.data.Identity;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
/**
* A directed weighted graph, where the vertices have some arbitrary label.
@@ -47,7 +47,7 @@ public class Graph<T> {
}
/** The backing representation of the graph. */
- private final IMap<T, IMap<T, Integer>> backing;
+ private final MapEx<T, MapEx<T, Integer>> backing;
/** Create a new empty graph. */
public Graph() {
@@ -72,19 +72,14 @@ public class Graph<T> {
public void addEdge(final T source, final T target, final int distance,
final boolean directed) {
/* Can't add edges with a null source or target. */
- if (source == null) {
- throw new NullPointerException("The source vertex cannot be null");
- } else if (target == null) {
- throw new NullPointerException("The target vertex cannot be null");
- }
+ if (source == null) throw new NullPointerException("The source vertex cannot be null");
+ else if (target == null) throw new NullPointerException("The target vertex cannot be null");
/* Initialize adjacency list for vertices if necessary. */
- if (!backing.containsKey(source)) {
- backing.put(source, new FunctionalMap<T, Integer>());
- }
+ if (!backing.containsKey(source)) backing.put(source, new FunctionalMap<T, Integer>());
/* Add the edge to the graph. */
- backing.get(source).put(target, distance);
+ backing.get(source).get().put(target, distance);
/* Handle possible directed edges. */
if (!directed) {
@@ -92,7 +87,7 @@ public class Graph<T> {
backing.put(target, new FunctionalMap<T, Integer>());
}
- backing.get(target).put(source, distance);
+ backing.get(target).get().put(source, distance);
}
}
@@ -110,16 +105,11 @@ public class Graph<T> {
*/
public void forAllEdgesMatchingAt(final T source,
final BiPredicate<T, Integer> matcher, final BiConsumer<T, Integer> action) {
- if (matcher == null) {
- throw new NullPointerException("Matcher must not be null");
- } else if (action == null) {
- throw new NullPointerException("Action must not be null");
- }
+ if (matcher == null) throw new NullPointerException("Matcher must not be null");
+ else if (action == null) throw new NullPointerException("Action must not be null");
getEdges(source).forEach((target, weight) -> {
- if (matcher.test(target, weight)) {
- action.accept(target, weight);
- }
+ if (matcher.test(target, weight)) action.accept(target, weight);
});
}
@@ -130,14 +120,15 @@ public class Graph<T> {
* The vertex to use as a source.
* @return All of the edges with the specified vertex as a source.
*/
- public IMap<T, Integer> getEdges(final T source) {
+ public MapEx<T, Integer> getEdges(final T source) {
/* Can't find edges for a null source. */
- if (source == null)
+ if (source == null) {
throw new NullPointerException("The source cannot be null.");
- else if (!backing.containsKey(source))
+ } else if (!backing.containsKey(source)) {
throw new IllegalArgumentException("Vertex " + source + " is not in graph");
+ }
- return backing.get(source);
+ return backing.get(source).get();
}
/**
@@ -168,7 +159,7 @@ public class Graph<T> {
final Set<T> visited = new HashSet<>();
/* Start at the initial vertex and visit it */
- final IHolder<T> source = new Identity<>(getInitial());
+ final Holder<T> source = new Identity<>(getInitial());
visited.add(source.getValue());
@@ -176,14 +167,17 @@ public class Graph<T> {
while (visited.size() != getVertexCount()) {
/* Grab all edges adjacent to the provided edge. */
- forAllEdgesMatchingAt(source.getValue(), (target, weight) -> !visited.contains(target), (target, weight) -> {
- final T vert = source.unwrap(vertex -> vertex);
+ forAllEdgesMatchingAt(source.getValue(),
+ (target, weight) -> !visited.contains(target),
+ (target, weight) -> {
+ final T vert = source.unwrap(vertex -> vertex);
- available.add(new Edge<>(vert, target, weight));
- });
+ available.add(new Edge<>(vert, target, weight));
+ }
+ );
/* Get the edge with the minimum distance. */
- final IHolder<Edge<T>> minimum = new Identity<>(available.poll());
+ final Holder<Edge<T>> minimum = new Identity<>(available.poll());
/*
* Only consider edges where we haven't visited the target of the edge.
@@ -219,7 +213,7 @@ public class Graph<T> {
*
* @return A unmodifiable set of all the vertices in the graph.
*/
- public IList<T> getVertices() {
+ public ListEx<T> getVertices() {
return backing.keyList();
}
@@ -253,7 +247,7 @@ public class Graph<T> {
throw new NoSuchElementException(msg);
}
- backing.get(source).remove(target);
+ backing.get(source).get().remove(target);
/*
* Uncomment this to turn the graph undirected
diff --git a/base/src/main/java/bjc/utils/gui/ExtensionFileFilter.java b/base/src/main/java/bjc/utils/gui/ExtensionFileFilter.java
index d9ebda5..c07543f 100644
--- a/base/src/main/java/bjc/utils/gui/ExtensionFileFilter.java
+++ b/base/src/main/java/bjc/utils/gui/ExtensionFileFilter.java
@@ -6,7 +6,7 @@ import java.util.List;
import javax.swing.filechooser.FileFilter;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A file filter based on extensions.
@@ -20,7 +20,7 @@ public class ExtensionFileFilter extends FileFilter {
/**
* The list holding all filtered extensions
*/
- private final IList<String> extensions;
+ private final ListEx<String> extensions;
/**
* Create a new filter only showing files with the specified extensions.
diff --git a/base/src/main/java/bjc/utils/gui/SimpleKeyedButton.java b/base/src/main/java/bjc/utils/gui/SimpleKeyedButton.java
index 0c40fbe..976b259 100644
--- a/base/src/main/java/bjc/utils/gui/SimpleKeyedButton.java
+++ b/base/src/main/java/bjc/utils/gui/SimpleKeyedButton.java
@@ -59,8 +59,9 @@ public class SimpleKeyedButton extends JButton {
* @param aevListener
* The listener that handles the implementation of the
* action.
+ * @return The action that was bound to the button.
*/
- public void setGlobalDefaultKeystroke(String eventName, String keystroke,
+ public Action setGlobalDefaultKeystroke(String eventName, String keystroke,
Consumer<ActionEvent> aevListener) {
Action act = new KeyedButtonAction(eventName, aevListener);
KeyStroke stroke = KeyStroke.getKeyStroke(keystroke);
@@ -69,6 +70,8 @@ public class SimpleKeyedButton extends JButton {
this.getInputMap(WHEN_IN_FOCUSED_WINDOW).put(stroke, eventName);
this.getActionMap().put(eventName, act);
this.setText(label);
+
+ return act;
}
/**
@@ -82,8 +85,9 @@ public class SimpleKeyedButton extends JButton {
* @param aevListener
* The listener that handles the implementation of the
* action.
+ * @return The action that was bound to the button.
*/
- public void addGlobalKeystroke(String eventName, String keystroke,
+ public Action addGlobalKeystroke(String eventName, String keystroke,
Consumer<ActionEvent> aevListener) {
Action act = new KeyedButtonAction(eventName, aevListener);
KeyStroke stroke = KeyStroke.getKeyStroke(keystroke);
@@ -91,5 +95,7 @@ public class SimpleKeyedButton extends JButton {
this.getInputMap(WHEN_IN_FOCUSED_WINDOW).put(stroke, eventName);
this.getActionMap().put(eventName, act);
this.setText(label);
+
+ return act;
}
}
diff --git a/base/src/main/java/bjc/utils/gui/awt/ExtensionFileFilter.java b/base/src/main/java/bjc/utils/gui/awt/ExtensionFileFilter.java
index 04b0b68..9b55306 100644
--- a/base/src/main/java/bjc/utils/gui/awt/ExtensionFileFilter.java
+++ b/base/src/main/java/bjc/utils/gui/awt/ExtensionFileFilter.java
@@ -5,7 +5,7 @@ import java.io.FilenameFilter;
import java.util.List;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Filter a set of filenames by extension.
@@ -16,7 +16,7 @@ import bjc.funcdata.IList;
*/
public class ExtensionFileFilter implements FilenameFilter {
/* The list of extensions to filter */
- private final IList<String> extensions;
+ private final ListEx<String> extensions;
/**
* Create a new filter only showing files with the specified extensions.
diff --git a/base/src/main/java/bjc/utils/gui/panels/DropdownListPanel.java b/base/src/main/java/bjc/utils/gui/panels/DropdownListPanel.java
index 8f65577..465be02 100644
--- a/base/src/main/java/bjc/utils/gui/panels/DropdownListPanel.java
+++ b/base/src/main/java/bjc/utils/gui/panels/DropdownListPanel.java
@@ -9,7 +9,7 @@ import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.ListSelectionModel;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.gui.layout.AutosizeLayout;
import bjc.utils.gui.layout.HLayout;
@@ -35,7 +35,7 @@ public class DropdownListPanel extends JPanel {
* The items to choose from
*/
public <T> DropdownListPanel(final String type, final DefaultListModel<T> model,
- final IList<T> choices) {
+ final ListEx<T> choices) {
setLayout(new AutosizeLayout());
final JPanel itemInputPanel = new JPanel();
diff --git a/base/src/main/java/bjc/utils/gui/panels/HolderOutputPanel.java b/base/src/main/java/bjc/utils/gui/panels/HolderOutputPanel.java
index d0b5383..dbf70f0 100644
--- a/base/src/main/java/bjc/utils/gui/panels/HolderOutputPanel.java
+++ b/base/src/main/java/bjc/utils/gui/panels/HolderOutputPanel.java
@@ -4,11 +4,11 @@ import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.Timer;
-import bjc.data.IHolder;
+import bjc.data.Holder;
import bjc.utils.gui.layout.HLayout;
/**
- * A panel that outputs a value bound to a {@link IHolder}
+ * A panel that outputs a value bound to a {@link Holder}
*
* @author ben
*
@@ -19,7 +19,7 @@ public class HolderOutputPanel extends JPanel {
private Timer updater;
private final JLabel value;
private final int nDelay;
- private final IHolder<String> val;
+ private final Holder<String> val;
/**
* Create a new display panel, backed by a holder
@@ -31,7 +31,7 @@ public class HolderOutputPanel extends JPanel {
* @param nDelay
* The delay in ms between value updates
*/
- public HolderOutputPanel(final String lab, final IHolder<String> valueHolder,
+ public HolderOutputPanel(final String lab, final Holder<String> valueHolder,
final int nDelay) {
this.val = valueHolder;
this.nDelay = nDelay;
diff --git a/base/src/main/java/bjc/utils/gui/panels/ListParameterPanel.java b/base/src/main/java/bjc/utils/gui/panels/ListParameterPanel.java
index 3b48309..4ba5a01 100644
--- a/base/src/main/java/bjc/utils/gui/panels/ListParameterPanel.java
+++ b/base/src/main/java/bjc/utils/gui/panels/ListParameterPanel.java
@@ -9,7 +9,7 @@ import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.ListSelectionModel;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.gui.SimpleJList;
import bjc.utils.gui.layout.HLayout;
import bjc.utils.gui.layout.VLayout;
@@ -54,7 +54,7 @@ public class ListParameterPanel<E> extends JPanel {
* The default values to put in the list
*/
public ListParameterPanel(final Supplier<E> add, final Consumer<E> edit,
- final Consumer<E> remove, final IList<E> defaults) {
+ final Consumer<E> remove, final ListEx<E> defaults) {
setLayout(new VLayout(2));
JList<E> list;
diff --git a/base/src/main/java/bjc/utils/gui/panels/SimpleSpinnerPanel.java b/base/src/main/java/bjc/utils/gui/panels/SimpleSpinnerPanel.java
index 8ca6f2b..406929a 100644
--- a/base/src/main/java/bjc/utils/gui/panels/SimpleSpinnerPanel.java
+++ b/base/src/main/java/bjc/utils/gui/panels/SimpleSpinnerPanel.java
@@ -8,7 +8,7 @@ import javax.swing.JSpinner;
import javax.swing.SpinnerModel;
/**
- * A simple spinner control
+ * A simple spinner control alongside a label
*
* @author ben
*
diff --git a/base/src/main/java/bjc/utils/gui/panels/SliderInputPanel.java b/base/src/main/java/bjc/utils/gui/panels/SliderInputPanel.java
index 835513a..dacba42 100644
--- a/base/src/main/java/bjc/utils/gui/panels/SliderInputPanel.java
+++ b/base/src/main/java/bjc/utils/gui/panels/SliderInputPanel.java
@@ -9,6 +9,7 @@ import javax.swing.JPanel;
import javax.swing.JSlider;
import bjc.utils.gui.layout.HLayout;
+import bjc.utils.gui.panels.SliderInputPanel.*;
/**
* A simple input panel for a slider-controlled value and a manual-input field
@@ -18,51 +19,6 @@ import bjc.utils.gui.layout.HLayout;
*
*/
public class SliderInputPanel extends JPanel {
- private final class NumberFormatter extends JFormattedTextField.AbstractFormatter {
- private static final long serialVersionUID = -4448291795913908270L;
-
- private final int minValue;
- private final int maxValue;
-
- private final int initValue;
-
- public NumberFormatter(final SliderSettings settings) {
- minValue = settings.minValue;
- maxValue = settings.maxValue;
-
- initValue = settings.initValue;
- }
-
- @Override
- public Object stringToValue(final String text) throws ParseException {
- try {
- final int val = Integer.parseInt(text);
-
- if (val < minValue)
- throw new ParseException("Value must be greater than " + minValue, 0);
- else if (val > maxValue)
- throw new ParseException("Value must be smaller than " + maxValue, 0);
- else
- return val;
- } catch (final NumberFormatException nfex) {
- final ParseException pex
- = new ParseException("Value must be a valid integer", 0);
-
- pex.initCause(nfex);
-
- throw pex;
- }
- }
-
- @Override
- public String valueToString(final Object value) throws ParseException {
- if (value == null)
- return Integer.toString(initValue);
-
- return Integer.toString((Integer) value);
- }
- }
-
/**
* Represents the settings for a slider
*
@@ -104,7 +60,7 @@ public class SliderInputPanel extends JPanel {
* @param max
* The maximum slider value
* @param init
- * Th initial slider value
+ * The initial slider value
*/
public SliderSettings(final int min, final int max, final int init) {
minValue = min;
@@ -153,7 +109,6 @@ public class SliderInputPanel extends JPanel {
final int val = slider.getValue();
field.setValue(val);
-
action.accept(val);
}
});
@@ -187,3 +142,46 @@ public class SliderInputPanel extends JPanel {
field.setValue(value);
}
}
+
+final class NumberFormatter extends JFormattedTextField.AbstractFormatter {
+ private static final long serialVersionUID = -4448291795913908270L;
+
+ private final int minValue;
+ private final int maxValue;
+
+ private final int initValue;
+
+ public NumberFormatter(final SliderSettings settings) {
+ minValue = settings.minValue;
+ maxValue = settings.maxValue;
+
+ initValue = settings.initValue;
+ }
+
+ @Override
+ public Object stringToValue(final String text) throws ParseException {
+ try {
+ final int val = Integer.parseInt(text);
+
+ if (val < minValue) throw new ParseException(
+ "Value must be greater than " + minValue, 0);
+ else if (val > maxValue) throw new ParseException(
+ "Value must be smaller than " + maxValue, 0);
+ else return val;
+ } catch (final NumberFormatException nfex) {
+ final ParseException pex
+ = new ParseException("Value must be a valid integer", 0);
+
+ pex.initCause(nfex);
+
+ throw pex;
+ }
+ }
+
+ @Override
+ public String valueToString(final Object value) throws ParseException {
+ if (value == null) return Integer.toString(initValue);
+
+ return Integer.toString((Integer) value);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/ioutils/LevelSplitter.java b/base/src/main/java/bjc/utils/ioutils/LevelSplitter.java
index c2467ae..93bc424 100644
--- a/base/src/main/java/bjc/utils/ioutils/LevelSplitter.java
+++ b/base/src/main/java/bjc/utils/ioutils/LevelSplitter.java
@@ -5,29 +5,24 @@ import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
-/**
- * Splits a string on a delimiter, respecting grouping delimiters.
+/** Splits a string on a delimiter, respecting grouping delimiters.
*
* By default, grouping delimiters are (), [], {}, and <>, as well as single and
* double quoted strings.
*
- * @author bjculkin
- *
- */
+ * @author bjculkin */
public class LevelSplitter {
- /**
- * Defaultly configured level splitter.
- */
+ /** Default configured level splitter. */
public final static LevelSplitter def = new LevelSplitter();
- /**
- * Check if a string contains any one of a specified number of things,
+ /** Should empty strings be ignored? */
+ public boolean ignoreEmpty = false;
+
+ /** Check if a string contains any one of a specified number of things,
* respecting groups.
*
- * @param haystack
- * The string to look in.
- * @param needles
- * The strings to look for.
+ * @param haystack The string to look in.
+ * @param needles The strings to look for.
* @return Whether or not any of the strings were contained outside of groups.
*/
public boolean levelContains(String haystack, String... needles) {
@@ -88,13 +83,11 @@ public class LevelSplitter {
return false;
}
- /**
- * Split a string, respecting groups.
+ /** Split a string, respecting groups.
*
- * @param phrase
- * The string to split.
- * @param splits
- * The strings to split on.
+ * @param phrase The string to split.
+ * @param splits The strings to split on.
+ *
* @return A list of split strings. If keepDelims is true, it also includes the
* delimiters in between the split strings.
*/
@@ -102,15 +95,12 @@ public class LevelSplitter {
return levelSplit(phrase, false, splits);
}
- /**
- * Split a string, respecting groups.
+ /** Split a string, respecting groups.
*
- * @param phrase
- * The string to split.
- * @param keepDelims
- * Whether or not to include the delimiters in the results.
- * @param splits
- * The strings to split on.
+ * @param phrase The string to split.
+ * @param keepDelims Whether or not to include the delimiters in the results.
+ * @param splits The strings to split on.
+ *
* @return A list of split strings. If keepDelims is true, it also includes the
* delimiters in between the split strings.
*/
@@ -140,8 +130,7 @@ public class LevelSplitter {
if (work.regionMatches(i, split, 0, split.length())) {
strangs.add(work.substring(0, i));
- if (keepDelims)
- strangs.add(split);
+ if (keepDelims) strangs.add(split);
work = work.substring(i + split.length());
i = 0;
diff --git a/base/src/main/java/bjc/utils/ioutils/LineReader.java b/base/src/main/java/bjc/utils/ioutils/LineReader.java
deleted file mode 100644
index 2ac2797..0000000
--- a/base/src/main/java/bjc/utils/ioutils/LineReader.java
+++ /dev/null
@@ -1,18 +0,0 @@
-package bjc.utils.ioutils;
-
-/**
- * A line reader
- *
- * @author bjculkin
- *
- */
-public class LineReader implements AutoCloseable {
- //private Scanner scn;
-
- @Override
- public void close() {
- //scn.close();
- }
-
- // @TODO Implement me - ben, 1/6/20
-}
diff --git a/base/src/main/java/bjc/utils/ioutils/LogStream.java b/base/src/main/java/bjc/utils/ioutils/LogStream.java
new file mode 100644
index 0000000..f5271dc
--- /dev/null
+++ b/base/src/main/java/bjc/utils/ioutils/LogStream.java
@@ -0,0 +1,341 @@
+package bjc.utils.ioutils;
+
+import java.io.*;
+
+/**
+ * Simple class used for logging with various levels.
+ *
+ * @author Ben Culkin
+ */
+public class LogStream {
+ /**
+ * Log level for printing nothing.
+ */
+ public static final int NOTHING = -1;
+
+ /**
+ * Log level for printing only fatal errors.
+ */
+ public static final int FATAL = 0;
+
+ /**
+ * Log level for printing all errors.
+ */
+ public static final int ERROR = 1;
+
+ /**
+ * Log level for printing warnings.
+ */
+ public static final int WARN = 2;
+
+ /**
+ * Log level for printing info messages.
+ */
+ public static final int INFO = 3;
+
+ /**
+ * Log level for printing debug messages.
+ */
+ public static final int DEBUG = 4;
+
+ /**
+ * Log level for printing trace messages.
+ */
+ public static final int TRACE = 5;
+
+ private int verbosity;
+
+ private PrintStream output;
+
+ /**
+ * Create a new log stream.
+ *
+ * Defaults to printing only fatal errors.
+ *
+ * @param out
+ * The output stream to place things into.
+ */
+ public LogStream(PrintStream out) {
+ output = out;
+ verbosity = FATAL;
+ }
+
+ /**
+ * Create a new log stream.
+ *
+ * @param out
+ * The output stream to place things into.
+ * @param level
+ * The verbosity level. Use the constants in this class for the
+ * values.
+ */
+ public LogStream(PrintStream out, int level) {
+ output = out;
+ verbosity = level;
+ }
+
+ /**
+ * Create a new log stream.
+ *
+ * Defaults to printing only fatal errors.
+ *
+ * @param out
+ * The output stream to place things into.
+ */
+ public LogStream(OutputStream out) {
+ output = new PrintStream(out);
+ verbosity = FATAL;
+ }
+
+ /**
+ * Create a new log stream.
+ *
+ * @param out
+ * The output stream to place things into.
+ * @param level
+ * The verbosity level. Use the constants in this class for the
+ * values.
+ */
+ public LogStream(OutputStream out, int level) {
+ output = new PrintStream(out);
+ verbosity = level;
+ }
+
+ /**
+ * Get the verbosity of the stream.
+ *
+ * @return The verbosity of the stream.
+ */
+ public int verbosity() {
+ return verbosity;
+ }
+
+ /**
+ * Set the verbosity of the stream.
+ *
+ * @param verb
+ * The verbosity of the stream.
+ */
+ public void verbosity(int verb) {
+ verbosity = verb;
+ }
+
+ /**
+ * Increment the verbosity of the stream.
+ */
+ public void louder() {
+ louder(1);
+ }
+
+ /**
+ * Increase the verbosity of the stream by an amount.
+ *
+ * @param amt
+ * The amount to increase the verbosity by.
+ */
+ public void louder(int amt) {
+ verbosity += amt;
+ }
+
+ /**
+ * Decrement the verbosity of the stream.
+ */
+ public void quieter() {
+ quieter(1);
+ }
+
+ /**
+ * Decrease the verbosity of the stream by an amount.
+ *
+ * @param amt
+ * The amount to decrease the verbosity by.
+ */
+ public void quieter(int amt) {
+ verbosity -= amt;
+ }
+
+ /**
+ * Print a message that will always be visible.
+ *
+ * @param msg
+ * The message to print.
+ */
+ public void print(String msg) {
+ output.print(msg);
+ }
+
+ /**
+ * Print a formatted message that will always be visible.
+ *
+ * @param msg
+ * The format string for the message to print.
+ *
+ * @param args
+ * The arguments to the format string.
+ */
+ public void printf(String msg, Object... args) {
+ output.printf(msg, args);
+ }
+
+ /**
+ * Print a message at a given verbosity level.
+ *
+ * @param lvl
+ * The verbosity level.
+ * @param msg
+ * The message to print.
+ */
+ public void message(int lvl, String msg) {
+ if (verbosity >= lvl) {
+ output.print(msg);
+ }
+ }
+
+ /**
+ * Print a formatted message at a given verbosity level.
+ *
+ * @param lvl
+ * The verbosity level.
+ * @param msg
+ * The message to print.
+ * @param args
+ * The arguments to the message.
+ */
+ public void messagef(int lvl, String msg, Object... args) {
+ if (verbosity >= lvl) {
+ output.printf(msg, args);
+ }
+ }
+
+ /**
+ * Emit a fatal error message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void fatal(String msg) {
+ message(FATAL, msg);
+ }
+
+ /**
+ * Emit a formatted fatal error message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void fatalf(String msg, Object... args) {
+ messagef(FATAL, msg, args);
+ }
+
+ /**
+ * Emit a normal error message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void error(String msg) {
+ message(ERROR, msg);
+ }
+
+ /**
+ * Emit a formatted normal error message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void errorf(String msg, Object... args) {
+ messagef(ERROR, msg, args);
+ }
+
+ /**
+ * Emit a warning message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void warn(String msg) {
+ message(WARN, msg);
+ }
+
+ /**
+ * Emit a formatted warning message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void warnf(String msg, Object... args) {
+ messagef(WARN, msg, args);
+ }
+
+ /**
+ * Emit an info message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void info(String msg) {
+ message(INFO, msg);
+ }
+
+ /**
+ * Emit a formatted info message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void infof(String msg, Object... args) {
+ messagef(INFO, msg, args);
+ }
+
+ /**
+ * Emit a debug message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void debug(String msg) {
+ message(DEBUG, msg);
+ }
+
+ /**
+ * Emit a formatted debug message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void debugf(String msg, Object... args) {
+ messagef(DEBUG, msg, args);
+ }
+
+ /**
+ * Emit a tracing message.
+ *
+ * @param msg
+ * The message to emit.
+ */
+ public void trace(String msg) {
+ message(TRACE, msg);
+ }
+
+ /**
+ * Emit a formatted tracing message.
+ *
+ * @param msg
+ * The message to emit.
+ * @param args
+ * The arguments to the message.
+ */
+ public void tracef(String msg, Object... args) {
+ messagef(TRACE, msg, args);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/ioutils/RegexStringEditor.java b/base/src/main/java/bjc/utils/ioutils/RegexStringEditor.java
index 3dad724..e7d8c54 100644
--- a/base/src/main/java/bjc/utils/ioutils/RegexStringEditor.java
+++ b/base/src/main/java/bjc/utils/ioutils/RegexStringEditor.java
@@ -8,7 +8,7 @@ import java.util.regex.Pattern;
import bjc.data.Toggle;
import bjc.data.ValueToggle;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.functypes.ID;
/**
@@ -84,7 +84,7 @@ public class RegexStringEditor {
/*
* Get all of the occurances.
*/
- final IList<String> occurances = listOccurances(input, rPatt);
+ final ListEx<String> occurances = listOccurances(input, rPatt);
/*
* Execute the correct action on every occurance.
@@ -118,13 +118,13 @@ public class RegexStringEditor {
*
* @return The string, with both actions applied.
*/
- public static IList<String> mapOccurances(final String input, final Pattern rPatt,
+ public static ListEx<String> mapOccurances(final String input, final Pattern rPatt,
final UnaryOperator<String> betweenAction,
final UnaryOperator<String> onAction) {
/*
* Get all of the occurances.
*/
- final IList<String> occurances = listOccurances(input, rPatt);
+ final ListEx<String> occurances = listOccurances(input, rPatt);
/*
* Execute the correct action on every occurance.
@@ -146,8 +146,8 @@ public class RegexStringEditor {
* @return The string, as a list of match/non-match segments, starting/ending
* with a non-match segment.
*/
- public static IList<String> listOccurances(final String input, final Pattern rPatt) {
- final IList<String> res = new FunctionalList<>();
+ public static ListEx<String> listOccurances(final String input, final Pattern rPatt) {
+ final ListEx<String> res = new FunctionalList<>();
/*
* Create the matcher and work buffer.
diff --git a/base/src/main/java/bjc/utils/ioutils/ReportWriter.java b/base/src/main/java/bjc/utils/ioutils/ReportWriter.java
index 3815f25..a04102f 100644
--- a/base/src/main/java/bjc/utils/ioutils/ReportWriter.java
+++ b/base/src/main/java/bjc/utils/ioutils/ReportWriter.java
@@ -26,8 +26,7 @@ public class ReportWriter extends Writer {
// Indent string w/ tabs replaced with spaces
public String indentStrSpacedTabs;
- public IndentVal() {
- }
+ public IndentVal() {}
}
// Writer to print to
@@ -265,9 +264,7 @@ public class ReportWriter extends Writer {
// that case; whether we should continue calling the function,
// or just adjust the counts and pretend that nothing
// significant happened
- while (pageLine > linesPerPage) {
- writePage();
- }
+ while (pageLine > linesPerPage) writePage();
}
/**
@@ -328,6 +325,11 @@ public class ReportWriter extends Writer {
refreshIndents(lvl);
}
+ // @FIXME Ben Culkin 11/13/2020 IntArgument
+ // Overloading an argument like this is bogus. This should be restructured,
+ // probably as three different functions; one to refresh a general level,
+ // one to refresh every level, and one to refresh the default indentation
+
// Parameter is the level of indents to refresh.
//
// Pass a index to refresh that level
@@ -345,9 +347,7 @@ public class ReportWriter extends Writer {
if (lvl == -2) {
refreshIndent(defIVal);
} else if (lvl == -1) {
- for (IndentVal ival : iVals) {
- refreshIndent(ival);
- }
+ for (IndentVal ival : iVals) refreshIndent(ival);
refreshIndent(defIVal);
} else {
@@ -364,10 +364,9 @@ public class ReportWriter extends Writer {
StringBuilder conv = new StringBuilder();
for (int i = 0; i < indentLength; i++) {
char c = vl.indentStr.charAt(i);
+
if (c == '\t') {
- for (int j = 0; j < tabEqv; j++) {
- conv.append(' ');
- }
+ for (int j = 0; j < tabEqv; j++) conv.append(' ');
vl.indentStrPos += tabEqv;
} else {
@@ -534,17 +533,14 @@ public class ReportWriter extends Writer {
// If we're printing CRLF pairs, make sure that we don't
// print incomplete pairs.
if (i < lineSpacing - 1) {
- if (c == '\n' && lastChar == '\r')
- contained.write('\r');
+ if (c == '\n' && lastChar == '\r') contained.write('\r');
}
}
// @NOTE 9/17/18
//
// Not sure if this should be here, or before the above loop
- if (pageLine > linesPerPage || c == '\f') {
- writePage();
- }
+ if (pageLine > linesPerPage || c == '\f') writePage();
linePos = 0;
indentPos = 0;
@@ -553,8 +549,7 @@ public class ReportWriter extends Writer {
@Override
public void write(char[] cbuf, int off, int len) throws IOException {
// Skip empty writes
- if (len == 0)
- return;
+ if (len == 0) return;
// Last character was a new line, print the indent string
if (lastCharWasNL) {
@@ -568,7 +563,10 @@ public class ReportWriter extends Writer {
char c = cbuf[idx];
- if ((c == '\n' && lastChar != '\r') || c == '\n' || c == '\r' || c == '\f') {
+ if ((c == '\n' && lastChar != '\r')
+ || c == '\n'
+ || c == '\r'
+ || c == '\f') {
writeNL(c);
} else {
if (lastCharWasNL) {
@@ -580,9 +578,7 @@ public class ReportWriter extends Writer {
if (c == '\t') {
linePos += tabEqv;
- for (int j = 0; j < tabEqv; j++) {
- contained.write(' ');
- }
+ for (int j = 0; j < tabEqv; j++) contained.write(' ');
} else {
linePos += 1;
contained.write(c);
@@ -597,12 +593,10 @@ public class ReportWriter extends Writer {
for (int j = 0; j < indentLevel; j++) {
IndentVal ival = iVals.get(j);
- if (printTabsAsSpaces)
- contained.write(ival.indentStrSpacedTabs);
- else
- contained.write(ival.indentStr);
+ if (printTabsAsSpaces) contained.write(ival.indentStrSpacedTabs);
+ else contained.write(ival.indentStr);
- linePos += ival.indentStrPos;
+ linePos += ival.indentStrPos;
indentPos += ival.indentStrPos;
}
}
@@ -626,4 +620,15 @@ public class ReportWriter extends Writer {
public String toString() {
return contained.toString();
}
+
+ /**
+ * Write out a formatted string.
+ *
+ * @param format The format string.
+ * @param items The arguments to the format string.
+ * @throws IOException Thrown if something goes wrong.
+ */
+ public void writef(String format, Object... items) throws IOException {
+ write(String.format(format, items));
+ }
}
diff --git a/base/src/main/java/bjc/utils/ioutils/RuleBasedConfigReader.java b/base/src/main/java/bjc/utils/ioutils/RuleBasedConfigReader.java
index 2ce2591..12534bd 100644
--- a/base/src/main/java/bjc/utils/ioutils/RuleBasedConfigReader.java
+++ b/base/src/main/java/bjc/utils/ioutils/RuleBasedConfigReader.java
@@ -6,14 +6,14 @@ import java.util.Scanner;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
-import bjc.data.IHolder;
-import bjc.data.IPair;
-import bjc.data.Identity;
+import bjc.data.Holder;
import bjc.data.Pair;
+import bjc.data.Identity;
+import bjc.data.SimplePair;
import bjc.utils.exceptions.UnknownPragma;
import bjc.funcdata.FunctionalMap;
import bjc.funcdata.FunctionalStringTokenizer;
-import bjc.funcdata.IMap;
+import bjc.funcdata.MapEx;
/**
* This class parses a rules based config file, and uses it to drive a provided
@@ -31,7 +31,7 @@ public class RuleBasedConfigReader<E> {
*
* Takes the tokenizer, and a pair of the read token and application state
*/
- private BiConsumer<FunctionalStringTokenizer, IPair<String, E>> start;
+ private BiConsumer<FunctionalStringTokenizer, Pair<String, E>> start;
/*
* Function to use when continuing a rule.
@@ -52,7 +52,7 @@ public class RuleBasedConfigReader<E> {
*
* Pragma actions are functions taking a tokenizer and application state
*/
- private final IMap<String, BiConsumer<FunctionalStringTokenizer, E>> pragmas;
+ private final MapEx<String, BiConsumer<FunctionalStringTokenizer, E>> pragmas;
/**
* Create a new rule-based config reader
@@ -65,7 +65,7 @@ public class RuleBasedConfigReader<E> {
* The action to fire when ending a rule
*/
public RuleBasedConfigReader(
- final BiConsumer<FunctionalStringTokenizer, IPair<String, E>> start,
+ final BiConsumer<FunctionalStringTokenizer, Pair<String, E>> start,
final BiConsumer<FunctionalStringTokenizer, E> continueRule,
final Consumer<E> end) {
this.start = start;
@@ -161,7 +161,7 @@ public class RuleBasedConfigReader<E> {
/*
* This is true when a rule's open
*/
- final IHolder<Boolean> isRuleOpen = new Identity<>(false);
+ final Holder<Boolean> isRuleOpen = new Identity<>(false);
/*
* Do something for every line of the file
@@ -237,7 +237,7 @@ public class RuleBasedConfigReader<E> {
* The action to execute on starting of a rule
*/
public void setStartRule(
- final BiConsumer<FunctionalStringTokenizer, IPair<String, E>> start) {
+ final BiConsumer<FunctionalStringTokenizer, Pair<String, E>> start) {
if (start == null)
throw new NullPointerException("Action on rule start must be non-null");
@@ -270,7 +270,7 @@ public class RuleBasedConfigReader<E> {
/*
* Handle pragmas
*/
- pragmas.getOrDefault(token, (tokenzer, stat) -> {
+ pragmas.get(token).orElse((tokenzer, stat) -> {
throw new UnknownPragma("Unknown pragma " + token);
}).accept(tokenizer, state);
} else {
@@ -284,7 +284,7 @@ public class RuleBasedConfigReader<E> {
/*
* Start a rule
*/
- start.accept(tokenizer, new Pair<>(nextToken, state));
+ start.accept(tokenizer, new SimplePair<>(nextToken, state));
isRuleOpen = true;
}
diff --git a/base/src/main/java/bjc/utils/ioutils/SimpleProperties.java b/base/src/main/java/bjc/utils/ioutils/SimpleProperties.java
index 754ed45..d380866 100644
--- a/base/src/main/java/bjc/utils/ioutils/SimpleProperties.java
+++ b/base/src/main/java/bjc/utils/ioutils/SimpleProperties.java
@@ -170,17 +170,20 @@ public class SimpleProperties implements Map<String, String> {
return props.isEmpty();
}
- @Override
+ @SuppressWarnings("unlikely-arg-type")
+ @Override
public boolean containsKey(final Object key) {
return props.containsKey(key);
}
- @Override
+ @SuppressWarnings("unlikely-arg-type")
+ @Override
public boolean containsValue(final Object value) {
return props.containsValue(value);
}
- @Override
+ @SuppressWarnings("unlikely-arg-type")
+ @Override
public String get(final Object key) {
return props.get(key);
}
@@ -190,7 +193,8 @@ public class SimpleProperties implements Map<String, String> {
return props.put(key, value);
}
- @Override
+ @SuppressWarnings("unlikely-arg-type")
+ @Override
public String remove(final Object key) {
return props.remove(key);
}
diff --git a/base/src/main/java/bjc/utils/gui/TextAreaOutputStream.java b/base/src/main/java/bjc/utils/ioutils/TextAreaOutputStream.java
index fbc58ed..10f5104 100644
--- a/base/src/main/java/bjc/utils/gui/TextAreaOutputStream.java
+++ b/base/src/main/java/bjc/utils/ioutils/TextAreaOutputStream.java
@@ -1,4 +1,4 @@
-package bjc.utils.gui;
+package bjc.utils.ioutils;
import java.io.IOException;
import java.io.OutputStream;
diff --git a/base/src/main/java/bjc/utils/misc/Direction.java b/base/src/main/java/bjc/utils/misc/Direction.java
index cc24b49..2026349 100644
--- a/base/src/main/java/bjc/utils/misc/Direction.java
+++ b/base/src/main/java/bjc/utils/misc/Direction.java
@@ -7,7 +7,7 @@ import org.apache.commons.lang3.text.WordUtils;
import bjc.utils.exceptions.DirectionInvalid;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A set of cardinal directions
@@ -54,7 +54,7 @@ public enum Direction {
*
* @return A list of all the cardinal directions
*/
- public static IList<Direction> cardinals() {
+ public static ListEx<Direction> cardinals() {
return new FunctionalList<>(NORTH, SOUTH, EAST, WEST);
}
@@ -84,7 +84,7 @@ public enum Direction {
+ nCardinals);
}
- IList<Direction> cards = cardinals();
+ ListEx<Direction> cards = cardinals();
for (int i = 0; i <= 4 - nCardinals; i++) {
Direction rDir = cards.randItem(RNG::nextInt);
diff --git a/base/src/main/java/bjc/utils/misc/RelativeDirection.java b/base/src/main/java/bjc/utils/misc/RelativeDirection.java
index 4239b0d..87d9e10 100644
--- a/base/src/main/java/bjc/utils/misc/RelativeDirection.java
+++ b/base/src/main/java/bjc/utils/misc/RelativeDirection.java
@@ -5,7 +5,7 @@ import java.util.function.Consumer;
import bjc.utils.exceptions.DirectionInvalid;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Represents a direction that is relative to another direction
@@ -62,7 +62,7 @@ public enum RelativeDirection {
+ numDirections);
}
- IList<RelativeDirection> relativeDirs = new FunctionalList<>(values());
+ ListEx<RelativeDirection> relativeDirs = new FunctionalList<>(values());
if (ignoreBackwards) {
relativeDirs.removeMatching(BACKWARD);
diff --git a/base/src/main/java/bjc/utils/parserutils/IPrecedent.java b/base/src/main/java/bjc/utils/parserutils/Precedent.java
index eb164b3..33b032c 100644
--- a/base/src/main/java/bjc/utils/parserutils/IPrecedent.java
+++ b/base/src/main/java/bjc/utils/parserutils/Precedent.java
@@ -7,7 +7,7 @@ package bjc.utils.parserutils;
*
*/
@FunctionalInterface
-public interface IPrecedent {
+public interface Precedent {
/**
* Create a new object with set precedence
*
@@ -15,7 +15,7 @@ public interface IPrecedent {
* The precedence of the object to handle
* @return A new object with set precedence
*/
- public static IPrecedent newSimplePrecedent(final int precedence) {
+ public static Precedent newSimplePrecedent(final int precedence) {
return () -> precedence;
}
diff --git a/base/src/main/java/bjc/utils/parserutils/ShuntingYard.java b/base/src/main/java/bjc/utils/parserutils/ShuntingYard.java
index 2418517..f0475ff 100644
--- a/base/src/main/java/bjc/utils/parserutils/ShuntingYard.java
+++ b/base/src/main/java/bjc/utils/parserutils/ShuntingYard.java
@@ -6,8 +6,8 @@ import java.util.function.Function;
import bjc.funcdata.FunctionalList;
import bjc.funcdata.FunctionalMap;
-import bjc.funcdata.IList;
-import bjc.funcdata.IMap;
+import bjc.funcdata.ListEx;
+import bjc.funcdata.MapEx;
import bjc.utils.funcutils.StringUtils;
/**
@@ -25,7 +25,7 @@ public class ShuntingYard<TokenType> {
* @author ben
*
*/
- public static enum Operator implements IPrecedent {
+ public static enum Operator implements Precedent {
/**
* Represents addition.
*/
@@ -59,7 +59,7 @@ public class ShuntingYard<TokenType> {
/*
* Holds all the shuntable operations.
*/
- private IMap<String, IPrecedent> operators;
+ private MapEx<String, Precedent> operators;
/**
* Create a new shunting yard with a default set of operators.
@@ -95,7 +95,7 @@ public class ShuntingYard<TokenType> {
/*
* Create the precedence marker
*/
- final IPrecedent prec = IPrecedent.newSimplePrecedent(precedence);
+ final Precedent prec = Precedent.newSimplePrecedent(precedence);
this.addOp(operator, prec);
}
@@ -109,14 +109,12 @@ public class ShuntingYard<TokenType> {
* @param precedence
* The precedence of the operator.
*/
- public void addOp(final String operator, final IPrecedent precedence) {
+ public void addOp(final String operator, final Precedent precedence) {
/*
* Complain about trying to add an incorrect operator
*/
- if (operator == null)
- throw new NullPointerException("Operator must not be null");
- else if (precedence == null)
- throw new NullPointerException("Precedence must not be null");
+ if (operator == null) throw new NullPointerException("Operator must not be null");
+ else if (precedence == null) throw new NullPointerException("Precedence must not be null");
/*
* Add the operator to the ones we handle
@@ -140,8 +138,8 @@ public class ShuntingYard<TokenType> {
/*
* Get the precedence of operators
*/
- final int rightPrecedence = operators.get(right).getPrecedence();
- final int leftPrecedence = operators.get(left).getPrecedence();
+ final int rightPrecedence = operators.get(right).get().getPrecedence();
+ final int leftPrecedence = operators.get(left).get().getPrecedence();
/*
* Evaluate what we were asked
@@ -160,20 +158,18 @@ public class ShuntingYard<TokenType> {
*
* @return A list of tokens in postfix notation.
*/
- public IList<TokenType> postfix(final IList<String> input,
+ public ListEx<TokenType> postfix(final ListEx<String> input,
final Function<String, TokenType> transformer) {
/*
* Check our input
*/
- if (input == null)
- throw new NullPointerException("Input must not be null");
- else if (transformer == null)
- throw new NullPointerException("Transformer must not be null");
+ if (input == null) throw new NullPointerException("Input must not be null");
+ else if (transformer == null) throw new NullPointerException("Transformer must not be null");
/*
* Here's what we're handing back
*/
- final IList<TokenType> output = new FunctionalList<>();
+ final ListEx<TokenType> output = new FunctionalList<>();
/*
* The stack to put operators on
@@ -226,9 +222,7 @@ public class ShuntingYard<TokenType> {
}
}
- for (String token : stack) {
- output.add(transformer.apply(token));
- }
+ for (String token : stack) output.add(transformer.apply(token));
return output;
}
@@ -244,10 +238,7 @@ public class ShuntingYard<TokenType> {
/*
* Check if we want to remove all operators
*/
- if (operator == null) {
- operators = new FunctionalMap<>();
- } else {
- operators.remove(operator);
- }
+ if (operator == null) operators = new FunctionalMap<>();
+ else operators.remove(operator);
}
}
diff --git a/base/src/main/java/bjc/utils/parserutils/StringDescaper.java b/base/src/main/java/bjc/utils/parserutils/StringDescaper.java
index f8868e6..7052588 100644
--- a/base/src/main/java/bjc/utils/parserutils/StringDescaper.java
+++ b/base/src/main/java/bjc/utils/parserutils/StringDescaper.java
@@ -12,7 +12,7 @@ import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
/**
- * <<<<<<< Updated upstream Customizable string escapes.
+ * Customizable string escapes.
*
* @author Benjamin Culkin
*/
diff --git a/base/src/main/java/bjc/utils/parserutils/TokenTransformer.java b/base/src/main/java/bjc/utils/parserutils/TokenTransformer.java
deleted file mode 100644
index 6cf2da5..0000000
--- a/base/src/main/java/bjc/utils/parserutils/TokenTransformer.java
+++ /dev/null
@@ -1,148 +0,0 @@
-package bjc.utils.parserutils;
-
-import java.util.Deque;
-import java.util.function.Consumer;
-import java.util.function.Function;
-import java.util.function.Predicate;
-import java.util.function.UnaryOperator;
-
-import bjc.data.IHolder;
-import bjc.data.ITree;
-import bjc.data.Pair;
-import bjc.data.Tree;
-import bjc.utils.parserutils.TreeConstructor.ConstructorState;
-import bjc.utils.parserutils.TreeConstructor.QueueFlattener;
-
-/*
- * Handle creating ASTs from tokens.
- */
-final class TokenTransformer<TokenType> implements Consumer<TokenType> {
- /*
- * Handle operators
- */
- private final class OperatorHandler
- implements UnaryOperator<ConstructorState<TokenType>> {
- /* The handled element. */
- private final TokenType element;
-
- /* Create a new operator handler. */
- public OperatorHandler(final TokenType element) {
- this.element = element;
- }
-
- @Override
- public ConstructorState<TokenType> apply(final ConstructorState<TokenType> pair) {
- /*
- * Replace the current AST with the result of handling an operator
- */
- return new ConstructorState<>(
- pair.bindLeft(queuedASTs -> handleOperator(queuedASTs)));
- }
-
- private ConstructorState<TokenType>
- handleOperator(final Deque<ITree<TokenType>> queuedASTs) {
- /*
- * The AST we're going to hand back
- */
- ITree<TokenType> newAST;
-
- /*
- * Handle special operators
- */
- if (isSpecialOperator.test(element)) {
- newAST = handleSpecialOperator.apply(element).apply(queuedASTs);
- } else {
- /*
- * Error if we don't have enough for a binary operator
- */
- if (queuedASTs.size() < 2) {
- final String msg = String.format(
- "Attempted to parse binary operator without enough operands\n\tProblem operator is: %s\n\tPossible operand is: %s",
- element.toString(), queuedASTs.peek().toString());
-
- throw new IllegalStateException(msg);
- }
-
- /*
- * Grab the two operands
- */
- final ITree<TokenType> right = queuedASTs.pop();
- final ITree<TokenType> left = queuedASTs.pop();
-
- /*
- * Create a new AST
- */
- newAST = new Tree<>(element, left, right);
- }
-
- /*
- * Stick it onto the stack
- */
- queuedASTs.push(newAST);
-
- /*
- * Hand back the state
- */
- return new ConstructorState<>(queuedASTs, newAST);
- }
- }
-
- /* The initial state of the transformer. */
- private final IHolder<ConstructorState<TokenType>> initialState;
-
- /* The predicate tot use to detect operators. */
- private final Predicate<TokenType> operatorPredicate;
-
- /* The predicate for detecting special operators. */
- private final Predicate<TokenType> isSpecialOperator;
- /* The function for handling special operators. */
- private final Function<TokenType, QueueFlattener<TokenType>> handleSpecialOperator;
-
- /**
- * Create a new transformer
- *
- * @param initialState
- * The initial state of the transformer.
- *
- * @param operatorPredicate
- * The predicate to use to identify operators.
- *
- * @param isSpecialOperator
- * The predicate used to identify special
- * operators.
- *
- * @param handleSpecialOperator
- * The function used for handling special
- * operators.
- */
- public TokenTransformer(final IHolder<ConstructorState<TokenType>> initialState,
- final Predicate<TokenType> operatorPredicate,
- final Predicate<TokenType> isSpecialOperator,
- final Function<TokenType, QueueFlattener<TokenType>> handleSpecialOperator) {
- this.initialState = initialState;
- this.operatorPredicate = operatorPredicate;
- this.isSpecialOperator = isSpecialOperator;
- this.handleSpecialOperator = handleSpecialOperator;
- }
-
- @Override
- public void accept(final TokenType element) {
- /*
- * Handle operators
- */
- if (operatorPredicate.test(element)) {
- initialState.transform(new OperatorHandler(element));
- } else {
- final ITree<TokenType> newAST = new Tree<>(element);
-
- /*
- * Insert the new tree into the AST
- */
- initialState.transform(pair -> new ConstructorState<>(pair.bindLeft(queue -> {
- queue.push(newAST);
-
- return new Pair<>(queue, newAST);
- })));
- }
- }
-}
diff --git a/base/src/main/java/bjc/utils/parserutils/TokenUtils.java b/base/src/main/java/bjc/utils/parserutils/TokenUtils.java
index 81a7ba0..860bbdf 100644
--- a/base/src/main/java/bjc/utils/parserutils/TokenUtils.java
+++ b/base/src/main/java/bjc/utils/parserutils/TokenUtils.java
@@ -4,13 +4,13 @@ import static bjc.utils.misc.PropertyDB.applyFormat;
import static bjc.utils.misc.PropertyDB.getCompiledRegex;
import static bjc.utils.misc.PropertyDB.getRegex;
-import java.util.LinkedList;
-import java.util.List;
+import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
+import bjc.data.*;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.utils.parserutils.splitter.TokenSplitter;
/**
@@ -28,7 +28,7 @@ public class TokenUtils {
*/
public static class StringTokenSplitter implements TokenSplitter {
@Override
- public IList<String> split(final String input) {
+ public ListEx<String> split(final String input) {
return new FunctionalList<>(TokenUtils.removeDQuotedStrings(input));
}
}
@@ -316,4 +316,28 @@ public class TokenUtils {
return false;
}
}
+
+ /**
+ * Split a line into a series of space-separated arguments, including string
+ * literals.
+ *
+ * @param com
+ * The command to split from
+ * @return The split arguments.
+ */
+ public static List<String> processArguments(String com) {
+ List<String> strings = new ArrayList<>();
+
+ BooleanToggle togg = new BooleanToggle();
+
+ for (String strang : removeDQuotedStrings(com)) {
+ if (togg.get()) {
+ strings.add(descapeString(strang));
+ } else {
+ for (String strung : strang.split("\\s+")) strings.add(strung);
+ }
+ }
+
+ return strings;
+ }
}
diff --git a/base/src/main/java/bjc/utils/parserutils/TreeConstructor.java b/base/src/main/java/bjc/utils/parserutils/TreeConstructor.java
index 3c7509b..bd907b5 100644
--- a/base/src/main/java/bjc/utils/parserutils/TreeConstructor.java
+++ b/base/src/main/java/bjc/utils/parserutils/TreeConstructor.java
@@ -2,15 +2,11 @@ package bjc.utils.parserutils;
import java.util.Deque;
import java.util.LinkedList;
-import java.util.function.Function;
-import java.util.function.Predicate;
+import java.util.function.*;
-import bjc.data.IHolder;
-import bjc.data.IPair;
-import bjc.data.ITree;
-import bjc.data.Identity;
-import bjc.data.Pair;
-import bjc.funcdata.IList;
+import bjc.data.*;
+import bjc.funcdata.ListEx;
+import bjc.utils.parserutils.TreeConstructor.*;
/**
* Creates a parse tree from a postfix expression.
@@ -26,7 +22,7 @@ public class TreeConstructor {
* The token type of the tree.
*/
public interface QueueFlattener<TokenType>
- extends Function<Deque<ITree<TokenType>>, ITree<TokenType>> {
+ extends Function<Deque<Tree<TokenType>>, Tree<TokenType>> {
/*
* Alias
*/
@@ -34,14 +30,14 @@ public class TreeConstructor {
/* Alias for constructor state. */
static final class ConstructorState<TokenType>
- extends Pair<Deque<ITree<TokenType>>, ITree<TokenType>> {
- public ConstructorState(final Deque<ITree<TokenType>> left,
- final ITree<TokenType> right) {
+ extends SimplePair<Deque<Tree<TokenType>>, Tree<TokenType>> {
+ public ConstructorState(final Deque<Tree<TokenType>> left,
+ final Tree<TokenType> right) {
super(left, right);
}
public ConstructorState(
- final IPair<Deque<ITree<TokenType>>, ITree<TokenType>> par) {
+ final Pair<Deque<Tree<TokenType>>, Tree<TokenType>> par) {
super(par.getLeft(), par.getRight());
}
}
@@ -63,8 +59,8 @@ public class TreeConstructor {
*
* @return A AST from the expression.
*/
- public static <TokenType> ITree<TokenType> constructTree(
- final IList<TokenType> tokens, final Predicate<TokenType> isOperator) {
+ public static <TokenType> Tree<TokenType> constructTree(
+ final ListEx<TokenType> tokens, final Predicate<TokenType> isOperator) {
/* Construct a tree with no special operators */
return constructTree(tokens, isOperator, op -> false, null);
}
@@ -96,26 +92,27 @@ public class TreeConstructor {
* @return A AST from the expression.
*
*/
- public static <TokenType> ITree<TokenType> constructTree(
- final IList<TokenType> tokens, final Predicate<TokenType> isOperator,
+ public static <TokenType> Tree<TokenType> constructTree(
+ final ListEx<TokenType> tokens, final Predicate<TokenType> isOperator,
final Predicate<TokenType> isSpecialOperator,
final Function<TokenType, QueueFlattener<TokenType>> handleSpecialOperator) {
/*
* Make sure our parameters are valid
*/
- if (tokens == null)
+ if (tokens == null) {
throw new NullPointerException("Tokens must not be null");
- else if (isOperator == null)
+ } else if (isOperator == null) {
throw new NullPointerException("Operator predicate must not be null");
- else if (isSpecialOperator == null)
+ } else if (isSpecialOperator == null) {
throw new NullPointerException(
"Special operator determiner must not be null");
+ }
final ConstructorState<TokenType> cstate
= new ConstructorState<>(new LinkedList<>(), null);
/* Here is the state for the tree construction */
- final IHolder<ConstructorState<TokenType>> initialState = new Identity<>(cstate);
+ final Holder<ConstructorState<TokenType>> initialState = new Identity<>(cstate);
/* Transform each of the tokens */
final TokenTransformer<TokenType> trans = new TokenTransformer<>(initialState,
@@ -127,3 +124,140 @@ public class TreeConstructor {
return initialState.unwrap(ConstructorState::getRight);
}
}
+
+/*
+ * Transform function on tokens
+ */
+class TokenTransformer<TokenType> implements Consumer<TokenType> {
+ /*
+ * Handle operators
+ */
+ private final class OperatorHandler
+ implements UnaryOperator<ConstructorState<TokenType>> {
+ /* The handled element. */
+ private final TokenType element;
+
+ /* Create a new operator handler. */
+ public OperatorHandler(final TokenType element) {
+ this.element = element;
+ }
+
+ @Override
+ public ConstructorState<TokenType> apply(final ConstructorState<TokenType> pair) {
+ /*
+ * Replace the current AST with the result of handling an operator
+ */
+ return new ConstructorState<>(
+ pair.bindLeft(queuedASTs -> handleOperator(queuedASTs)));
+ }
+
+ private ConstructorState<TokenType>
+ handleOperator(final Deque<Tree<TokenType>> queuedASTs) {
+ /*
+ * The AST we're going to hand back
+ */
+ Tree<TokenType> newAST;
+
+ /*
+ * Handle special operators
+ */
+ if (isSpecialOperator.test(element)) {
+ newAST = handleSpecialOperator.apply(element).apply(queuedASTs);
+ } else {
+ /*
+ * Error if we don't have enough for a binary operator
+ */
+ if (queuedASTs.size() < 2) {
+ final String msg = String.format(
+ "Attempted to parse binary operator without enough operands\n\tProblem operator is: %s\n\tPossible operand is: %s",
+ element.toString(), queuedASTs.peek().toString());
+
+ throw new IllegalStateException(msg);
+ }
+
+ /*
+ * Grab the two operands
+ */
+ final Tree<TokenType> right = queuedASTs.pop();
+ final Tree<TokenType> left = queuedASTs.pop();
+
+ /*
+ * Create a new AST
+ */
+ newAST = new SimpleTree<>(element, left, right);
+ }
+
+ /*
+ * Stick it onto the stack
+ */
+ queuedASTs.push(newAST);
+
+ /*
+ * Hand back the state
+ */
+ return new ConstructorState<>(queuedASTs, newAST);
+ }
+ }
+
+ /* The initial state of the transformer. */
+ private final Holder<ConstructorState<TokenType>> initialState;
+
+ /* The predicate tot use to detect operators. */
+ private final Predicate<TokenType> operatorPredicate;
+
+ /* The predicate for detecting special operators. */
+ private final Predicate<TokenType> isSpecialOperator;
+ /* The function for handling special operators. */
+ private final Function<TokenType, QueueFlattener<TokenType>> handleSpecialOperator;
+
+ /**
+ * Create a new transformer
+ *
+ * @param initialState
+ * The initial state of the transformer.
+ *
+ * @param operatorPredicate
+ * The predicate to use to identify operators.
+ *
+ * @param isSpecialOperator
+ * The predicate used to identify special
+ * operators.
+ *
+ * @param handleSpecialOperator
+ * The function used for handling special
+ * operators.
+ */
+ public TokenTransformer(final Holder<ConstructorState<TokenType>> initialState,
+ final Predicate<TokenType> operatorPredicate,
+ final Predicate<TokenType> isSpecialOperator,
+ final Function<TokenType, QueueFlattener<TokenType>> handleSpecialOperator) {
+ this.initialState = initialState;
+ this.operatorPredicate = operatorPredicate;
+ this.isSpecialOperator = isSpecialOperator;
+ this.handleSpecialOperator = handleSpecialOperator;
+ }
+
+ @Override
+ public void accept(final TokenType element) {
+ /*
+ * Handle operators
+ */
+ if (operatorPredicate.test(element)) {
+ initialState.transform(new OperatorHandler(element));
+ } else {
+ final Tree<TokenType> newAST = new SimpleTree<>(element);
+
+ /*
+ * Insert the new tree into the AST
+ */
+ initialState.transform(pair -> new ConstructorState<>(
+ pair.bindLeft(queue -> {
+ queue.push(newAST);
+
+ return new SimplePair<>(queue, newAST);
+ })
+ )
+ );
+ }
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/parserutils/delims/DelimiterGroup.java b/base/src/main/java/bjc/utils/parserutils/delims/DelimiterGroup.java
index ba61531..75f777c 100644
--- a/base/src/main/java/bjc/utils/parserutils/delims/DelimiterGroup.java
+++ b/base/src/main/java/bjc/utils/parserutils/delims/DelimiterGroup.java
@@ -11,12 +11,12 @@ import java.util.Set;
import java.util.function.BiPredicate;
import java.util.function.Function;
-import bjc.data.IPair;
-import bjc.data.ITree;
import bjc.data.Pair;
import bjc.data.Tree;
+import bjc.data.SimplePair;
+import bjc.data.SimpleTree;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Represents a possible delimiter group to match.
@@ -37,12 +37,12 @@ public class DelimiterGroup<T> {
/*
* The contents of this group.
*/
- private final Deque<ITree<T>> contents;
+ private final Deque<Tree<T>> contents;
/*
* The contents of the current subgroup.
*/
- private IList<ITree<T>> currentGroup;
+ private ListEx<Tree<T>> currentGroup;
/*
* The token that opened the group, and any opening parameters.
@@ -74,7 +74,7 @@ public class DelimiterGroup<T> {
* @param itm
* The item to add to this group instance.
*/
- public void addItem(final ITree<T> itm) {
+ public void addItem(final Tree<T> itm) {
currentGroup.add(itm);
}
@@ -91,8 +91,8 @@ public class DelimiterGroup<T> {
/*
* Add all of the contents to the subgroup.
*/
- final ITree<T> subgroupContents = new Tree<>(chars.contents);
- for (final ITree<T> itm : currentGroup) {
+ final Tree<T> subgroupContents = new SimpleTree<>(chars.contents);
+ for (final Tree<T> itm : currentGroup) {
subgroupContents.addChild(itm);
}
@@ -100,7 +100,7 @@ public class DelimiterGroup<T> {
* Handle subordinate sub-groups.
*/
while (!contents.isEmpty()) {
- final ITree<T> possibleSubordinate = contents.peek();
+ final Tree<T> possibleSubordinate = contents.peek();
/*
* Subordinate lower priority subgroups.
@@ -118,8 +118,8 @@ public class DelimiterGroup<T> {
}
}
- final Tree<T> subgroup
- = new Tree<>(chars.subgroup, subgroupContents, new Tree<>(marker));
+ final SimpleTree<T> subgroup
+ = new SimpleTree<>(chars.subgroup, subgroupContents, new SimpleTree<>(marker));
contents.push(subgroup);
@@ -137,7 +137,7 @@ public class DelimiterGroup<T> {
*
* @return This group as a tree.
*/
- public ITree<T> toTree(final T closer, final SequenceCharacteristics<T> chars) {
+ public Tree<T> toTree(final T closer, final SequenceCharacteristics<T> chars) {
/*
* Mark any implied subgroups.
*/
@@ -146,7 +146,7 @@ public class DelimiterGroup<T> {
}
/* The resulting tree. */
- final ITree<T> res = new Tree<>(chars.contents);
+ final Tree<T> res = new SimpleTree<>(chars.contents);
/*
* Add either the contents of the current group, or subgroups if they're
@@ -162,7 +162,7 @@ public class DelimiterGroup<T> {
currentGroup.forEach(res::addChild);
}
- return new Tree<>(groupName, new Tree<>(opener), res, new Tree<>(closer));
+ return new SimpleTree<>(groupName, new SimpleTree<>(opener), res, new SimpleTree<>(closer));
}
@Override
@@ -259,18 +259,18 @@ public class DelimiterGroup<T> {
*
* @return The name of the group T opens, or null if it doesn't open one.
*/
- public IPair<T, T[]> doesOpen(final T marker) {
+ public Pair<T, T[]> doesOpen(final T marker) {
if (openDelimiters.containsKey(marker))
- return new Pair<>(openDelimiters.get(marker), null);
+ return new SimplePair<>(openDelimiters.get(marker), null);
- for (final Function<T, IPair<T, T[]>> pred : predOpeners) {
- final IPair<T, T[]> par = pred.apply(marker);
+ for (final Function<T, Pair<T, T[]>> pred : predOpeners) {
+ final Pair<T, T[]> par = pred.apply(marker);
if (par.getLeft() != null)
return par;
}
- return new Pair<>(null, null);
+ return new SimplePair<>(null, null);
}
/**
@@ -312,7 +312,7 @@ public class DelimiterGroup<T> {
private final Map<T, T> impliedSubgroups;
/* Allows more complex openings */
- private final List<Function<T, IPair<T, T[]>>> predOpeners;
+ private final List<Function<T, Pair<T, T[]>>> predOpeners;
/* Allow more complex closings */
private final List<BiPredicate<T, T[]>> predClosers;
@@ -562,7 +562,7 @@ public class DelimiterGroup<T> {
* @param pred
* The predicate that defines the opener and its parameters.
*/
- public void addPredOpener(final Function<T, IPair<T, T[]>> pred) {
+ public void addPredOpener(final Function<T, Pair<T, T[]>> pred) {
predOpeners.add(pred);
}
diff --git a/base/src/main/java/bjc/utils/parserutils/delims/RegexOpener.java b/base/src/main/java/bjc/utils/parserutils/delims/RegexOpener.java
index f08201c..db5c3ca 100644
--- a/base/src/main/java/bjc/utils/parserutils/delims/RegexOpener.java
+++ b/base/src/main/java/bjc/utils/parserutils/delims/RegexOpener.java
@@ -4,8 +4,8 @@ import java.util.function.Function;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
-import bjc.data.IPair;
import bjc.data.Pair;
+import bjc.data.SimplePair;
/**
* A predicated opener for use with {@link RegexCloser}
@@ -13,7 +13,7 @@ import bjc.data.Pair;
* @author bjculkin
*
*/
-public class RegexOpener implements Function<String, IPair<String, String[]>> {
+public class RegexOpener implements Function<String, Pair<String, String[]>> {
/* The name of the group. */
private final String name;
/* The pattern that marks an opening group. */
@@ -35,7 +35,7 @@ public class RegexOpener implements Function<String, IPair<String, String[]>> {
}
@Override
- public IPair<String, String[]> apply(final String str) {
+ public Pair<String, String[]> apply(final String str) {
final Matcher m = patt.matcher(str);
if (m.matches()) {
@@ -47,9 +47,9 @@ public class RegexOpener implements Function<String, IPair<String, String[]>> {
parms[i] = m.group(i);
}
- return new Pair<>(name, parms);
+ return new SimplePair<>(name, parms);
}
- return new Pair<>(null, null);
+ return new SimplePair<>(null, null);
}
}
diff --git a/base/src/main/java/bjc/utils/parserutils/delims/SequenceDelimiter.java b/base/src/main/java/bjc/utils/parserutils/delims/SequenceDelimiter.java
index 195c1f1..b3f6dc4 100644
--- a/base/src/main/java/bjc/utils/parserutils/delims/SequenceDelimiter.java
+++ b/base/src/main/java/bjc/utils/parserutils/delims/SequenceDelimiter.java
@@ -9,13 +9,13 @@ import com.google.common.collect.HashMultiset;
import com.google.common.collect.Multimap;
import com.google.common.collect.Multiset;
-import bjc.data.IPair;
-import bjc.data.ITree;
+import bjc.data.Pair;
import bjc.data.Tree;
+import bjc.data.SimpleTree;
import bjc.esodata.PushdownMap;
import bjc.esodata.SimpleStack;
import bjc.esodata.Stack;
-import bjc.funcdata.IMap;
+import bjc.funcdata.MapEx;
import bjc.utils.funcutils.StringUtils;
/**
@@ -82,7 +82,7 @@ public class SequenceDelimiter<T> {
* delimitation.
*
*/
- public ITree<T> delimitSequence(final SequenceCharacteristics<T> chars,
+ public Tree<T> delimitSequence(final SequenceCharacteristics<T> chars,
@SuppressWarnings("unchecked") final T... seq) throws DelimiterException {
if (initialGroup == null) {
throw new NullPointerException("Initial group must be specified.");
@@ -105,7 +105,7 @@ public class SequenceDelimiter<T> {
allowedDelimiters.push(HashMultimap.create());
/* Map of who forbid what for debugging purposes. */
- final IMap<T, T> whoForbid = new PushdownMap<>();
+ final MapEx<T, T> whoForbid = new PushdownMap<>();
/*
* Process each member of the sequence.
@@ -114,7 +114,7 @@ public class SequenceDelimiter<T> {
final T tok = seq[i];
/* Check if this token could open a group. */
- final IPair<T, T[]> possibleOpenPar = groupStack.top().doesOpen(tok);
+ final Pair<T, T[]> possibleOpenPar = groupStack.top().doesOpen(tok);
T possibleOpen = possibleOpenPar.getLeft();
if (possibleOpen == null) {
@@ -143,11 +143,7 @@ public class SequenceDelimiter<T> {
if (isForbidden(groupStack, forbiddenDelimiters, possibleOpen)) {
T forbiddenBy;
- if (whoForbid.containsKey(tok)) {
- forbiddenBy = whoForbid.get(tok);
- } else {
- forbiddenBy = groupStack.top().getName();
- }
+ forbiddenBy = whoForbid.get(tok).orElse(groupStack.top().getName());
final String ctxList
= StringUtils.toEnglishList(groupStack.toArray(), "then");
@@ -221,7 +217,7 @@ public class SequenceDelimiter<T> {
groupStack.top().markSubgroup(tok, chars);
} else {
/* Add an item to the group. */
- groupStack.top().addItem(new Tree<>(tok));
+ groupStack.top().addItem(new SimpleTree<>(tok));
}
}
diff --git a/base/src/main/java/bjc/utils/parserutils/delims/StringDelimiter.java b/base/src/main/java/bjc/utils/parserutils/delims/StringDelimiter.java
index 6035ede..575066d 100644
--- a/base/src/main/java/bjc/utils/parserutils/delims/StringDelimiter.java
+++ b/base/src/main/java/bjc/utils/parserutils/delims/StringDelimiter.java
@@ -1,6 +1,6 @@
package bjc.utils.parserutils.delims;
-import bjc.data.ITree;
+import bjc.data.Tree;
/**
* A sequence delimiter specialized for strings.
@@ -26,7 +26,7 @@ public class StringDelimiter extends SequenceDelimiter<String> {
*
* @see SequenceDelimiter
*/
- public ITree<String> delimitSequence(final String... seq) throws DelimiterException {
+ public Tree<String> delimitSequence(final String... seq) throws DelimiterException {
return super.delimitSequence(
new SequenceCharacteristics<>("root", "contents", "subgroup"), seq);
}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/ChainTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/ChainTokenSplitter.java
index 0844b5b..c60b6f2 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/ChainTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/ChainTokenSplitter.java
@@ -1,7 +1,7 @@
package bjc.utils.parserutils.splitter;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A token splitter that chains several other splitters together.
@@ -10,7 +10,7 @@ import bjc.funcdata.IList;
*
*/
public class ChainTokenSplitter implements TokenSplitter {
- private final IList<TokenSplitter> spliters;
+ private final ListEx<TokenSplitter> spliters;
/**
* Create a new chain token splitter.
@@ -40,8 +40,8 @@ public class ChainTokenSplitter implements TokenSplitter {
}
@Override
- public IList<String> split(final String input) {
- final IList<String> initList = new FunctionalList<>(input);
+ public ListEx<String> split(final String input) {
+ final ListEx<String> initList = new FunctionalList<>(input);
return spliters.reduceAux(initList, (splitter, strangs) -> strangs.flatMap(splitter::split));
}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/ConfigurableTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/ConfigurableTokenSplitter.java
index 16c1dc3..26d9dbe 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/ConfigurableTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/ConfigurableTokenSplitter.java
@@ -6,7 +6,7 @@ import java.util.LinkedHashSet;
import java.util.Set;
import java.util.regex.Pattern;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Split a string into pieces around a regular expression, and offer an easy way
@@ -111,7 +111,7 @@ public class ConfigurableTokenSplitter extends SimpleTokenSplitter {
}
@Override
- public IList<String> split(final String input) {
+ public ListEx<String> split(final String input) {
if (spliter == null)
throw new IllegalStateException("Must compile splitter before use");
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/ExcludingTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/ExcludingTokenSplitter.java
index 9a0cd65..52ce4bf 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/ExcludingTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/ExcludingTokenSplitter.java
@@ -5,7 +5,7 @@ import java.util.Set;
import java.util.function.Predicate;
import bjc.funcdata.FunctionalList;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A token splitter that will not split certain tokens.
@@ -15,7 +15,7 @@ import bjc.funcdata.IList;
*/
public class ExcludingTokenSplitter implements TokenSplitter {
private final Set<String> literalExclusions;
- private final IList<Predicate<String>> predExclusions;
+ private final ListEx<Predicate<String>> predExclusions;
private final TokenSplitter spliter;
@@ -59,7 +59,7 @@ public class ExcludingTokenSplitter implements TokenSplitter {
}
@Override
- public IList<String> split(final String input) {
+ public ListEx<String> split(final String input) {
if (literalExclusions.contains(input))
return new FunctionalList<>(input);
else if (predExclusions.anyMatch(pred -> pred.test(input)))
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/FilteredTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/FilteredTokenSplitter.java
index 85d72e2..70abbbc 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/FilteredTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/FilteredTokenSplitter.java
@@ -2,7 +2,7 @@ package bjc.utils.parserutils.splitter;
import java.util.function.Predicate;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A token splitter that removes tokens that match a predicate from the stream
@@ -31,7 +31,7 @@ public class FilteredTokenSplitter implements TokenSplitter {
}
@Override
- public IList<String> split(String input) {
+ public ListEx<String> split(String input) {
return source.split(input).getMatching(filter);
}
}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/IdentityTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/IdentityTokenSplitter.java
new file mode 100644
index 0000000..b3b1b29
--- /dev/null
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/IdentityTokenSplitter.java
@@ -0,0 +1,16 @@
+package bjc.utils.parserutils.splitter;
+
+import bjc.funcdata.*;
+
+/**
+ * The token splitter that doesn't actually perform any splitting.
+ *
+ * @author Ben Culkin
+ *
+ */
+public class IdentityTokenSplitter implements TokenSplitter {
+ @Override
+ public ListEx<String> split(String input) {
+ return new FunctionalList<>(input);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/SimpleTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/SimpleTokenSplitter.java
index 6d88b20..df9ce70 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/SimpleTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/SimpleTokenSplitter.java
@@ -2,7 +2,7 @@ package bjc.utils.parserutils.splitter;
import java.util.regex.Pattern;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
import bjc.functypes.ID;
import bjc.utils.ioutils.RegexStringEditor;
@@ -39,7 +39,7 @@ public class SimpleTokenSplitter implements TokenSplitter {
}
@Override
- public IList<String> split(final String input) {
+ public ListEx<String> split(final String input) {
if (keepDelim) {
return RegexStringEditor.mapOccurances(input, spliter, ID.id(), ID.id());
}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitter.java
index 59e73e8..e833c21 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitter.java
@@ -1,6 +1,6 @@
package bjc.utils.parserutils.splitter;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* Split a string into a list of pieces.
@@ -17,5 +17,5 @@ public interface TokenSplitter {
*
* @return The pieces of the string.
*/
- public IList<String> split(String input);
+ public ListEx<String> split(String input);
}
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitters.java b/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitters.java
index 15d6b8b..4aed8f6 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitters.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/TokenSplitters.java
@@ -25,6 +25,15 @@ public class TokenSplitters {
}
/**
+ * Create a new identity token splitter, which doesn't actually do any splitting.
+ *
+ * @return A new identity splitter.
+ */
+ public static TokenSplitter identitySplitter() {
+ return new IdentityTokenSplitter();
+ }
+
+ /**
* Create a new transforming token splitter.
*
* @param splitter
diff --git a/base/src/main/java/bjc/utils/parserutils/splitter/TransformTokenSplitter.java b/base/src/main/java/bjc/utils/parserutils/splitter/TransformTokenSplitter.java
index b9fbedc..63c3206 100644
--- a/base/src/main/java/bjc/utils/parserutils/splitter/TransformTokenSplitter.java
+++ b/base/src/main/java/bjc/utils/parserutils/splitter/TransformTokenSplitter.java
@@ -2,7 +2,7 @@ package bjc.utils.parserutils.splitter;
import java.util.function.UnaryOperator;
-import bjc.funcdata.IList;
+import bjc.funcdata.ListEx;
/**
* A token splitter that performs a transform on the tokens from another
@@ -31,7 +31,7 @@ public class TransformTokenSplitter implements TokenSplitter {
}
@Override
- public IList<String> split(String input) {
+ public ListEx<String> split(String input) {
return source.split(input).map(transform);
}
diff --git a/base/src/main/java/bjc/utils/patterns/ComplexPattern.java b/base/src/main/java/bjc/utils/patterns/ComplexPattern.java
new file mode 100644
index 0000000..c6d72ec
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/ComplexPattern.java
@@ -0,0 +1,223 @@
+package bjc.utils.patterns;
+
+import java.util.function.*;
+import java.util.regex.*;
+
+import bjc.data.*;
+
+/**
+ * A pattern that can be matched against.
+ *
+ * @author Ben Culkin
+ *
+ * @param <InputType> The type of object being matched against.
+ * @param <ReturnType> The type returned by the pattern.
+ * @param <PredType> The state type returned by the predicate.
+ */
+public interface ComplexPattern<ReturnType, PredType, InputType> {
+ /**
+ * Whether or not the given input matches this pattern.
+ *
+ * @param input The object to check against this pattern.
+ *
+ * @return Whether or not this pattern is matched, as well as a state value
+ * that will get passed to the pattern if it did match.
+ */
+ Pair<Boolean, PredType> matches(InputType input);
+
+ /**
+ * Apply this pattern, once it has matched.
+ *
+ * @param input The object to apply this pattern to.
+ * @param state The state from the matcher.
+ *
+ * @return The result of applying this pattern.
+ */
+ ReturnType apply(InputType input, PredType state);
+
+ /* Pattern producing functions */
+
+ /**
+ * Create a pattern composed from a predicate & a function.
+ *
+ * @param <RetType> The type returned by the pattern.
+ * @param <PreType> The type used as intermediate state.
+ * @param <InpType> The type initially matched against.
+ *
+ * @param matcher The predicate that says what this pattern matches.
+ * @param accepter The action that happens when this pattern matches.
+ *
+ * @return A pattern composed from the passed in functions.
+ */
+ static <RetType, PreType, InpType> ComplexPattern<RetType, PreType, InpType> from(
+ Function<InpType, Pair<Boolean, PreType>> matcher,
+ BiFunction<InpType, PreType, RetType> accepter)
+ {
+ return new FunctionalPattern<>(matcher, accepter);
+ }
+
+ /**
+ * Create a pattern which checks if an object is of a given type (or a subtype of it).
+ *
+ * @param <ClassType> The type to check if the object is an instance of.
+ * @param <RetType> The type returned by the action.
+ * @param <InpType> The type of the thing to match.
+ *
+ * @param clasz The Class instance for the type you want to check.
+ * @param action The action to execute if the pattern does match.
+ *
+ * @return A pattern which follows the specified condition.
+ */
+ @SuppressWarnings("unchecked")
+ static <ClassType, RetType, InpType> ComplexPattern<RetType, ?, InpType> ofClass(
+ Class<ClassType> clasz,
+ Function<ClassType, RetType> action)
+ {
+ return from(
+ (input) -> Pair.pair(clasz.isInstance(input), null),
+ (input, ignored) -> action.apply((ClassType)input)
+ );
+ }
+
+ /**
+ * Creates a pattern which matches a given object.
+ *
+ * @param <RetType> The type returned when the pattern matches.
+ * @param <InpType> The type of the thing to match.
+ *
+ * @param obj The object being tested for equality.
+ * @param action The action to execute when the object matches.
+ *
+ * @return A pattern which tests against the equality of an object.
+ */
+ static <RetType, InpType> ComplexPattern<RetType, ?, InpType> matchesObject(
+ InpType obj,
+ Function<InpType, RetType> action
+ )
+ {
+ return from(
+ (input) -> Pair.pair(obj.equals(input), null),
+ (input, ignored) -> action.apply(input)
+ );
+ }
+
+ /**
+ * Tests if the toString rendition of an object matches a given condition.
+ *
+ * @param <RetType> The type returned by the pattern.
+ * @param <InpType> The type of the thing to match.
+ *
+ * @param pattern The string to check against.
+ * @param action The action to check when the toString of the object matches
+ * the provided string. This is passed both the object, and its
+ * string form (in the event that you don't want to call toString
+ * multiple times, for whatever reason)
+ *
+ * @return A pattern which tests against the toString representation of an object.
+ */
+ static <RetType, InpType> ComplexPattern<RetType, ?, InpType> equalsString(
+ String pattern,
+ BiFunction<InpType, String, RetType> action
+ )
+ {
+ Function<InpType, Pair<Boolean, String>> matcher = (input) -> {
+ String objString = input.toString();
+
+ return Pair.pair(pattern.equals(objString), objString);
+ };
+
+ return from(
+ matcher,
+ (input, objString) -> action.apply(input, objString)
+ );
+ }
+
+ /**
+ * Check if the toString of a given object matches a regex.
+ *
+ * @param <RetType> The type returned by the pattern.
+ * @param <InpType> The type of object to match against.
+ *
+ * @param regex The regex to match against.
+ * @param cond The predicate to use to determine if the regex matched.
+ * @param action The action to call when the regex matched.
+ *
+ * @return A pattern which does the regex matching.
+ */
+ static <RetType, InpType> ComplexPattern<RetType, Matcher, InpType> matchesRegex(
+ String regex,
+ Predicate<Matcher> cond,
+ BiFunction<InpType, Matcher, RetType> action
+ )
+ {
+ java.util.regex.Pattern regexPat = java.util.regex.Pattern.compile(regex);
+
+ Function<InpType, Pair<Boolean, Matcher>> matcher = (input) -> {
+ String inpString = input.toString();
+
+ Matcher mat = regexPat.matcher(inpString);
+
+ if (cond.test(mat)) return Pair.pair(true, mat);
+ else return Pair.pair(false, null);
+ };
+
+ return from(
+ matcher,
+ (input, res) -> action.apply(input, res)
+ );
+ }
+
+ // @TODO Nov 21, 2020 Ben Culkin :MorePatterns
+ // Try and write something to iterate over Iterator in a type-safe manner
+ // Also, something for doing a sub-pattern match
+
+ /**
+ * Create a pattern which will always execute.
+ *
+ * @param <RetType> The type returned.
+ * @param <InpType> The type being matched against.
+ *
+ * @param action The action to execute.
+ *
+ * @return A pattern which will be executed.
+ */
+ static <RetType, InpType> ComplexPattern<RetType, ?, InpType> otherwise(
+ Function<InpType, RetType> action
+ )
+ {
+ return from(
+ (input) -> Pair.pair(true, null),
+ (input, ignored) -> action.apply(input)
+ );
+ }
+
+ /**
+ * Create a pattern which checks if the string form of a given object starts
+ * with a specific string.
+ *
+ * @param <RetType> The type returned by the matcher.
+ * @param <InpType> The type being matched against.
+ *
+ * @param pattern The string to check against.
+ * @param action The action to execute.
+ *
+ * @return A pattern which functions as described.
+ */
+ static <RetType, InpType> ComplexPattern<RetType, String, InpType> startsWith(
+ String pattern,
+ Function<String, RetType> action)
+ {
+ return from((input) -> {
+ String objString = input.toString();
+
+ if (objString.startsWith(pattern)) {
+ return Pair.pair(
+ true,
+ objString.substring(
+ pattern.length()));
+ } else {
+ return Pair.pair(false, null);
+ }
+ }, (ignored, input) -> action.apply(input));
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/patterns/FunctionalPattern.java b/base/src/main/java/bjc/utils/patterns/FunctionalPattern.java
new file mode 100644
index 0000000..1e9dbab
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/FunctionalPattern.java
@@ -0,0 +1,47 @@
+package bjc.utils.patterns;
+
+import java.util.*;
+import java.util.function.*;
+
+import bjc.data.*;
+
+class FunctionalPattern<ReturnType, PredType, InputType>
+ implements ComplexPattern<ReturnType, PredType, InputType> {
+ private final Function<InputType, Pair<Boolean, PredType>> matcher;
+ private final BiFunction<InputType, PredType, ReturnType> accepter;
+
+ FunctionalPattern(
+ Function<InputType, Pair<Boolean, PredType>> matcher,
+ BiFunction<InputType, PredType, ReturnType> accepter) {
+ super();
+ this.matcher = matcher;
+ this.accepter = accepter;
+ }
+
+ @Override
+ public Pair<Boolean, PredType> matches(InputType input) {
+ return matcher.apply(input);
+ }
+
+ @Override
+ public ReturnType apply(InputType input, PredType state) {
+ return accepter.apply(input, state);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(accepter, matcher);
+ }
+
+ @Override
+ public boolean equals(Object obj) {
+ if (this == obj) return true;
+ if (obj == null) return false;
+ if (getClass() != obj.getClass()) return false;
+
+ FunctionalPattern<?, ?, ?> other = (FunctionalPattern<?, ?, ?>) obj;
+
+ return Objects.equals(accepter, other.accepter)
+ && Objects.equals(matcher, other.matcher);
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/patterns/FunctionalPatternMatcher.java b/base/src/main/java/bjc/utils/patterns/FunctionalPatternMatcher.java
new file mode 100644
index 0000000..e370fa0
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/FunctionalPatternMatcher.java
@@ -0,0 +1,31 @@
+package bjc.utils.patterns;
+
+import bjc.functypes.*;
+
+/**
+ * A simple pattern matcher backed by a function.
+ *
+ * @author Ben Culkin
+ *
+ * @param <ReturnType> The type returned by the matcher.
+ * @param <InputType> The type to match against.
+ */
+public class FunctionalPatternMatcher<ReturnType, InputType>
+ implements PatternMatcher<ReturnType, InputType> {
+
+ private final ThrowFunction<InputType, ReturnType, NonExhaustiveMatch> matcher;
+
+ /**
+ * Create a new function-backed pattern matcher.
+ *
+ * @param matcher The function backing this matcher.
+ */
+ public FunctionalPatternMatcher(ThrowFunction<InputType, ReturnType, NonExhaustiveMatch> matcher) {
+ this.matcher = matcher;
+ }
+
+ @Override
+ public ReturnType matchFor(InputType input) throws NonExhaustiveMatch {
+ return matcher.apply(input);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/patterns/MutablePatternMatcher.java b/base/src/main/java/bjc/utils/patterns/MutablePatternMatcher.java
new file mode 100644
index 0000000..176f588
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/MutablePatternMatcher.java
@@ -0,0 +1,86 @@
+package bjc.utils.patterns;
+
+import java.util.*;
+
+import bjc.data.*;
+
+/**
+ * A pattern matcher over a mutable set of patterns.
+ *
+ * Note that modifying a pattern matcher while it is currently doing pattern
+ * matching is a wonderful way to cause strange behavior.
+ *
+ * @author Ben Culkin
+ *
+ * @param <ReturnType> The type returned by the pattern matcher.
+ * @param <InputType> The type of the input to match against.
+ */
+public class MutablePatternMatcher<ReturnType, InputType>
+ implements PatternMatcher<ReturnType, InputType> {
+ private final List<ComplexPattern<ReturnType, Object, InputType>> patterns;
+
+ /**
+ * Create a new mutable pattern matcher with no patterns.
+ */
+ public MutablePatternMatcher() {
+ patterns = new ArrayList<>();
+ }
+
+ /**
+ * Create a new mutable pattern matcher with the given set of patterns.
+ *
+ * @param patterns The set of patterns to match on.
+ */
+ @SuppressWarnings("unchecked")
+ public MutablePatternMatcher(ComplexPattern<ReturnType, ?, InputType>... patterns) {
+ this();
+
+ for (ComplexPattern<ReturnType, ?, InputType> pattern : patterns) {
+ // Note: this may seem a somewhat questionable cast, but because we never
+ // actually do anything with the value who has a type matching the second
+ // parameter, this should be safe
+ this.patterns.add((ComplexPattern<ReturnType, Object, InputType>) pattern);
+ }
+ }
+
+ @Override
+ public ReturnType matchFor(InputType input) throws NonExhaustiveMatch {
+ Iterator<ComplexPattern<ReturnType, Object, InputType>> iterator;
+ iterator = new NonCMEIterator<>(patterns);
+ while(iterator.hasNext()) {
+ ComplexPattern<ReturnType, Object, InputType> pattern = iterator.next();
+
+ Pair<Boolean, Object> matches = pattern.matches(input);
+
+ matches.doWith((bool, obj) -> {
+ if (bool) pattern.apply(input, obj);
+ });
+ }
+
+ throw new NonExhaustiveMatch("Non-exhaustive match against " + input);
+ }
+
+ /**
+ * Add a pattern to this pattern matcher.
+ *
+ * @param pattern The pattern to add.
+ *
+ * @return Whether or not the pattern was added.
+ */
+ @SuppressWarnings("unchecked")
+ public boolean addPattern(ComplexPattern<ReturnType, ?, InputType> pattern) {
+ return patterns.add((ComplexPattern<ReturnType, Object, InputType>) pattern);
+ }
+
+ /**
+ * Remove a pattern from this pattern matcher.
+ *
+ * @param pattern The pattern to remove.
+ *
+ * @return Whether or not the pattern was removed.
+ */
+ @SuppressWarnings("unlikely-arg-type")
+ public boolean removePattern(ComplexPattern<ReturnType, ?, InputType> pattern) {
+ return patterns.remove(pattern);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/patterns/NonExhaustiveMatch.java b/base/src/main/java/bjc/utils/patterns/NonExhaustiveMatch.java
new file mode 100644
index 0000000..aaa7e89
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/NonExhaustiveMatch.java
@@ -0,0 +1,20 @@
+package bjc.utils.patterns;
+
+/**
+ * Exception thrown when a non-exhaustive match happens.
+ * @author Ben Culkin
+ *
+ */
+public class NonExhaustiveMatch extends Exception {
+ private static final long serialVersionUID = 3892904574888418544L;
+
+ /**
+ * Create a new non-exhaustive match.
+ *
+ * @param message The message for the exception.
+ */
+ public NonExhaustiveMatch(String message) {
+ super(message);
+ }
+
+}
diff --git a/base/src/main/java/bjc/utils/patterns/Pattern.java b/base/src/main/java/bjc/utils/patterns/Pattern.java
new file mode 100644
index 0000000..c9902e8
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/Pattern.java
@@ -0,0 +1,14 @@
+package bjc.utils.patterns;
+
+/**
+ * A simpler version of ComplexPattern, which always applies against Object
+ *
+ * @author Ben Culkin
+ *
+ * @param <ReturnType> The type returned by the pattern.
+ * @param <PredType> The state type returned by the predicate.
+ */
+public interface Pattern<ReturnType, PredType>
+ extends ComplexPattern<ReturnType, PredType, Object> {
+ /* Pattern factory methods */
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/patterns/PatternMatcher.java b/base/src/main/java/bjc/utils/patterns/PatternMatcher.java
new file mode 100644
index 0000000..40bf42b
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/PatternMatcher.java
@@ -0,0 +1,85 @@
+package bjc.utils.patterns;
+
+import java.util.function.*;
+
+import bjc.functypes.*;
+
+/**
+ * Represents a pattern matcher against a series of patterns.
+ *
+ * @author Ben Culkin
+ *
+ * @param <ReturnType> The type returned from matching the patterns.
+ * @param <InputType> The type to match against.
+ */
+@FunctionalInterface
+public interface PatternMatcher<ReturnType, InputType> {
+ /**
+ * Match an input object against a set of patterns.
+ *
+ * @param input The object to match against.
+ *
+ * @return The result of matching against the object.
+ *
+ * @throws NonExhaustiveMatch If none of the patterns in this set match
+ */
+ ReturnType matchFor(InputType input) throws NonExhaustiveMatch;
+
+ /**
+ * Create a pattern matcher against a static set of patterns.
+ *
+ * @param <RetType> The type returned from matching the patterns.
+ * @param <InpType> The type to match against.
+ *
+ * @param patterns The set of patterns to match on.
+ *
+ * @return A pattern matcher which matches on the given patterns.
+ */
+ @SafeVarargs
+ static <RetType, InpType> PatternMatcher<RetType, InpType> matchingOn(
+ ComplexPattern<RetType, ?, InpType>... patterns) {
+ return new SimplePatternMatcher<>(patterns);
+ }
+
+ /**
+ * Create a pattern matcher from a handler function.
+ *
+ * @param <RetType> The type returned by the matcher.
+ * @param <InpType> The type to match against.
+ *
+ * @param handler The handler function.
+ *
+ * @return A pattern matcher defined by the given handler.
+ */
+ static <RetType, InpType> PatternMatcher<RetType, InpType> from(
+ ThrowFunction<InpType, RetType, NonExhaustiveMatch> handler) {
+ return new FunctionalPatternMatcher<>(handler);
+ }
+
+ /**
+ * Create a pattern matcher which applies a transform to its input.
+ *
+ * @param <NewInput> The new input type to use.
+ * @param transformer The function to convert from the new input to the old input.
+ *
+ * @return A pattern matcher which takes values of the new type instead.
+ */
+ default <NewInput> PatternMatcher<ReturnType, NewInput> transformInput(
+ Function<NewInput, InputType> transformer) {
+ return from(inp -> matchFor(transformer.apply(inp)));
+ }
+
+ /**
+ * Create a pattern matcher which applies a transform to its output.
+ *
+ * @param <NewOutput> The new output type to use.
+ *
+ * @param transformer The function to convert from the new output to the old output.
+ *
+ * @return A pattern matcher which takes values of the new type instead.
+ */
+ default <NewOutput> PatternMatcher<NewOutput, InputType> transformOutput(
+ Function<ReturnType, NewOutput> transformer) {
+ return from(inp -> transformer.apply(matchFor(inp)));
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/patterns/SimplePatternMatcher.java b/base/src/main/java/bjc/utils/patterns/SimplePatternMatcher.java
new file mode 100644
index 0000000..9fae976
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/SimplePatternMatcher.java
@@ -0,0 +1,42 @@
+package bjc.utils.patterns;
+
+import bjc.data.*;
+
+/**
+ * Implements pattern-matching (of a sort) against a collection of patterns.
+ *
+ * @author Ben Culkin
+ *
+ * @param <InputType> The type input to the pattern matcher.
+ * @param <ReturnType> The type returned by the pattern.
+ */
+public class SimplePatternMatcher<ReturnType, InputType>
+ implements PatternMatcher<ReturnType, InputType> {
+ private final ComplexPattern<ReturnType, Object, InputType>[] patterns;
+
+ /**
+ * Create a new pattern matcher.
+ *
+ * @param patterns The set of patterns to match against.
+ */
+ @SuppressWarnings("unchecked")
+ @SafeVarargs
+ public SimplePatternMatcher(ComplexPattern<ReturnType, ?, InputType>...patterns) {
+ // Note: this may seem a somewhat questionable cast, but because we never
+ // actually do anything with the value who has a type matching the second
+ // parameter, this should be safe
+ this.patterns = (ComplexPattern<ReturnType, Object, InputType>[]) patterns;
+ }
+
+ @Override
+ public ReturnType matchFor(InputType input) throws NonExhaustiveMatch {
+ for (ComplexPattern<ReturnType, Object, InputType> pattern : patterns) {
+ Pair<Boolean, Object> matches = pattern.matches(input);
+ if (matches.getLeft()) {
+ pattern.apply(input, matches.getRight());
+ }
+ }
+
+ throw new NonExhaustiveMatch("Non-exhaustive match against " + input);
+ }
+}
diff --git a/base/src/main/java/bjc/utils/patterns/SimplePatttern.java b/base/src/main/java/bjc/utils/patterns/SimplePatttern.java
new file mode 100644
index 0000000..db53287
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/SimplePatttern.java
@@ -0,0 +1,40 @@
+package bjc.utils.patterns;
+
+import bjc.data.*;
+
+/**
+ * A simpler form of a pattern.
+ *
+ * @author Ben Culkin
+ *
+ * @param <ReturnType> The type returned by matching the pattern.
+ */
+public interface SimplePatttern<ReturnType> extends Pattern<ReturnType, Void> {
+ /**
+ * Test if this pattern does match a given object.
+ *
+ * @param input The object to test against.
+ *
+ * @return Whether the object matches this pattern.
+ */
+ boolean doesMatch(Object input);
+
+ /**
+ * Applies this pattern to the input object.
+ *
+ * @param input The object that passed the condition.
+ *
+ * @return The result of applying this action to the input.
+ */
+ ReturnType doApply(Object input);
+
+ @Override
+ default ReturnType apply(Object input, Void state) {
+ return doApply(input);
+ }
+
+ @Override
+ default Pair<Boolean, Void> matches(Object input) {
+ return new SimplePair<>(doesMatch(input), null);
+ }
+} \ No newline at end of file
diff --git a/base/src/main/java/bjc/utils/patterns/package-info.java b/base/src/main/java/bjc/utils/patterns/package-info.java
new file mode 100644
index 0000000..a562d95
--- /dev/null
+++ b/base/src/main/java/bjc/utils/patterns/package-info.java
@@ -0,0 +1,6 @@
+package bjc.utils.patterns;
+
+/*
+ * Pattern matching (of a sort) in Java. Based off of
+ * http://kerflyn.wordpress.com/2012/05/09/towards-pattern-matching-in-java/
+ */ \ No newline at end of file
diff --git a/base/src/test/java/bjc/utils/test/ioutils/LevelSplitterTest.java b/base/src/test/java/bjc/utils/test/ioutils/LevelSplitterTest.java
index 5fb3756..b0db059 100644
--- a/base/src/test/java/bjc/utils/test/ioutils/LevelSplitterTest.java
+++ b/base/src/test/java/bjc/utils/test/ioutils/LevelSplitterTest.java
@@ -3,6 +3,8 @@ package bjc.utils.test.ioutils;
import static bjc.utils.funcutils.TestUtils.assertListEquals;
import static bjc.utils.test.ioutils.LevelSplitterTest.RXPair.pair;
+import java.util.*;
+
import org.junit.Test;
import bjc.utils.ioutils.LevelSplitter;
@@ -28,21 +30,21 @@ public class LevelSplitterTest {
}
}
- /**
- * Test regex splitter.
- */
+ /** Test regex splitter. */
@Test
public void testRXSplit() {
// LevelSplitter splitter = LevelSplitter.def;
// Check generic splitting works
- assertRXSplit("\\s+", pair("", ""), pair("a", "a"), pair("a b", "a", "b"),
+ assertRXSplit("\\s+",
+ pair("", ""), pair("a", "a"), pair("a b", "a", "b"),
pair("a b", "a", "b"), pair("a\t \tb", "a", "b"));
}
private static void assertRXSplit(String pat, RXPair... pairs) {
for (RXPair pair : pairs) {
- assertListEquals(LevelSplitter.def.levelSplitRX(pair.inp, pat), pair.outp);
+ List<String> res = LevelSplitter.def.levelSplitRX(pair.inp, pat);
+ assertListEquals(res, pair.outp);
}
}
}
diff --git a/clformat/TODO b/clformat/TODO
new file mode 100644
index 0000000..abe3a37
--- /dev/null
+++ b/clformat/TODO
@@ -0,0 +1,20 @@
+@TODO Nov 13th, 2020 Ben Culkin :InvokeDirective
+This is more a 'mad-science' kind of thing, but creating a directive that allows
+people to invoke invokable items passed as format arguments could be interesting.
+ The key-name for this directive would probably be `I, and there are a number
+ of invokable types that could be interesting
+ * Supplier<String> to just insert the resulting string.
+ * Function<Object, String> to convert a format item to a string.
+
+@TODO Nov 13th, 2020 Ben Culkin :ConfigDirectives
+Add a way to configure user-based directives from reading a config file, which
+allows you to bind directives to various characters and customize their behavior.
+One thing that will need to be involved is that all of the directives should
+probably have their interfaces tweaked to allow you to provide a custom form of
+them. One could argue this is the sort of thing that macros could do, but I can
+see no compelling reason not to implement both features
+
+@TODO Nov 14th, 2020 Ben Culkin :OnceDirective
+Implement a directive (`O probably) that will only be executed once per given
+invocation of 'format', or maybe one for 'once per enclosing directive'. Could
+be useful, especially once we have more stateful directives. \ No newline at end of file
diff --git a/clformat/data/clformat.sprop b/clformat/data/clformat.sprop
index ba1e83c..1d31de6 100644
--- a/clformat/data/clformat.sprop
+++ b/clformat/data/clformat.sprop
@@ -27,9 +27,9 @@ clFormatPrefixParam (?:(?:[-+]?\d+|'.|[Vv]|#|%|".*?(?<!\\)")?)
##
## Parameters are seperated by ','
##
-## Named parameters are indicated by a preceeding #, which is followed by a
-## parameter name (any sequence of non-space, non-seperator (',', ':', ';')
-## printing characters), a seperator (: or ;), then the value of the parameter
+## Named parameters are indicated by a preceding #, which is followed by a
+## parameter name (any sequence of non-space, non-separator (',', ':', ';')
+## printing characters), a separator (: or ;), then the value of the parameter
clFormatPrefix (?:(?:#[\S&&[^,:;]]+[:;]%1$s)|%1$s)
## Match a format string modifier
@@ -42,9 +42,10 @@ clFormatModifier (?:[@$:*]+)
## Matches a directive name.
## A directive name is either
-## 1) A single, non-whitespace, non-/ character, optionally preceded by a grave
+## 1) A single, non-whitespace, non-/ character, optionally preceded by any
+## number of grave characters.
## 2) A name enclosed in /'s
-clFormatName (?:(?<name>`?[\S&&[^/]])|(?:/(?<funcname>[\S&&[^/]]+)/))
+clFormatName (?:(?<name>`*[\S&&[^/]])|(?:/(?<funcname>[\S&&[^/]]+)/))
#################################################
# Generic format strings for regular expressions.
diff --git a/clformat/docs/Makefile b/clformat/docs/Makefile
new file mode 100644
index 0000000..c2ad7f4
--- /dev/null
+++ b/clformat/docs/Makefile
@@ -0,0 +1,22 @@
+.PHONY: all clean
+
+all: clformat.pdf
+
+clean:
+ rm clformat.pdf
+
+clformat.pdf: clformat.ps
+ ps2pdf clformat.ps clformat.pdf
+ rm clformat.ps
+
+clformat.ps: clformat.ms
+ groff -ms -s -t -Tps clformat.ms > clformat.ps
+
+clformat.ms: clformat.tmac
+ touch clformat.ms
+
+clformat.tmac:
+ touch clformat.tmac
+
+preview: clformat.pdf
+ fbpdf clformat.pdf
diff --git a/clformat/docs/clformat.ms b/clformat/docs/clformat.ms
new file mode 100644
index 0000000..1a2731f
--- /dev/null
+++ b/clformat/docs/clformat.ms
@@ -0,0 +1,231 @@
+\# CLFormat documentation
+\# Load macros and stuff
+.so clformat.tmac
+.TL
+CLFormat Documentation
+.AU
+Ben Culkin
+\# No abstract needed
+.AB no
+.AE
+.PP
+CLFormat is an implementation of something that is strongly reminiscent of the
+Common Lisp FORMAT function. I say 'strongly reminiscent', because we don't
+support quite all the directives they, but support several new ones they don't.
+.Section "Directive Syntax"
+FORMAT strings are made up of free-form text, with all of the functionality
+coming from \fIdirectives\fP. A directive is always started with a tilde, and
+consists of the following parts:
+.Numlist
+.Numitem
+.Italic "An optional set of " "prefix parameters" "."
+.Numitem
+.Italic "An optional set of " "directive modifiers" "."
+.Numitem
+The name of the directive
+.PP
+To execute a format string, and get it to provide formatting, a set of arguments
+called \fIformat items\fP must be provided. These serve the same purpose as the
+arguments provided to printf and other similar functions after their format
+string.
+.Subsection "Prefix Parameters"
+Prefix parameters are used to configure the options for a directive. These can
+include things varying from the number of columns to print a field in, to the
+radix that a number should be printed in. They are separated from each by
+commas.
+.PP
+A prefix parameter can be any one of the following:
+.Defn "A signed decimal number"
+The value of this parameter is the provided number.
+.Defn "A single character, preceeded by a single-quote"
+The value of this parameter is the provided character
+.Defn "The letter V"
+The value of this parameter will be read from the format string items.
+.Defn "The character #"
+The value of this parameter will be set to the number of format string
+parameters remaining
+.Defn "The character %"
+The value of this paramter will be set to the current position in the list
+of format string items.
+.Defn "A double quoted string"
+The value of this parameter is the represented string.
+.PP
+Prefix parameters are numbered positionally by default, but they can be given a
+name by starting the parameter with a # character, following that with a
+parameter name (which can be any series of printing characters that is not a
+whitespace character, and not one of the following separator characters: ',',
+',', ':' or ';'). It is then followed by a name-separator character (either ':'
+or ';') then its value from the above list.
+.Supersection "Notes"
+.Bullet
+The difference between ':' and ';' is that if you use ':', that
+parameter can't be addressed positionally, while if you use ';', it still can.
+.Bullet
+If you specify more parameters than a directive is noted to take, the behavior
+is based off of whether you've provided extra named, non-positional parameters or extra
+positional ones. You can specify as many named non-positional parameters as you
+want, and the directive will be fine. However, if you specify more or less
+positional parameters, that is an an error.
+.Bullet
+As a 'useful' feature, when you are naming named parameters, you may abbreviate
+the name to its shortest unambigous value. However, as an exception to the
+normal rule that you can provide unused named parameters, passing an ambiguous
+parameter name will cause an error. For instance, if a directive took the
+parameters 'foo', 'foobar' and 'fizz', these are some of the results:
+.RS
+.CWDefn "foo"
+ Refers to foo
+.CWDefn "foobar"
+ Refers to foobar
+.CWDefn "foob"
+ Refers to foobar
+.CWDefn "fizz"
+ Refers to fizz
+.CWDefn "fi"
+ Refers to fizz
+.CWDefn "f"
+ Error, could refer to foo, foobar or fizz
+.RE
+.Endsupersection
+.Endsubsection
+.Subsection "Directive modifiers"
+Directive modifiers, or \fImodifier sets\fP, are any combination of the
+following characters (with duplicate characters having no effect)
+.Bullet
+.CW "$"
+.Bullet
+.CW ":"
+.Bullet
+.CW "*"
+.Bullet
+.CW "@"
+.Endsubsection
+.Subsection "Directive names"
+A directive name can be either
+.Bullet
+A single, non-whitespace, non-/ character, optionally preceeded by a grave
+.Bullet
+A name bracketed by /
+.PP
+If the directive is a bracketed name, then it is a call to a user-specified
+function. Otherwise, it is an invocation of one of the built-in directives.
+.Endsubsection
+.Endsection
+.Section "Directive List"
+The following is a table of all of the directives that are currently implemented,
+as well as a short description of what each directive does.
+.KS
+.TS H
+center allbox ;
+c s s
+c | c | c .
+Directive Overview
+Directive Name Brief Description
+_
+.TH
+.T&
+lfCR | l | l .
+A Aesthetic General string printer.
+S Alias for the A directive\[dg]
+C Character Print out a single character
+B Binary Print out a base-2 integer
+O Octal Print out a base-8 integer
+D Decimal Print out a base-10 integer
+X Hexadecimal Print out a base-16 integer
+R Radix Print out an integer in an arbitrary base\[bu]
+& Fresh Line Print a newline character, if we didn't just print one
+% Newline Print out a literal new line
+| Formfeed Print out a 'formfeed' and start a new page
+~ Tilde Print out a literal tilde
+? Recursive Invoke a sub-frmat string
+* Go-to Move around in the item list
+^ Escape Escape from an enclosing directive
+[ Conditional Select a sub-format string, based on a conditional
+{ Iteration Repeatedly invoke a sub-format string
+( Case Perform case-manipulation on text
+\`[ Inflection Perform inflection (pluralization/singularization) on a sub-format string
+T Tabulate Print items in a tabular format
+.TE
+.ce 1
+\fITable 1: Directive Overview\fP
+.KE
+.FS \[dg]
+CommonLisp has S use a slightly different format, but there isn't one that makes
+sense for us to use. In the future, we may either come up with a format, or
+reassign S to do something else entirely
+.FE
+.FS \[bu]
+At some point, it may make sense to remove some of the radix-specific ones, and
+just tell people to use R. However, for now, we're sticking with it.
+.FE
+.PP
+The following are directives that aren't valid for use outside of a specific
+context (usually, they must follow another directive
+.KS
+.TS H
+center allbox ;
+c s s
+c | c | c .
+Context-Sensitive Directives
+Directive Name Brief Description
+_
+.TH
+.T&
+lfCR | l | l .
+] End Conditional Ends a conditional directive
+; Clause Separator Separates two clauses in a block directive
+) End Case Ends a case directive
+\'] End Inflection Ends a inflection directive
+< Inflection Start Starts an inflection control
+> Inflection End Ends an inflection control
+.TE
+.ce 1
+\fITable 2: Context-Sensitive Directives\fP
+.KE
+.PP
+And the following are directives which have not yet been implemented
+.KS
+.TS H
+center allbox ;
+c s s
+c | c | c .
+Unimplemented Directives
+Directive Name Brief Description
+_
+.TH
+.T&
+lfCR | l | l .
+\`< Start Layout Start a layout-control block
+\`> End Layout Ends a layout-control block
+F Fixed-Format Float Print a floating-point number in a fixed-point format
+E Exponential Float Print a engineering-style (1.2e2) floating point number
+G General Float Print a general floating point number
+$ Monetary Float Print a floating point number like currency
+W String Print Use the A directive instead
+P Plural Use the `[ and `] directives instead
+.TE
+.ce 1
+\fITable 3: Unimplemented Directives\fP
+.KE
+.PP
+The following section contains a detailed reference for the directives mentioned
+above. This details how the directive works, how many parameters it takes, and
+of what sort they should be, and whatever other details are necessary for that
+particular directive.
+.Subsection "A Directive"
+.PP
+The A directive is the general purpose string formatting directive, serving as
+the equivalent to the S specifier in printf and other similar formats.
+
+It takes either zero, one or four parameters, as well as one format string item.
+.KS
+.TS
+allbox;
+c c
+lfCR l .
+Sample Format String Description
+~a Prints out the format string item as a string
+.TE
+.KE
+.Endsubsection
+.Endsection
diff --git a/clformat/docs/clformat.tmac b/clformat/docs/clformat.tmac
new file mode 100644
index 0000000..0d7da96
--- /dev/null
+++ b/clformat/docs/clformat.tmac
@@ -0,0 +1,63 @@
+\# Useful macros and settings
+\# Section header
+.de Section
+.NH 1
+\s+4\\$1\s-4
+\# Place current section title in right corner
+.ds RH "\\$1
+.PP
+..
+\# Section footer
+.de Endsection
+..
+\# Subsection header
+.de Subsection
+.NH 2
+\s+2\\$1\s-2
+.RS
+.PP
+..
+\# Subsection footer
+.de Endsubsection
+.RE
+..
+\# Supersection header
+.de Supersection
+.NH 3
+\\$1
+.RS
+.PP
+..
+\# Supersection footer
+.de Endsupersection
+.RE
+..
+\# Bulleted list item
+.de Bullet
+.IP \[bu]
+..
+\# Simple numbered list
+.de Numlist
+.nr listitem 0 1
+..
+\# Item in numbered list
+.de Numitem
+.IP "\\n+[listitem]."
+..
+\# Italicize text
+.de Italic
+.I "\\$2" "\\$3" "\\$1"
+..
+\# Definition-list item
+.de Defn
+.IP "\fB\\$1\fP"
+..
+\# Constant-width Definition-list item
+.de CWDefn
+.IP "\f(CB\\$1\fP"
+..
+\# Set in 12pt font
+.nr PS 12p
+\# Put manual title in left corner
+.ds LH "CLFormat Manual
+.DA
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLFormatter.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLFormatter.java
index 9263beb..276356c 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLFormatter.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLFormatter.java
@@ -8,7 +8,6 @@ import bjc.utils.ioutils.ReportWriter;
import bjc.utils.ioutils.format.directives.*;
// Grab our easy converters/constructors
-import static bjc.utils.funcutils.IteratorUtils.AI;
import static bjc.utils.funcutils.IteratorUtils.I;
/**
@@ -70,6 +69,8 @@ public class CLFormatter {
builtinDirectives.put("`[", new InflectDirective());
builtinDirectives.put("T", new TabulateDirective());
+
+ builtinDirectives.put("`D", new DecimalDirective());
}
/**
@@ -101,14 +102,32 @@ public class CLFormatter {
*/
public static void checkItem(Object itm, char directive) {
if (itm == null) {
- String msg
- = String.format("No argument provided for %c directive", directive);
+ String msg = String.format("No argument provided for %c directive", directive);
throw new IllegalArgumentException(msg);
}
}
/**
+ * Check that an item is valid for a directive.
+ *
+ * @param itm
+ * The item to check.
+ *
+ * @param directive
+ * The directive to check for.
+ *
+ * @throws IllegalArgumentException
+ * if itm is null.
+ */
+ public static void checkItem(Object itm, String directive) {
+ if (itm == null) {
+ String msg = String.format("No argument provided for %s directive", directive);
+
+ throw new IllegalArgumentException(msg);
+ }
+ }
+ /**
* Format a string in the style of CL's FORMAT.
*
* @param format
@@ -123,7 +142,7 @@ public class CLFormatter {
* if something goes wrong during formatting the string.
*/
public String formatString(String format, Object... params) throws IOException {
- return formatString(format, I(AI(params)));
+ return formatString(format, I(I(params)));
}
/**
@@ -280,103 +299,91 @@ public class CLFormatter {
boolean isToplevel) throws IOException {
try {
while (cltok.hasNext()) {
- Decree decr = cltok.next();
+ SimpleDecree decr = cltok.next();
if (decr.isLiteral) {
rw.write(decr.name);
- continue;
- }
-
- Object item = tParams.item();
-
- if (decr.isUserCall) {
+ } else if (decr.isUserCall) {
/*
* @TODO implement user-called functions.
*/
- continue;
- }
-
- if (extraDirectives.containsKey(decr.name)) {
+ } else if (extraDirectives.containsKey(decr.name)) {
FormatParameters params
- = new FormatParameters(rw, item, decr, tParams, cltok, this);
+ = new FormatParameters(rw, tParams.item(), decr,
+ tParams, cltok, this);
extraDirectives.get(decr.name).format(params);
-
- continue;
- }
-
- if (builtinDirectives.containsKey(decr.name)) {
+ } else if (builtinDirectives.containsKey(decr.name)) {
FormatParameters params
- = new FormatParameters(rw, item, decr, tParams, cltok, this);
+ = new FormatParameters(rw, tParams.item(), decr,
+ tParams, cltok, this);
builtinDirectives.get(decr.name).format(params);
- continue;
- }
-
- if (decr.name == null)
- decr.name = "<null>";
-
- switch (decr.name) {
- case "]":
- throw new IllegalArgumentException(
- "Found conditional-end outside of conditional.");
- case ";":
- throw new IllegalArgumentException(
- "Found seperator outside of block.");
- case "}":
- throw new IllegalArgumentException(
- "Found iteration-end outside of iteration");
- case ")":
- throw new IllegalArgumentException(
- "Case-conversion end outside of case conversion");
- case "`]":
- throw new IllegalArgumentException(
- "Inflection-end outside of inflection");
- case "<":
- case ">":
- throw new IllegalArgumentException(
- "Inflection marker outside of inflection");
- case "`<":
- case "`>":
- throw new IllegalArgumentException(
- "Layout-control directives aren't implemented yet.");
- case "F":
- case "E":
- case "G":
- case "$":
- /*
- * @TODO
- *
- * implement floating point directives.
- */
- throw new IllegalArgumentException(
- "Floating-point directives aren't implemented yet.");
- case "W":
- /*
- * @TODO
- *
- * figure out if we want to implement someting for these directives
- * instead of punting.
- */
- throw new IllegalArgumentException(
- "S and W aren't implemented. Use A instead");
- case "P":
- throw new IllegalArgumentException(
- "These directives aren't implemented yet");
- case "\n":
- /*
- * Ignored newline.
- */
- break;
- default:
- String msg
- = String.format("Unknown format directive '%s'", decr.name);
- throw new IllegalArgumentException(msg);
+ } else {
+ // All of these conditions are an error in some way
+ if (decr.name == null) decr.name = "<null>";
+
+ switch (decr.name) {
+ case "]":
+ throw new IllegalArgumentException(
+ "Found conditional-end outside of conditional.");
+ case ";":
+ throw new IllegalArgumentException(
+ "Found seperator outside of block.");
+ case "}":
+ throw new IllegalArgumentException(
+ "Found iteration-end outside of iteration");
+ case ")":
+ throw new IllegalArgumentException(
+ "Case-conversion end outside of case conversion");
+ case "`]":
+ throw new IllegalArgumentException(
+ "Inflection-end outside of inflection");
+ case "<":
+ case ">":
+ throw new IllegalArgumentException(
+ "Inflection marker outside of inflection");
+ case "`<":
+ case "`>":
+ throw new IllegalArgumentException(
+ "Layout-control directives aren't implemented yet.");
+ case "F":
+ case "E":
+ case "G":
+ case "$":
+ /*
+ * @TODO
+ *
+ * implement floating point directives.
+ */
+ throw new IllegalArgumentException(
+ "For now, floating point directives are implemented via the `D directive. Use that instead");
+ case "W":
+ /*
+ * @TODO
+ *
+ * figure out if we want to implement someting for these directives
+ * instead of punting.
+ */
+ throw new IllegalArgumentException(
+ "S and W aren't implemented. Use A instead");
+ case "P":
+ throw new IllegalArgumentException(
+ "These directives aren't implemented yet");
+ case "\n":
+ /*
+ * Ignored newline.
+ */
+ break;
+ default:
+ String msg
+ = String.format("Unknown format directive '%s'", decr.name);
+ throw new IllegalArgumentException(msg);
+ }
}
}
} catch (DirectiveEscape eex) {
- if (!isToplevel)
- throw eex;
+ if (!isToplevel) throw eex;
}
}
@@ -404,10 +411,9 @@ public class CLFormatter {
*
* @return A set of edicts compiled from the decrees.
*/
- public List<Edict> compile(Iterable<Decree> decrees) {
+ public List<Edict> compile(Iterable<SimpleDecree> decrees) {
// If we have no decrees, there are no edicts.
- if (decrees == null)
- return new ArrayList<>();
+ if (decrees == null) return new ArrayList<>();
CLTokenizer it = CLTokenizer.fromTokens(decrees);
return compile(it);
@@ -422,10 +428,8 @@ public class CLFormatter {
* @return The set of edicts compiled from the clause.
*/
public List<Edict> compile(ClauseDecree clause) {
- if (clause == null)
- return new ArrayList<>();
-
- return compile(clause.body);
+ if (clause == null) return new ArrayList<>();
+ else return compile(clause.body);
}
/**
@@ -440,96 +444,86 @@ public class CLFormatter {
List<Edict> result = new ArrayList<>();
while (cltok.hasNext()) {
- Decree decr = cltok.next();
+ SimpleDecree decr = cltok.next();
String nam = decr.name;
CompileContext compCTX = new CompileContext(cltok, this, decr);
if (decr.isLiteral) {
result.add(new StringEdict(decr.name));
-
- continue;
- }
-
- if (decr.isUserCall) {
+ } else if (decr.isUserCall) {
/*
* @TODO implement user-called functions.
*/
throw new IllegalArgumentException(
"User-called functions have not yet been implemented");
- }
-
- if (extraDirectives.containsKey(nam)) {
+ } else if (extraDirectives.containsKey(nam)) {
Edict edt = extraDirectives.get(nam).compile(compCTX);
result.add(edt);
-
- continue;
} else if (builtinDirectives.containsKey(nam)) {
Edict edt = builtinDirectives.get(nam).compile(compCTX);
result.add(edt);
-
- continue;
- }
-
- if (nam == null)
- nam = "<null>";
-
- switch (nam) {
- case "]":
- throw new IllegalArgumentException(
- "Found conditional-end outside of conditional.");
- case ";":
- throw new IllegalArgumentException("Found seperator outside of block.");
- case "}":
- throw new IllegalArgumentException(
- "Found iteration-end outside of iteration");
- case ")":
- throw new IllegalArgumentException(
- "Case-conversion end outside of case conversion");
- case "`]":
- throw new IllegalArgumentException(
- "Inflection-end outside of inflection");
- case "<":
- case ">":
- throw new IllegalArgumentException(
- "Inflection marker outside of inflection");
- case "`<":
- case "`>":
- throw new IllegalArgumentException(
- "Layout-control directives aren't implemented yet.");
- case "F":
- case "E":
- case "G":
- case "$":
- /*
- * @TODO
- *
- * implement floating point directives.
- */
- throw new IllegalArgumentException(
- "Floating-point directives aren't implemented yet.");
- case "W":
- /*
- * @TODO
- *
- * figure out if we want to implement someting for these directives
- * instead of punting.
- */
- throw new IllegalArgumentException(
- "S and W aren't implemented. Use A instead");
- case "P":
- throw new IllegalArgumentException(
- "These directives aren't implemented yet");
- case "\n":
- /*
- * Ignored newline.
- */
- break;
- default:
- String msg = String.format("Unknown format directive '%s'", nam);
- throw new IllegalArgumentException(msg);
+ } else {
+ // All of these conditions are an error in some way
+ if (nam == null) nam = "<null>";
+
+ switch (nam) {
+ case "]":
+ throw new IllegalArgumentException(
+ "Found conditional-end outside of conditional.");
+ case ";":
+ throw new IllegalArgumentException("Found seperator outside of block.");
+ case "}":
+ throw new IllegalArgumentException(
+ "Found iteration-end outside of iteration");
+ case ")":
+ throw new IllegalArgumentException(
+ "Case-conversion end outside of case conversion");
+ case "`]":
+ throw new IllegalArgumentException(
+ "Inflection-end outside of inflection");
+ case "<":
+ case ">":
+ throw new IllegalArgumentException(
+ "Inflection marker outside of inflection");
+ case "`<":
+ case "`>":
+ throw new IllegalArgumentException(
+ "Layout-control directives aren't implemented yet.");
+ case "F":
+ case "E":
+ case "G":
+ case "$":
+ /*
+ * @TODO
+ *
+ * implement floating point directives.
+ */
+ throw new IllegalArgumentException(
+ "Floating-point directives aren't implemented yet.");
+ case "W":
+ /*
+ * @TODO
+ *
+ * figure out if we want to implement someting for these directives
+ * instead of punting.
+ */
+ throw new IllegalArgumentException(
+ "S and W aren't implemented. Use A instead");
+ case "P":
+ throw new IllegalArgumentException(
+ "These directives aren't implemented yet");
+ case "\n":
+ /*
+ * Ignored newline.
+ */
+ break;
+ default:
+ String msg = String.format("Unknown format directive '%s'", nam);
+ throw new IllegalArgumentException(msg);
+ }
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLModifiers.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLModifiers.java
index e2cb8b4..c0bd75d 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLModifiers.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLModifiers.java
@@ -60,10 +60,10 @@ public class CLModifiers {
boolean starMod = false;
if (modString != null) {
- atMod = modString.contains("@");
- colonMod = modString.contains(":");
+ atMod = modString.contains("@");
+ colonMod = modString.contains(":");
dollarMod = modString.contains("$");
- starMod = modString.contains("*");
+ starMod = modString.contains("*");
}
return new CLModifiers(atMod, colonMod, dollarMod, starMod);
@@ -73,14 +73,10 @@ public class CLModifiers {
public String toString() {
StringBuilder sb = new StringBuilder();
- if (atMod)
- sb.append('@');
- if (colonMod)
- sb.append(':');
- if (dollarMod)
- sb.append('$');
- if (starMod)
- sb.append('*');
+ if (atMod) sb.append('@');
+ if (colonMod) sb.append(':');
+ if (dollarMod) sb.append('$');
+ if (starMod) sb.append('*');
return sb.toString();
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLParameters.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLParameters.java
index aef8fbe..61d88bf 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLParameters.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLParameters.java
@@ -5,6 +5,10 @@ import java.util.*;
import bjc.esodata.*;
import bjc.utils.parserutils.TokenUtils;
+// @TODO Nov 13th, 2020 Ben Culkin :ParameterDefaulting
+// Implement a method which will 'overlay' a set of parameters onto another
+// paramater set, with support for preferring one or the other.
+
/**
* Represents a set of parameters to a CL format directive.
*
@@ -13,13 +17,13 @@ import bjc.utils.parserutils.TokenUtils;
public class CLParameters {
private static String MSG_FMT = "Invalid %s (%s) \"%s\" to %s directive";
- private static String RX_TRUE = "(?i)y(?:es)?|t(?:rue)?(?i)";
+ private static String RX_TRUE = "(?i)y(?:es)?|t(?:rue)?(?i)";
private static String RX_FALSE = "(?i)no?|f(?:alse)?(?i)";
private CLValue[] params;
private Set<String> abbrevWords;
- private AbbrevMap2 nameAbbrevs;
+ private AbbrevMap2 nameAbbrevs;
private Map<String, CLValue> namedParams;
private Map<String, Integer> nameIndices;
@@ -80,19 +84,13 @@ public class CLParameters {
// with here, as these objects are fairly temporary.
//
// If it becomes an issue, I'll resolve it
- for (String key : namedParams.keySet()) {
- refreshAbbrev(key);
- }
-
- for (String key : nameIndices.keySet()) {
- refreshAbbrev(key);
- }
+ for (String key : namedParams.keySet()) refreshAbbrev(key);
+ for (String key : nameIndices.keySet()) refreshAbbrev(key);
}
// Refresh a particular abbreviation
private void refreshAbbrev(String key) {
- if (abbrevWords.contains(key))
- return;
+ if (abbrevWords.contains(key)) return;
abbrevWords.add(key);
nameAbbrevs.add(key);
@@ -110,8 +108,7 @@ public class CLParameters {
for (int i = 0; i < opts.length; i++) {
String opt = opts[i];
- if (!opt.equals(""))
- mapIndex(opt, i);
+ if (!opt.equals("")) mapIndex(opt, i);
}
refreshAbbrevs();
@@ -139,8 +136,7 @@ public class CLParameters {
nameIndices.put(opt.toUpperCase(), idx);
- if (doRefresh)
- refreshAbbrevs();
+ if (doRefresh) refreshAbbrevs();
}
/**
@@ -152,8 +148,7 @@ public class CLParameters {
* @return The value at that index.
*/
public CLValue getByIndex(int idx) {
- if (idx < 0 || idx >= params.length)
- return null;
+ if (idx < 0 || idx >= params.length) return null;
return params[idx];
}
@@ -235,8 +230,7 @@ public class CLParameters {
if (ch == ':' || ch == ';') {
// Semicolon says to add as
// indexed parameter
- if (ch == ';')
- setIndex = true;
+ if (ch == ';') setIndex = true;
nameIdx = i;
break;
@@ -251,8 +245,7 @@ public class CLParameters {
namedParams.put(paramName.toUpperCase(), actVal);
- if (setIndex)
- parameters.add(actVal);
+ if (setIndex) parameters.add(actVal);
} else {
parameters.add(parseParam(param));
}
@@ -311,10 +304,11 @@ public class CLParameters {
sb.append("\". Could've meant: ");
boolean isFirst = true;
for (String possKey : keys) {
- if (!isFirst)
- sb.append(", ");
- if (isFirst)
+ if (isFirst) {
isFirst = false;
+ } else {
+ sb.append(", ");
+ }
sb.append("\"");
sb.append(possKey);
@@ -335,10 +329,8 @@ public class CLParameters {
// @NOTE 9/22/18
//
// Consider whether we should throw an exception here.
- if (idx < 0 || idx >= params.length)
- return null;
-
- return params[idx];
+ if (idx < 0 || idx >= params.length) return null;
+ else return params[idx];
}
return null;
@@ -370,10 +362,8 @@ public class CLParameters {
String bol = resolveKey(key).getValue(parms);
if (!bol.equals("")) {
- if (bol.matches(RX_TRUE))
- return true;
- else if (bol.matches(RX_FALSE))
- return false;
+ if (bol.matches(RX_TRUE)) return true;
+ else if (bol.matches(RX_FALSE)) return false;
else {
String msg = String.format(MSG_FMT, paramName, key, bol, directive);
throw new IllegalArgumentException(msg);
@@ -412,8 +402,7 @@ public class CLParameters {
//
// This raises the question of what to do if the empty string is a valid
// value for a parameter
- if (!vl.equals(""))
- return vl;
+ if (!vl.equals("")) return vl;
return def;
}
@@ -508,25 +497,23 @@ public class CLParameters {
int idx = 0;
// First off, the named parameters
for (Map.Entry<String, CLValue> param : namedParams.entrySet()) {
- String paramName = param.getKey();
+ String paramName = param.getKey();
CLValue paramValue = param.getValue();
if (nameIndices.containsKey(paramName)) {
int paramIdx = nameIndices.get(paramName);
- String msg
- = String.format("%s(%d):'%s'", paramName, paramIdx, paramValue);
+ String msg = String.format("%s(%d):'%s'",
+ paramName, paramIdx, paramValue);
- if (idx != 0)
- sb.append(", ");
+ if (idx != 0) sb.append(", ");
sb.append(msg);
seenIndices.add(idx);
} else {
String msg = String.format("%s:'%s'", paramName, paramValue);
- if (idx != 0)
- sb.append(", ");
+ if (idx != 0) sb.append(", ");
sb.append(msg);
}
@@ -541,14 +528,12 @@ public class CLParameters {
int paramIdx = paramMap.getValue();
// We've already gotten this argument before
- if (seenIndices.contains(paramIdx))
- continue;
+ if (seenIndices.contains(paramIdx)) continue;
- String msg
- = String.format("%d(%s):'%s'", paramIdx, paramName, params[paramIdx]);
+ String msg = String.format("%d(%s):'%s'",
+ paramIdx, paramName, params[paramIdx]);
- if (idx != 0)
- sb.append(", ");
+ if (idx != 0) sb.append(", ");
sb.append(msg);
seenIndices.add(paramIdx);
@@ -559,13 +544,11 @@ public class CLParameters {
// Third, unnamed indexed parameters
for (idx = 0; idx < params.length; idx++) {
// We've already gotten this argument before
- if (seenIndices.contains(idx))
- continue;
+ if (seenIndices.contains(idx)) continue;
String msg = String.format("%d:'%s'", idx, params[idx]);
- if (idx != 0)
- sb.append(", ");
+ if (idx != 0) sb.append(", ");
sb.append(msg);
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLString.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLString.java
index 01e7617..6d77503 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLString.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLString.java
@@ -98,9 +98,7 @@ public class CLString {
*/
public String format(FormatContext formCTX) throws IOException {
try {
- for (Edict edt : edicts) {
- edt.format(formCTX);
- }
+ for (Edict edt : edicts) edt.format(formCTX);
} catch (DirectiveEscape eex) {
// General escape exception, so stop formatting.
}
@@ -114,10 +112,8 @@ public class CLString {
* @return If this format string is empty.
*/
public boolean isEmpty() {
- if (edicts.size() == 0)
- return true;
-
- return false;
+ if (edicts.size() == 0) return true;
+ else return false;
}
@Override
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLTokenizer.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLTokenizer.java
index af7c935..7ed76d2 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLTokenizer.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLTokenizer.java
@@ -9,7 +9,7 @@ import java.util.regex.*;
* @author bjculkin
*
*/
-public class CLTokenizer implements Iterator<Decree> {
+public class CLTokenizer implements Iterator<SimpleDecree> {
/**
* Whether or not the tokenizer is in debug mode or not.
*/
@@ -19,34 +19,30 @@ public class CLTokenizer implements Iterator<Decree> {
* Internal class for a tokenizer that returns a specific set of tokens.
*/
private static class SetCLTokenizer extends CLTokenizer {
- private Iterator<Decree> body;
+ private Iterator<SimpleDecree> body;
- public SetCLTokenizer(Iterator<Decree> bod) {
+ public SetCLTokenizer(Iterator<SimpleDecree> bod) {
body = bod;
}
- public SetCLTokenizer(Iterable<Decree> bod) {
+ public SetCLTokenizer(Iterable<SimpleDecree> bod) {
body = bod.iterator();
}
@Override
public boolean hasNext() {
- boolean nxt = body.hasNext();
-
- return nxt;
+ return body.hasNext();
}
@Override
- public Decree next() {
- Decree nxt = body.next();
-
- return nxt;
+ public SimpleDecree next() {
+ return body.next();
}
}
private Matcher mat;
- private Decree dir;
+ private SimpleDecree dir;
/**
* Empty constructor that should only be invoked if you are a subclass who
@@ -74,7 +70,7 @@ public class CLTokenizer implements Iterator<Decree> {
*
* @return A tokenizer yielding the given set of decrees.
*/
- public static CLTokenizer fromTokens(Iterator<Decree> bod) {
+ public static CLTokenizer fromTokens(Iterator<SimpleDecree> bod) {
return new SetCLTokenizer(bod);
}
@@ -86,30 +82,25 @@ public class CLTokenizer implements Iterator<Decree> {
*
* @return A tokenizer yielding the given set of decrees.
*/
- public static CLTokenizer fromTokens(Iterable<Decree> bod) {
+ public static CLTokenizer fromTokens(Iterable<SimpleDecree> bod) {
return new SetCLTokenizer(bod);
}
@Override
public boolean hasNext() {
- boolean nxt = !mat.hitEnd();
-
- return nxt;
+ return !mat.hitEnd();
}
@Override
- public Decree next() {
- Decree tk = getNext();
-
- return tk;
+ public SimpleDecree next() {
+ return getNext();
}
- private Decree getNext() {
- if (!hasNext())
- throw new NoSuchElementException("No possible decrees remaining");
+ private SimpleDecree getNext() {
+ if (!hasNext()) throw new NoSuchElementException("No possible decrees remaining");
if (dir != null) {
- Decree tmp = dir;
+ SimpleDecree tmp = dir;
dir = null;
@@ -124,39 +115,35 @@ public class CLTokenizer implements Iterator<Decree> {
String tmp = sb.toString();
{
- String dirName = mat.group("name");
- String dirFunc = mat.group("funcname");
- String dirMods = mat.group("modifiers");
- String dirParams = mat.group("params");
+ String directiveName = mat.group("name");
+ String directiveFunction = mat.group("funcname");
+ String directiveModifierString = mat.group("modifiers");
+ String directiveParameterString = mat.group("params");
- if (dirMods == null) {
- dirMods = "";
- }
-
- if (dirParams == null) {
- dirParams = "";
- }
+ if (directiveModifierString == null) directiveModifierString = "";
+ if (directiveParameterString == null) directiveParameterString = "";
- boolean isUser = dirName == null && dirFunc != null;
+ boolean isUser = directiveName == null && directiveFunction != null;
- dir = new Decree(dirName, isUser, CLParameters.fromDirective(dirParams),
- CLModifiers.fromString(dirMods));
+ dir = new SimpleDecree(directiveName, isUser,
+ CLParameters.fromDirective(directiveParameterString),
+ CLModifiers.fromString(directiveModifierString));
}
if (tmp.equals("")) {
- Decree dcr = dir;
+ SimpleDecree dcr = dir;
dir = null;
return dcr;
}
- return new Decree(sb.toString());
+ return new SimpleDecree(sb.toString());
}
mat.appendTail(sb);
- return new Decree(sb.toString());
+ return new SimpleDecree(sb.toString());
}
/**
@@ -170,7 +157,7 @@ public class CLTokenizer implements Iterator<Decree> {
*
* @return A group decree with the given properties.
*/
- public GroupDecree nextGroup(Decree openedWith, String desiredClosing) {
+ public GroupDecree nextGroup(SimpleDecree openedWith, String desiredClosing) {
return nextGroup(openedWith, desiredClosing, null);
}
@@ -190,19 +177,18 @@ public class CLTokenizer implements Iterator<Decree> {
*
* @return A group decree with the given properties.
*/
- public GroupDecree nextGroup(Decree openedWith, String desiredClosing,
+ public GroupDecree nextGroup(SimpleDecree openedWith, String desiredClosing,
String clauseSep) {
GroupDecree newGroup = new GroupDecree();
newGroup.opening = openedWith;
- if (!hasNext())
- throw new NoSuchElementException("No decrees available");
+ if (!hasNext()) throw new NoSuchElementException("No decrees available");
ClauseDecree curClause = new ClauseDecree();
int nestingLevel = 1;
- Decree curDecree;
+ SimpleDecree curDecree;
do {
curDecree = next();
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/CLValue.java b/clformat/src/main/java/bjc/utils/ioutils/format/CLValue.java
index e4b4c32..c623e6f 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/CLValue.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/CLValue.java
@@ -2,6 +2,25 @@ package bjc.utils.ioutils.format;
import bjc.esodata.*;
+// @TODO Nov 13th, 2020 Ben Culkin :DefaultValues
+// Create some way to specify default values for the arguments to the various
+// asWhatever methods. This will cleanup use-site code for them
+
+/* @TODO Nov 13th, 2020 Ben Culkin :ParamVariables
+ * Create a new CLValue type that implements variables in a way. There are a
+ * number of different sorts of 'scopes' that could be useful, so here is the
+ * list of them:
+ * - Static variables, stored on the CLString instance
+ * - Global variables, stored on a map set on the FormatContext when formatting
+ * starts, and copied over whenever a new context is built.
+ * - Lexical variables, stored on the FormatContext, but in some sort of map which
+ * models lexical scopes (extend on FunctionalMap may do the right thing, not sure)
+ * - Local variables, stored on the FormatContext as well, but these
+ * aren't copied over when a new context is built.
+ *
+ * For static/global variables, maybe some equivalent of 'local' from perl to
+ * localize them.
+ */
/**
* Represents a parameter value to an edict that may have a dynamic value
* obtained from the format arguments.
@@ -18,23 +37,14 @@ public interface CLValue {
* @return The CLValue represented by the string.
*/
public static CLValue parse(String val) {
- if (val == null)
- return new NullValue();
-
- if (val.equalsIgnoreCase("V")) {
- return new VValue();
- }
+ if (val == null) return new NullValue();
switch (val) {
- case "V":
- case "v":
- return new VValue();
- case "#":
- return new HashValue();
- case "%":
- return new PercValue();
- default:
- return new LiteralValue(val);
+ case "V": // Fall-through, V is the same as v
+ case "v": return new VValue();
+ case "#": return new HashValue();
+ case "%": return new PercValue();
+ default: return new LiteralValue(val);
}
}
@@ -60,7 +70,7 @@ public interface CLValue {
* The format parameters to use.
*
* @param paramName
- * The user-intelligble name for the value.
+ * The user-intelligible name for the value.
*
* @param directive
* The directive this value is for.
@@ -107,7 +117,7 @@ public interface CLValue {
* The format parameters to use.
*
* @param paramName
- * The user-intelligble name for the value.
+ * The user-intelligible name for the value.
*
* @param directive
* The directive the value is for.
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/ClauseDecree.java b/clformat/src/main/java/bjc/utils/ioutils/format/ClauseDecree.java
index aa4efbe..db38eca 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/ClauseDecree.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/ClauseDecree.java
@@ -13,16 +13,16 @@ import bjc.utils.ioutils.ReportWriter;
*
* @author Ben Culkin
*/
-public class ClauseDecree {
+public class ClauseDecree implements Decree {
/**
* The decrees that make up the body of this clause.
*/
- public List<Decree> body;
+ public List<SimpleDecree> body;
/**
* The decree that terminated this clause.
*/
- public Decree terminator;
+ public SimpleDecree terminator;
/**
* Create a new blank clause decree.
@@ -38,12 +38,10 @@ public class ClauseDecree {
* @param children
* The decrees to form the body of the clause.
*/
- public ClauseDecree(Decree... children) {
+ public ClauseDecree(SimpleDecree... children) {
this();
- for (Decree child : children) {
- body.add(child);
- }
+ for (SimpleDecree child : children) body.add(child);
}
/**
@@ -55,7 +53,7 @@ public class ClauseDecree {
* @param children
* The decrees that form the body of the clause.
*/
- public ClauseDecree(Decree term, Decree... children) {
+ public ClauseDecree(SimpleDecree term, SimpleDecree... children) {
this(children);
this.terminator = term;
@@ -67,32 +65,40 @@ public class ClauseDecree {
* @param child
* The decree to add to this clause.
*/
- public void addChild(Decree child) {
+ public void addChild(SimpleDecree child) {
body.add(child);
}
@Override
public String toString() {
- try (ReportWriter rw = new ReportWriter()) {
- String term = "<null>";
- if (terminator != null)
- term = terminator.toString();
-
- rw.write("ClauseDecree (terminator " + term.toString() + ")");
- rw.indent();
- rw.write("\n");
-
- for (Decree kid : body) {
- rw.write("Child: " + kid.toString() + "\n");
- }
+ try (ReportWriter writer = new ReportWriter()) {
+ toReportWriter(writer);
- rw.dedent();
-
- return rw.toString();
+ return writer.toString();
} catch (IOException ioex) {
return "<IOEXCEPTION>";
}
// return String.format("ClauseDecree [body=%s, terminator=%s]", body,
// terminator);
}
+
+ /**
+ * Write the string version of this decree to a report writer.
+ * @param writer The report write to write to.
+ * @throws IOException If something goes wrong
+ */
+ public void toReportWriter(ReportWriter writer) throws IOException {
+ String term = "<null>";
+ if (terminator != null) term = terminator.toString();
+
+ writer.writef("ClauseDecree (terminator %s)", term);
+ writer.indent();
+ writer.write("\n");
+
+ int idx = 0;
+ for (SimpleDecree kid : body)
+ writer.writef("Child %d: %s\n", idx, kid.toString());
+
+ writer.dedent();
+ }
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/Decree.java b/clformat/src/main/java/bjc/utils/ioutils/format/Decree.java
index 9a90285..e06caf0 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/Decree.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/Decree.java
@@ -1,131 +1,14 @@
package bjc.utils.ioutils.format;
/**
- * A decree is the building blocks of what we need to pick and call a directive.
+ * Interface for all decrees.
+ *
+ * At the moment, this is just a marker interface, but there may be things added
+ * here.
+ *
+ * @author Ben Culkin
*
- * Namely, it is the name of the directive, any modifiers attached to the
- * directive, and any prefix parameters that are also attached to the directive.
- *
- * @author Ben Culkin.
*/
-public class Decree {
- /**
- * The name of the directive.
- */
- public String name;
-
- /**
- * Is this directive an actual directive, or just a literal string?
- */
- public boolean isLiteral;
-
- /**
- * Is this directive a user function call?
- */
- public boolean isUserCall;
-
- /**
- * The prefix parameters for this directive.
- */
- public CLParameters parameters;
-
- /**
- * The modifiers for this directive.
- */
- public CLModifiers modifiers;
-
- /**
- * Create a new blank decree.
- */
- public Decree() {
-
- }
-
- /**
- * Create a new literal text directive.
- *
- * @param txt
- * The text of the directive.
- */
- public Decree(String txt) {
- this.name = txt;
-
- this.isLiteral = true;
- }
-
- /**
- * Create a new directive.
- *
- * @param name
- * The name of the directive. Whether or not it is an actual
- * directive will be auto-determined (if it starts with a ~, it's
- * a directive.)
- *
- * @param params
- * The prefix parameters to the directive.
- *
- * @param mods
- * The modifiers to the directive.
- */
- public Decree(String name, CLParameters params, CLModifiers mods) {
- this.name = name;
-
- this.parameters = params;
-
- this.modifiers = mods;
-
- this.isLiteral = false;
- }
-
- /**
- * Create a new directive that may be a user function.
- *
- * @param name
- * The name of the directive. Whether or not it is an actual
- * directive will be auto-determined (if it starts with a ~ and is
- * not a user function, it's a directive.)
- *
- * @param isUser
- * Is this directive a user function?
- *
- * @param params
- * The prefix parameters to the directive.
- *
- * @param mods
- * The modifiers to the directive.
- */
- public Decree(String name, boolean isUser, CLParameters params, CLModifiers mods) {
- this.name = name;
-
- this.parameters = params;
-
- this.modifiers = mods;
-
- this.isUserCall = isUser;
-
- this.isLiteral = isUser;
- }
-
- /**
- * Check if this decree is a non-literal, with a particular name.
- *
- * @param nam
- * The name to see if we have.
- *
- * @return Whether or not the provided name equals our name.
- */
- public boolean isNamed(String nam) {
- // Literals don't have a meaningful name
- if (isLiteral)
- return false;
-
- return name.equals(nam);
- }
-
- @Override
- public String toString() {
- return String.format(
- "Decree [name='%s', isLiteral=%s, isUserCall=%s, parameters=%s, modifiers='%s']",
- name, isLiteral, isUserCall, parameters, modifiers);
- }
+public interface Decree {
+ // Marker interface, for now
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/GroupDecree.java b/clformat/src/main/java/bjc/utils/ioutils/format/GroupDecree.java
index 736e15a..c1c5c7b 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/GroupDecree.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/GroupDecree.java
@@ -13,16 +13,16 @@ import bjc.utils.ioutils.ReportWriter;
*
* @author Ben Culkin
*/
-public class GroupDecree implements Iterable<ClauseDecree> {
+public class GroupDecree implements Iterable<ClauseDecree>, Decree {
/**
* The decree that opened this group.
*/
- public Decree opening;
+ public SimpleDecree opening;
/**
* The decree that closed this group.
*/
- public Decree closing;
+ public SimpleDecree closing;
/**
* The clauses that make up the body of this group.
@@ -45,9 +45,7 @@ public class GroupDecree implements Iterable<ClauseDecree> {
public GroupDecree(ClauseDecree... children) {
this();
- for (ClauseDecree child : children) {
- body.add(child);
- }
+ for (ClauseDecree child : children) body.add(child);
}
/**
@@ -62,7 +60,7 @@ public class GroupDecree implements Iterable<ClauseDecree> {
* @param children
* The decree making up the body of the group.
*/
- public GroupDecree(Decree opening, Decree closing, ClauseDecree... children) {
+ public GroupDecree(SimpleDecree opening, SimpleDecree closing, ClauseDecree... children) {
this(children);
this.opening = opening;
@@ -105,33 +103,32 @@ public class GroupDecree implements Iterable<ClauseDecree> {
*
* @return The decrees that make up the body of the first clause.
*/
- public List<Decree> unwrap() {
+ public List<SimpleDecree> unwrap() {
return body.get(0).body;
}
@Override
public String toString() {
- try (ReportWriter rw = new ReportWriter()) {
- String open = "<null>";
+ try (ReportWriter writer = new ReportWriter()) {
+ String open = "<null>";
String close = "<null>";
- if (opening != null)
- open = opening.toString();
- if (closing != null)
- close = closing.toString();
-
- rw.write("GroupDecree (opening " + open + ") (closing " + close + ")");
- rw.indent();
- rw.write("\n");
+ if (opening != null) open = opening.toString();
+ if (closing != null) close = closing.toString();
+ writer.writef("GroupDecree (opening %s) (closing %s)\n", open, close);
+ writer.indent();
+
int idx = 0;
for (ClauseDecree clause : body) {
- rw.write("Clause " + idx++ + ": " + clause.toString() + "\n");
+ writer.writef("Clause %d:", idx++);
+
+ clause.toReportWriter(writer);
}
- rw.dedent();
+ writer.dedent();
- return rw.toString();
+ return writer.toString();
} catch (IOException ioex) {
return "<IOEXCEPTION>";
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/SimpleDecree.java b/clformat/src/main/java/bjc/utils/ioutils/format/SimpleDecree.java
new file mode 100644
index 0000000..a7db42f
--- /dev/null
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/SimpleDecree.java
@@ -0,0 +1,129 @@
+package bjc.utils.ioutils.format;
+
+/**
+ * A decree is the building blocks of what we need to pick and call a directive.
+ *
+ * Namely, it is the name of the directive, any modifiers attached to the
+ * directive, and any prefix parameters that are also attached to the directive.
+ *
+ * @author Ben Culkin.
+ */
+public class SimpleDecree implements Decree {
+ /**
+ * The name of the directive.
+ */
+ public String name;
+
+ /**
+ * Is this directive an actual directive, or just a literal string?
+ */
+ public boolean isLiteral;
+
+ /**
+ * Is this directive a user function call?
+ */
+ public boolean isUserCall;
+
+ /**
+ * The prefix parameters for this directive.
+ */
+ public CLParameters parameters;
+
+ /**
+ * The modifiers for this directive.
+ */
+ public CLModifiers modifiers;
+
+ /**
+ * Create a new blank decree.
+ */
+ public SimpleDecree() {
+
+ }
+
+ /**
+ * Create a new literal text directive.
+ *
+ * @param txt
+ * The text of the directive.
+ */
+ public SimpleDecree(String txt) {
+ this.name = txt;
+
+ this.isLiteral = true;
+ }
+
+ /**
+ * Create a new directive.
+ *
+ * @param name
+ * The name of the directive. Whether or not it is an actual
+ * directive will be auto-determined (if it starts with a ~, it's
+ * a directive.)
+ *
+ * @param params
+ * The prefix parameters to the directive.
+ *
+ * @param mods
+ * The modifiers to the directive.
+ */
+ public SimpleDecree(String name, CLParameters params, CLModifiers mods) {
+ this.name = name;
+
+ this.parameters = params;
+
+ this.modifiers = mods;
+
+ this.isLiteral = false;
+ }
+
+ /**
+ * Create a new directive that may be a user function.
+ *
+ * @param name
+ * The name of the directive. Whether or not it is an actual
+ * directive will be auto-determined (if it starts with a ~ and is
+ * not a user function, it's a directive.)
+ *
+ * @param isUser
+ * Is this directive a user function?
+ *
+ * @param params
+ * The prefix parameters to the directive.
+ *
+ * @param mods
+ * The modifiers to the directive.
+ */
+ public SimpleDecree(String name, boolean isUser, CLParameters params, CLModifiers mods) {
+ this.name = name;
+
+ this.parameters = params;
+
+ this.modifiers = mods;
+
+ this.isUserCall = isUser;
+
+ this.isLiteral = isUser;
+ }
+
+ /**
+ * Check if this decree is a non-literal, with a particular name.
+ *
+ * @param nam
+ * The name to see if we have.
+ *
+ * @return Whether or not the provided name equals our name.
+ */
+ public boolean isNamed(String nam) {
+ // Literals don't have a meaningful name
+ if (isLiteral) return false;
+ else return name.equals(nam);
+ }
+
+ @Override
+ public String toString() {
+ return String.format(
+ "Decree [name='%s', isLiteral=%s, isUserCall=%s, parameters=%s, modifiers='%s']",
+ name, isLiteral, isUserCall, parameters, modifiers);
+ }
+}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/AestheticDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/AestheticDirective.java
index 4fec0e0..0589992 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/AestheticDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/AestheticDirective.java
@@ -79,44 +79,47 @@ class AestheticEdict implements Edict {
// Check that we have an item
CLFormatter.checkItem(itemTape.item(), 'A');
- String tmp = itemTape.item().toString();
+ String itemString = itemTape.item().toString();
StringBuilder work = new StringBuilder();
- char padchar = padcharPar.asChar(itemTape, "padding character", "A", ' ');
+ char padchar = padcharPar.asChar(itemTape,
+ "padding character", "A", ' ');
- int mincol = mincolPar.asInt(itemTape, "minimum column count", "A", 0);
- int colinc = colincPar.asInt(itemTape, "padding increment", "A", 1);
- int minpad = minpadPar.asInt(itemTape, "minumum amount of padding", "A", 0);
+ int mincol = mincolPar.asInt(itemTape,
+ "minimum column count", "A", 0);
+ int colinc = colincPar.asInt(itemTape,
+ "padding increment", "A", 1);
+ int minpad = minpadPar.asInt(itemTape,
+ "minumum amount of padding", "A", 0);
- if (padBefore) {
- for (int i = 0; i < minpad; i++) {
- work.append(padchar);
- }
+ work.append(itemString);
- for (int i = work.length() + tmp.length(); i < mincol; i++) {
- for (int k = 0; k < colinc; k++) {
- work.append(padchar);
- }
- }
- }
+ String padding = createPadding(work, padchar, mincol, colinc, minpad);
+ if (padBefore) work.insert(0, padding);
+ else work.append(padding);
- work.append(tmp);
+ formCTX.writer.write(work.toString());
- if (!padBefore) {
- for (int i = 0; i < minpad; i++) {
- work.append(padchar);
- }
+ itemTape.right();
+ }
- for (int i = work.length(); i < mincol; i++) {
- for (int k = 0; k < colinc; k++) {
- work.append(padchar);
- }
+ private String createPadding(
+ StringBuilder work,
+ char padchar,
+ int mincol,
+ int colinc,
+ int minpad)
+ {
+ StringBuilder padding = new StringBuilder();
+ for (int i = 0; i < minpad; i++) padding.append(padchar);
+
+ for (int i = work.length(); i < mincol; i++) {
+ for (int k = 0; k < colinc; k++) {
+ padding.append(padchar);
}
}
- formCTX.writer.write(work.toString());
-
- itemTape.right();
+ return padding.toString();
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CaseDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CaseDirective.java
index 7e1ca87..2e45600 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CaseDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CaseDirective.java
@@ -75,10 +75,11 @@ class CaseEdict implements Edict {
while (mat.find()) {
mat.appendReplacement(sb, "");
- String word = mat.group(1);
- String ward = word.substring(0, 1).toUpperCase() + word.substring(1);
+ String word = mat.group(1);
+ String upperWord = word.substring(0, 1).toUpperCase()
+ + word.substring(1);
- sb.append(ward);
+ sb.append(upperWord);
sb.append(mat.group(2));
}
@@ -100,7 +101,8 @@ class CaseEdict implements Edict {
if (doCap) {
doCap = false;
- word = word.substring(0, 1).toUpperCase() + word.substring(1);
+ word = word.substring(0, 1).toUpperCase()
+ + word.substring(1);
}
sb.append(word);
@@ -116,8 +118,10 @@ class CaseEdict implements Edict {
strang = strang.toLowerCase();
break;
default:
- throw new IllegalArgumentException("INTERNAL ERROR: CaseEdict mode "
- + caseMode + " is not supported. This is a bug.");
+ String msg = String.format(
+ "INTERNAL ERROR: CaseEdict mode %s is not supported. This is a bug.",
+ caseMode);
+ throw new IllegalArgumentException(msg);
}
formCTX.writer.write(strang);
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CharacterDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CharacterDirective.java
index ca0432c..6fd2309 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CharacterDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CharacterDirective.java
@@ -30,24 +30,21 @@ class CharacterEdict implements Edict {
@Override
public void format(FormatContext formCTX) throws IOException {
- Object o = formCTX.items.item();
+ Object item = formCTX.items.item();
- CLFormatter.checkItem(o, 'C');
+ CLFormatter.checkItem(item, 'C');
- if (!(o instanceof Character)) {
- throw new IllegalFormatConversionException('C', o.getClass());
+ if (!(item instanceof Character)) {
+ throw new IllegalFormatConversionException('C', item.getClass());
}
- char ch = (Character) o;
+ char ch = (Character) item;
int codepoint = ch;
ReportWriter rw = formCTX.writer;
- if (printCharName) {
- rw.write(Character.getName(codepoint));
- } else {
- rw.write(ch);
- }
+ if (printCharName) rw.write(Character.getName(codepoint));
+ else rw.write(ch);
formCTX.items.right();
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CompileContext.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CompileContext.java
index 3c5f692..77c8402 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/CompileContext.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/CompileContext.java
@@ -21,7 +21,7 @@ public class CompileContext {
/**
* The decree that is currently being parsed.
*/
- public Decree decr;
+ public SimpleDecree decr;
/**
* Create a new compilation context.
@@ -35,7 +35,7 @@ public class CompileContext {
* @param dcr
* The decree currently being compiled.
*/
- public CompileContext(CLTokenizer dirs, CLFormatter fmt, Decree dcr) {
+ public CompileContext(CLTokenizer dirs, CLFormatter fmt, SimpleDecree dcr) {
directives = dirs;
formatter = fmt;
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/ConditionalDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/ConditionalDirective.java
index 491d629..c7b2999 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/ConditionalDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/ConditionalDirective.java
@@ -18,25 +18,24 @@ import bjc.utils.ioutils.format.*;
public class ConditionalDirective implements Directive {
@Override
public Edict compile(CompileContext compCTX) {
- CLModifiers mods = compCTX.decr.modifiers;
+ CLModifiers mods = compCTX.decr.modifiers;
CLParameters params = compCTX.decr.parameters;
+ // :ConfigDirectives
GroupDecree clauses = compCTX.directives.nextGroup(compCTX.decr, "]", ";");
ClauseDecree defClause = null;
- boolean isDefault = false;
+ boolean isDefault = false;
for (ClauseDecree clause : clauses) {
- if (isDefault)
- defClause = clause;
+ if (isDefault) defClause = clause;
if (clause.terminator != null && clause.terminator.modifiers.colonMod) {
isDefault = true;
}
}
- if (mods.starMod && clauses.size() > 0)
- defClause = clauses.clause();
+ if (mods.starMod && clauses.size() > 0)defClause = clauses.clause();
CLValue index = null;
@@ -56,8 +55,8 @@ public class ConditionalDirective implements Directive {
mode = ConditionalEdict.Mode.INDEX_CLAUSE;
}
- return new ConditionalEdict(mode, mods.dollarMod, index, clauses, defClause,
- compCTX.formatter);
+ return new ConditionalEdict(mode, mods.dollarMod, index, clauses,
+ defClause, compCTX.formatter);
}
}
@@ -87,7 +86,9 @@ class ConditionalEdict implements Edict {
this.clauses = new ArrayList<>();
for (ClauseDecree clause : clauses) {
- this.clauses.add(new CLString(fmt.compile(clause)));
+ List<Edict> compiled = fmt.compile(clause);
+
+ this.clauses.add(new CLString(compiled));
}
this.defClause = new CLString(fmt.compile(defClause));
@@ -101,47 +102,47 @@ class ConditionalEdict implements Edict {
try {
switch (condMode) {
case FIRST_SECOND: {
- Object o = items.item();
+ Object item = items.item();
items.right();
- boolean res = false;
- if (o == null) {
+ boolean conditionResult = false;
+ if (item == null) {
// throw new IllegalArgumentException("No parameter provided for [
// directive.");
- } else if (!(o instanceof Boolean)) {
- throw new IllegalFormatConversionException('[', o.getClass());
+ } else if (!(item instanceof Boolean)) {
+ throw new IllegalFormatConversionException('[', item.getClass());
} else {
- res = (Boolean) o;
+ conditionResult = (Boolean) item;
}
- CLString frmt;
- if (res) {
- frmt = clauses.get(1);
+ CLString pickedFormat;
+ if (conditionResult) {
+ pickedFormat = clauses.get(1);
} else {
- frmt = clauses.get(0);
+ pickedFormat = clauses.get(0);
}
- frmt.format(formCTX);
+ pickedFormat.format(formCTX);
}
break;
case OUTPUT_TRUE: {
- boolean res = false;
- Object o = items.item();
+ boolean conditionResult = false;
+ Object item = items.item();
- if (o == null) {
+ if (item == null) {
// throw new IllegalArgumentException("No parameter provided for [
// directive.");
- } else if (o instanceof Integer) {
- if ((Integer) o != 0) {
- res = true;
+ } else if (item instanceof Integer) {
+ if ((Integer) item != 0) {
+ conditionResult = true;
}
- } else if (o instanceof Boolean) {
- res = (Boolean) o;
+ } else if (item instanceof Boolean) {
+ conditionResult = (Boolean) item;
} else {
- throw new IllegalFormatConversionException('[', o.getClass());
+ throw new IllegalFormatConversionException('[', item.getClass());
}
- if (res) {
+ if (conditionResult) {
clauses.get(0).format(formCTX);
} else {
items.right();
@@ -149,34 +150,32 @@ class ConditionalEdict implements Edict {
}
break;
case INDEX_CLAUSE: {
- int res;
+ int clauseIndex;
if (index != null) {
- res = index.asInt(items, "conditional choice", "[", 0);
+ clauseIndex = index.asInt(items, "conditional choice", "[", 0);
} else {
- Object o = items.item();
+ Object item = items.item();
- if (o == null) {
+ if (item == null) {
throw new IllegalArgumentException(
"No parameter provided for [ directive.");
- } else if (!(o instanceof Number)) {
- throw new IllegalFormatConversionException('[', o.getClass());
+ } else if (!(item instanceof Number)) {
+ throw new IllegalFormatConversionException('[', item.getClass());
}
- res = ((Number) o).intValue();
+ clauseIndex = ((Number) item).intValue();
items.right();
}
if (decrementIndex)
- res -= 1;
+ clauseIndex -= 1;
- if (clauses.size() == 0 || res < 0 || res >= clauses.size()) {
- if (defClause != null) {
- defClause.format(formCTX.writer, items);
- }
+ if (clauses.size() == 0 || clauseIndex < 0 || clauseIndex >= clauses.size()) {
+ if (defClause != null) defClause.format(formCTX.writer, items);
} else {
- CLString frmt = clauses.get(res);
+ CLString frmt = clauses.get(clauseIndex);
frmt.format(formCTX.writer, items);
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/DecimalDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/DecimalDirective.java
new file mode 100644
index 0000000..6447eef
--- /dev/null
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/DecimalDirective.java
@@ -0,0 +1,89 @@
+package bjc.utils.ioutils.format.directives;
+
+import java.io.*;
+import java.text.*;
+
+import bjc.esodata.*;
+import bjc.utils.ioutils.format.*;
+
+/**
+ * Implementation of the `D directive.
+ *
+ * This is the most general directive for printing out decimal-numbers (floating
+ * point).
+ *
+ * @author Ben Culkin
+ */
+public class DecimalDirective implements Directive {
+ @Override
+ public Edict compile(CompileContext compCTX) {
+ CLParameters params = compCTX.decr.parameters;
+ @SuppressWarnings("unused")
+ CLModifiers mods = compCTX.decr.modifiers;
+
+ CLValue decForm = CLValue.nil();
+
+ switch(params.length()) {
+ case 0:
+ // Use the default
+ break;
+ case 1:
+ // Use the specified format
+ params.mapIndices("format");
+
+ decForm = params.resolveKey("format");
+ break;
+ default:
+ // @TODO 16 Oct, 2020 - Ben Culkin - :Preformat
+ // Add ability to specify a common/fixed set of formats
+ //
+ // @TODO 16 Oct, 2020 - Ben Culkin - :ErrorFix
+ // Instead of using IllegalArgumentException here, use a custom
+ // subtype of it with an appropriate name/auto-message forming
+ throw new IllegalArgumentException("Must provide 0 or 1 arguments to `D directive");
+ }
+
+ return new DecimalEdict(decForm);
+ }
+}
+
+class DecimalEdict implements Edict {
+ private CLValue decFormat;
+
+ public DecimalEdict(CLValue decForm) {
+ this.decFormat = decForm;
+ }
+
+ @Override
+ public void format(FormatContext formCTX) throws IOException {
+ Tape<Object> itemTape = formCTX.items;
+
+ CLFormatter.checkItem(itemTape.item(), "`D");
+
+ NumberFormat numForm = NumberFormat.getInstance();
+
+ String decFormString = decFormat.getValue(itemTape);
+
+ if (decFormString == null || decFormString.equals("")) {
+ // Use the default if not provided.
+ } else {
+ if (numForm instanceof DecimalFormat) {
+ ((DecimalFormat)numForm).applyPattern(decFormString);
+ } else {
+ String clsName = numForm.getClass().getName();
+
+ String msg = String.format("INTERNAL ERROR: Unknown NumberFormat type %s, expected DecimalFormat or compatible", clsName);
+
+ throw new UnsupportedOperationException(msg);
+ }
+ }
+
+ //StringBuffer work = new StringBuffer();
+
+ //numForm.format(itemTape.item(), work, ZERO_FIELD);
+
+ // formCTX.writer.write(work.toString());
+ formCTX.writer.write(numForm.format(itemTape.item()));
+ itemTape.right();
+ }
+}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/EscapeDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/EscapeDirective.java
index 343a54d..3610bda 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/EscapeDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/EscapeDirective.java
@@ -14,7 +14,7 @@ public class EscapeDirective implements Directive {
@Override
public Edict compile(CompileContext compCTX) {
CLParameters params = compCTX.decr.parameters;
- CLModifiers mods = compCTX.decr.modifiers;
+ CLModifiers mods = compCTX.decr.modifiers;
CLValue param1 = CLValue.nil();
CLValue param2 = CLValue.nil();
@@ -27,28 +27,31 @@ public class EscapeDirective implements Directive {
break;
case 1:
mode = EscapeEdict.Mode.COUNT;
+
params.mapIndices("count");
param1 = params.resolveKey("count");
break;
case 2:
+ mode = EscapeEdict.Mode.EQUALITY;
+
params.mapIndices("lhand", "rhand");
param1 = params.resolveKey("lhand");
param2 = params.resolveKey("rhand");
- mode = EscapeEdict.Mode.EQUALITY;
break;
case 3:
+ mode = EscapeEdict.Mode.RANGE;
+
params.mapIndices("lower", "ival", "upper");
param1 = params.resolveKey("lower");
param2 = params.resolveKey("ival");
param3 = params.resolveKey("upper");
- mode = EscapeEdict.Mode.RANGE;
break;
default:
throw new IllegalArgumentException("Too many parameters to ^ directive");
}
- return new EscapeEdict(mods.atMod, mode, mods.colonMod, param1, param2, param3,
- mods.dollarMod);
+ return new EscapeEdict(mods.atMod, mode, mods.colonMod,
+ param1, param2, param3, mods.dollarMod);
}
}
@@ -88,8 +91,7 @@ class EscapeEdict implements Edict {
Tape<Object> items = formCTX.items;
- if (advance)
- items.right();
+ if (advance) items.right();
switch (mode) {
case END:
@@ -102,15 +104,15 @@ class EscapeEdict implements Edict {
}
break;
case EQUALITY: {
- int left = param1.asInt(items, "left-hand condition", "^", 0);
+ int left = param1.asInt(items, "left-hand condition", "^", 0);
int right = param2.asInt(items, "right-hand condition", "^", 0);
shouldExit = (left == right);
}
break;
case RANGE: {
- int low = param1.asInt(items, "lower-bound condition", "^", 0);
- int mid = param2.asInt(items, "interval condition", "^", 0);
+ int low = param1.asInt(items, "lower-bound condition", "^", 0);
+ int mid = param2.asInt(items, "interval condition", "^", 0);
int high = param3.asInt(items, "upper-bound condition", "^", 0);
shouldExit = (low <= mid) && (mid <= high);
@@ -121,15 +123,9 @@ class EscapeEdict implements Edict {
"Escape condition mode " + mode + " isn't supported");
}
- if (advance)
- items.left();
-
- if (isNegated) {
- shouldExit = !shouldExit;
- }
+ if (advance) items.left();
+ if (isNegated) shouldExit = !shouldExit;
- if (shouldExit) {
- throw new DirectiveEscape(terminateIteration);
- }
+ if (shouldExit) throw new DirectiveEscape(terminateIteration);
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/FormatParameters.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/FormatParameters.java
index 3ffa4ed..e2884d3 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/FormatParameters.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/FormatParameters.java
@@ -18,7 +18,7 @@ public class FormatParameters {
/**
* The {@link ReportWriter} used for output.
*/
- public ReportWriter rw;
+ public ReportWriter writer;
/**
* The current format parameter.
@@ -28,7 +28,7 @@ public class FormatParameters {
/**
* The current decree.
*/
- public Decree decr;
+ public SimpleDecree decr;
/**
* The current format parameters.
@@ -43,12 +43,12 @@ public class FormatParameters {
/**
* The formatter we are going from.
*/
- public CLFormatter fmt;
+ public CLFormatter formatter;
/**
* Create a new set of format parameters.
*
- * @param rw
+ * @param writer
* The writer we are sending output to.
*
* @param item
@@ -63,12 +63,12 @@ public class FormatParameters {
* @param dirIter
* The set of format decrees.
*
- * @param fmt
+ * @param formatter
* The formatter we are using
*/
- public FormatParameters(ReportWriter rw, Object item, Decree decr,
- Tape<Object> tParams, CLTokenizer dirIter, CLFormatter fmt) {
- this.rw = rw;
+ public FormatParameters(ReportWriter writer, Object item, SimpleDecree decr,
+ Tape<Object> tParams, CLTokenizer dirIter, CLFormatter formatter) {
+ this.writer = writer;
this.item = item;
@@ -78,7 +78,7 @@ public class FormatParameters {
this.dirIter = dirIter;
- this.fmt = fmt;
+ this.formatter = formatter;
}
/**
@@ -105,7 +105,7 @@ public class FormatParameters {
* @return The compilation context from these parameters.
*/
public CompileContext toCompileCTX() {
- return new CompileContext(dirIter, fmt, decr);
+ return new CompileContext(dirIter, formatter, decr);
}
/**
@@ -114,6 +114,6 @@ public class FormatParameters {
* @return The format context from these parameters.
*/
public FormatContext toFormatCTX() {
- return new FormatContext(rw, tParams);
+ return new FormatContext(writer, tParams);
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/FreshlineDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/FreshlineDirective.java
index 7402c98..dedbbac 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/FreshlineDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/FreshlineDirective.java
@@ -38,11 +38,8 @@ class FreshlineEdict implements Edict {
public void format(FormatContext formCTX) throws IOException {
int nTimes = times.asInt(formCTX.items, "occurance count", "&", 1);
- if (formCTX.writer.isLastCharNL())
- nTimes -= 1;
+ if (formCTX.writer.isLastCharNL()) nTimes -= 1;
- for (int i = 0; i < nTimes; i++) {
- formCTX.writer.write("\n");
- }
+ for (int i = 0; i < nTimes; i++) formCTX.writer.write("\n");
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/GeneralNumberDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/GeneralNumberDirective.java
index f12880e..f2fca71 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/GeneralNumberDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/GeneralNumberDirective.java
@@ -57,7 +57,7 @@ public abstract class GeneralNumberDirective implements Directive {
*/
protected NumberParams getParams(CompileContext compCTX, int argidx) {
CLParameters params = compCTX.decr.parameters;
- CLModifiers mods = compCTX.decr.modifiers;
+ CLModifiers mods = compCTX.decr.modifiers;
NumberParams np = new NumberParams();
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/IndentDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/IndentDirective.java
index a19866f..4ccf627 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/IndentDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/IndentDirective.java
@@ -14,9 +14,7 @@ public class IndentDirective implements Directive {
CLParameters params = compCTX.decr.parameters;
CLModifiers mods = compCTX.decr.modifiers;
- if (mods.dollarMod) {
- return new IndentConfigureEdict();
- }
+ if (mods.dollarMod) return new IndentConfigureEdict();
CLValue indentCount = CLValue.nil();
if (params.length() >= 1) {
@@ -52,11 +50,8 @@ class IndentEdict implements Edict {
}
if (isRelative) {
- if (dedent) {
- formCTX.writer.dedent(numIndents);
- } else {
- formCTX.writer.indent(numIndents);
- }
+ if (dedent) formCTX.writer.dedent(numIndents);
+ else formCTX.writer.indent(numIndents);
} else {
if (dedent) {
throw new IllegalArgumentException("Cannot have negative indent level");
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/InflectDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/InflectDirective.java
index 8ed39fb..937bd9d 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/InflectDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/InflectDirective.java
@@ -15,13 +15,13 @@ import bjc.utils.ioutils.format.*;
public class InflectDirective implements Directive {
@Override
public Edict compile(CompileContext compCTX) {
- List<Decree> body = new ArrayList<>();
+ List<SimpleDecree> body = new ArrayList<>();
int nestLevel = 1;
- Iterator<Decree> dirIter = compCTX.directives;
+ Iterator<SimpleDecree> dirIter = compCTX.directives;
while (dirIter.hasNext()) {
- Decree decr = dirIter.next();
+ SimpleDecree decr = dirIter.next();
if (decr.isLiteral) {
body.add(decr);
@@ -30,11 +30,11 @@ public class InflectDirective implements Directive {
String dirName = decr.name;
+ // @FIXME Nov 13th 2020 Ben Culkin :GroupDecree
+ // Would it be appropriate to convert this to use GroupDecree?
if (dirName != null) {
if (dirName.equals("`[")) {
- if (nestLevel > 0) {
- body.add(decr);
- }
+ if (nestLevel > 0) body.add(decr);
nestLevel += 1;
} else if (Directive.isOpening(dirName)) {
@@ -44,9 +44,8 @@ public class InflectDirective implements Directive {
} else if (dirName.equals("`]")) {
nestLevel = Math.max(0, nestLevel - 1);
- /* End the iteration. */
- if (nestLevel == 0)
- break;
+ /* End the iteration if this was the final `]. */
+ if (nestLevel == 0) break;
} else if (Directive.isClosing(dirName)) {
nestLevel = Math.max(0, nestLevel - 1);
} else {
@@ -63,17 +62,17 @@ public class InflectDirective implements Directive {
class InflectEdict implements Edict {
private CLString body;
- public InflectEdict(List<Decree> body, CLFormatter fmt) {
+ public InflectEdict(List<SimpleDecree> body, CLFormatter fmt) {
this.body = new CLString(fmt.compile(body));
}
@Override
public void format(FormatContext formCTX) throws IOException {
- ReportWriter nrw = formCTX.writer.duplicate(new StringWriter());
+ ReportWriter newWriter = formCTX.writer.duplicate(new StringWriter());
- body.format(nrw, formCTX.items);
+ body.format(newWriter, formCTX.items);
- String strang = nrw.toString();
+ String strang = newWriter.toString();
strang = InflectionML.inflect(strang);
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/IterationDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/IterationDirective.java
index b8cdf36..34f815d 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/IterationDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/IterationDirective.java
@@ -16,30 +16,26 @@ public class IterationDirective implements Directive {
public Edict compile(CompileContext compCTX) {
IterationEdict.Mode mode;
- List<Decree> body = new ArrayList<>();
+ List<SimpleDecree> body = new ArrayList<>();
- Iterator<Decree> dirIter = compCTX.directives;
+ Iterator<SimpleDecree> dirIter = compCTX.directives;
+ // :GroupDecree
while (dirIter.hasNext()) {
- Decree decr = dirIter.next();
+ SimpleDecree decr = dirIter.next();
if (decr.isLiteral) {
body.add(decr);
- continue;
- }
-
- String dirName = decr.name;
-
- if (dirName != null) {
- if (dirName.equals("}")) {
- break;
- } else {
- /* Not a special directive. */
- body.add(decr);
+ } else {
+ String dirName = decr.name;
+
+ if (dirName != null) {
+ if (dirName.equals("}")) break;
+ else body.add(decr);
}
}
}
CLParameters params = compCTX.decr.parameters;
- CLModifiers mods = compCTX.decr.modifiers;
+ CLModifiers mods = compCTX.decr.modifiers;
CLValue maxItr = CLValue.nil();
if (params.length() > 0) {
@@ -77,7 +73,7 @@ class IterationEdict implements Edict {
private CLValue maxItrVal;
- public IterationEdict(Mode mode, List<Decree> body, CLFormatter fmt, CLValue maxItr) {
+ public IterationEdict(Mode mode, List<SimpleDecree> body, CLFormatter fmt, CLValue maxItr) {
this.mode = mode;
this.body = new CLString(fmt.compile(body));
@@ -96,7 +92,7 @@ class IterationEdict implements Edict {
Object iter = formCTX.items.item();
boolean usingString = false;
- String strang = null;
+ String currBody = null;
if (body.isEmpty()) {
/* Grab an argument. */
@@ -105,7 +101,7 @@ class IterationEdict implements Edict {
}
usingString = true;
- strang = (String) iter;
+ currBody = (String) iter;
if (!formCTX.items.right()) {
throw new IllegalArgumentException(
@@ -119,8 +115,7 @@ class IterationEdict implements Edict {
case ALL_SUBLISTS:
try {
do {
- if (numIterations > maxIterations)
- break;
+ if (numIterations > maxIterations) break;
numIterations += 1;
if (!(iter instanceof Iterable<?>)) {
@@ -129,8 +124,8 @@ class IterationEdict implements Edict {
}
@SuppressWarnings("unchecked")
- Iterable<Object> nitr = (Iterable<Object>) iter;
- Tape<Object> nParams = new SingleTape<>(nitr);
+ Iterable<Object> nitr = (Iterable<Object>) iter;
+ Tape<Object> nParams = new SingleTape<>(nitr);
try {
if (usingString) {
@@ -140,15 +135,13 @@ class IterationEdict implements Edict {
// CLString and then caching those compiled string. However,
// we aren't
// doing that now. -- ben, 12/17/19
- fmt.doFormatString(strang, formCTX.writer, nParams, false);
+ fmt.doFormatString(currBody, formCTX.writer, nParams, false);
} else {
body.format(formCTX.writer, nParams);
}
} catch (DirectiveEscape eex) {
if (eex.endIteration) {
- if (formCTX.items.atEnd()) {
- throw eex;
- }
+ if (formCTX.items.atEnd()) throw eex;
}
}
@@ -162,14 +155,13 @@ class IterationEdict implements Edict {
case ALL:
try {
while (!formCTX.items.atEnd()) {
- if (numIterations > maxIterations)
- break;
+ if (numIterations > maxIterations) break;
numIterations += 1;
if (usingString) {
// :DynamicFormatString
- fmt.doFormatString(strang, formCTX.writer, formCTX.items, false);
+ fmt.doFormatString(currBody, formCTX.writer, formCTX.items, false);
} else {
body.format(formCTX);
}
@@ -192,8 +184,7 @@ class IterationEdict implements Edict {
while (itr.hasNext()) {
Object obj = itr.next();
- if (numIterations > maxIterations)
- break;
+ if (numIterations > maxIterations) break;
numIterations += 1;
if (!(obj instanceof Iterable<?>)) {
@@ -208,13 +199,12 @@ class IterationEdict implements Edict {
try {
if (usingString) {
// :DynamicString
- fmt.doFormatString(strang, formCTX.writer, nParams, false);
+ fmt.doFormatString(currBody, formCTX.writer, nParams, false);
} else {
body.format(formCTX.writer, nParams);
}
} catch (DirectiveEscape eex) {
- if (eex.endIteration && !itr.hasNext())
- throw eex;
+ if (eex.endIteration && !itr.hasNext()) throw eex;
}
}
} catch (DirectiveEscape eex) {
@@ -232,13 +222,12 @@ class IterationEdict implements Edict {
Tape<Object> nParams = new SingleTape<>(itr);
while (!nParams.atEnd()) {
- if (numIterations > maxIterations)
- break;
+ if (numIterations > maxIterations) break;
numIterations += 1;
if (usingString) {
// :DynamicString
- fmt.doFormatString(strang, formCTX.writer, nParams, false);
+ fmt.doFormatString(currBody, formCTX.writer, nParams, false);
} else {
body.format(formCTX.writer, nParams);
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/LiteralDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/LiteralDirective.java
index c63bf4e..3e4a5ef 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/LiteralDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/LiteralDirective.java
@@ -63,8 +63,6 @@ class LiteralEdict implements Edict {
num = nTimes.asInt(formCTX.items, "occurance count", "literal", 1);
}
- for (int i = 0; i < num; i++) {
- formCTX.writer.write(lit);
- }
+ for (int i = 0; i < num; i++) formCTX.writer.write(lit);
}
}
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/NumberDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/NumberDirective.java
index c7d4ab6..de1bb28 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/NumberDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/NumberDirective.java
@@ -71,9 +71,8 @@ class NumberEdict implements Edict {
long val = ((Number) item).longValue();
- int mincol = np.mincol.asInt(formCTX.items, "minimum column count", directive, 0);
- char padchar
- = np.padchar.asChar(formCTX.items, "padding character", directive, ' ');
+ int mincol = np.mincol.asInt(formCTX.items, "minimum column count", directive, 0);
+ char padchar = np.padchar.asChar(formCTX.items, "padding character", directive, ' ');
boolean signed = np.signed;
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/RadixDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/RadixDirective.java
index e3ccfb4..24af908 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/RadixDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/RadixDirective.java
@@ -27,13 +27,9 @@ public class RadixDirective extends GeneralNumberDirective {
NumberParams np = null;
if (params.length() == 0) {
- if (mods.atMod) {
- mode = RadixEdict.Mode.ROMAN;
- } else if (mods.colonMod) {
- mode = RadixEdict.Mode.ORDINAL;
- } else {
- mode = RadixEdict.Mode.CARDINAL;
- }
+ if (mods.atMod) mode = RadixEdict.Mode.ROMAN;
+ else if (mods.colonMod) mode = RadixEdict.Mode.ORDINAL;
+ else mode = RadixEdict.Mode.CARDINAL;
} else {
mode = RadixEdict.Mode.NORMAL;
@@ -96,11 +92,9 @@ class RadixEdict implements Edict {
res = NumberUtils.toCardinal(val);
break;
case NORMAL: {
- int radix = radixVal.asInt(formCTX.items, "radix", "R", 10);
-
- int mincol = np.mincol.asInt(formCTX.items, "minimum column count", "R", 0);
- char padchar
- = np.padchar.asChar(formCTX.items, "padding character", "R", ' ');
+ int radix = radixVal.asInt(formCTX.items, "radix", "R", 10);
+ int mincol = np.mincol.asInt(formCTX.items, "minimum column count", "R", 0);
+ char padchar = np.padchar.asChar(formCTX.items, "padding character", "R", ' ');
boolean signed = np.signed;
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/RecursiveDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/RecursiveDirective.java
index 41c7adb..c60d0bb 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/RecursiveDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/RecursiveDirective.java
@@ -48,9 +48,7 @@ class RecursiveEdict implements Edict {
fmt.doFormatString(bod, formCTX.writer, formCTX.items, true);
} catch (DirectiveEscape eex) {
- if (eex.endIteration) {
- throw new UnexpectedColonEscape();
- }
+ if (eex.endIteration) throw new UnexpectedColonEscape();
}
} else {
if (formCTX.items.atEnd()) {
@@ -58,11 +56,11 @@ class RecursiveEdict implements Edict {
"? directive requires two format parameters");
}
- Object o = formCTX.items.item();
+ Object item = formCTX.items.item();
formCTX.items.right();
- if (!(o instanceof Iterable<?>)) {
- throw new MismatchedFormatArgType("?", Iterable.class, o.getClass());
+ if (!(item instanceof Iterable<?>)) {
+ throw new MismatchedFormatArgType("?", Iterable.class, item.getClass());
}
if (!(body instanceof String)) {
@@ -70,7 +68,7 @@ class RecursiveEdict implements Edict {
}
@SuppressWarnings("unchecked")
- Iterable<Object> itb = (Iterable<Object>) o;
+ Iterable<Object> itb = (Iterable<Object>) item;
Tape<Object> newParams = new SingleTape<>(itb);
try {
diff --git a/clformat/src/main/java/bjc/utils/ioutils/format/directives/TabulateDirective.java b/clformat/src/main/java/bjc/utils/ioutils/format/directives/TabulateDirective.java
index 91b40c3..a9814ba 100644
--- a/clformat/src/main/java/bjc/utils/ioutils/format/directives/TabulateDirective.java
+++ b/clformat/src/main/java/bjc/utils/ioutils/format/directives/TabulateDirective.java
@@ -68,7 +68,7 @@ class TabulateEdict implements Edict {
@Override
public void format(FormatContext formCTX) throws IOException {
- ReportWriter rw = formCTX.writer;
+ ReportWriter writer = formCTX.writer;
Tape<Object> itms = formCTX.items;
@@ -76,24 +76,19 @@ class TabulateEdict implements Edict {
int currCol;
- if (fromIndent) {
- currCol = rw.getIndentPos();
- } else {
- currCol = rw.getLinePos();
- }
+ if (fromIndent) currCol = writer.getIndentPos();
+ else currCol = writer.getLinePos();
if (isRelative) {
int colinc = colincVal.asInt(itms, "column increment", "T", 1);
int colrel = colidVal.asInt(itms, "relative column number", "T", 1);
- for (int i = 0; i < colrel; i++) {
- rw.write(padchar);
- }
+ for (int i = 0; i < colrel; i++) writer.write(padchar);
int nSpaces = 0;
while ((currCol + nSpaces) % colinc != 0) {
- rw.write(padchar);
+ writer.write(padchar);
nSpaces++;
}
@@ -102,17 +97,14 @@ class TabulateEdict implements Edict {
int colnum = colidVal.asInt(itms, "column number", "T", 1);
if (currCol < colnum) {
- for (int i = currCol; i < colnum; i++) {
- rw.write(padchar);
- }
+ for (int i = currCol; i < colnum; i++) writer.write(padchar);
} else {
- if (colinc == 0)
- return;
+ if (colinc == 0) return;
int k = 0;
while (colnum > (currCol + (k * colinc))) {
- rw.write(padchar);
+ writer.write(padchar);
k++;
}
diff --git a/clformat/src/test/java/bjc/utils/test/ioutils/CLFormatterTest.java b/clformat/src/test/java/bjc/utils/test/ioutils/CLFormatterTest.java
index 8cd3e61..629e802 100644
--- a/clformat/src/test/java/bjc/utils/test/ioutils/CLFormatterTest.java
+++ b/clformat/src/test/java/bjc/utils/test/ioutils/CLFormatterTest.java
@@ -125,7 +125,8 @@ public class CLFormatterTest {
assertFormat("foo", "~A", "foo");
assertFormat("foobar ", "~7A", "foobar");
assertFormat(" foobar", "~7@A", "foobar");
- assertFormat(" foobar", "~#mincol;8,#colinc;2,#minpad;1,#padchar;' @A",
+ assertFormat(" foobar",
+ "~#mincol;8,#colinc;2,#minpad;1,#padchar;' @A",
"foobar");
}
@@ -138,11 +139,22 @@ public class CLFormatterTest {
}
@Test
+ public void testFloatPrinting() {
+ assertFormat("1", "~`D", 1);
+ assertFormat("1.1", "~`D", 1.1);
+ }
+
+ @Test
public void testRandomCases() {
// Random test cases
- assertEquals("3 dogs are here", format("~D dog~:[s are~; is~] here", 3, 3 == 1));
+ assertEquals("3 dogs are here",
+ format("~D dog~:[s are~; is~] here", 3, 3 == 1));
}
+ /*private void assertFormat(String msg, String res, String fomt, Object... params) {
+ assertEquals(msg, res, format(fomt, params));
+ }*/
+
private void assertFormat(String res, String fomt, Object... params) {
assertEquals(res, format(fomt, params));
}
diff --git a/clformat/src/test/java/bjc/utils/test/ioutils/CLTokenizerTest.java b/clformat/src/test/java/bjc/utils/test/ioutils/CLTokenizerTest.java
index 429c089..20d64b0 100644
--- a/clformat/src/test/java/bjc/utils/test/ioutils/CLTokenizerTest.java
+++ b/clformat/src/test/java/bjc/utils/test/ioutils/CLTokenizerTest.java
@@ -18,7 +18,7 @@ public class CLTokenizerTest {
CLTokenizer tokenzer = new CLTokenizer("");
assertTrue("Empty tokenizer has a decree", tokenzer.hasNext());
- Decree dec = tokenzer.next();
+ SimpleDecree dec = tokenzer.next();
assertFalse("Empty tokenizer has only one decree", tokenzer.hasNext());
assertTrue("Decree from empty tokenizer is a literal", dec.isLiteral);
diff --git a/commander/.classpath b/commander/.classpath
new file mode 100644
index 0000000..7d66676
--- /dev/null
+++ b/commander/.classpath
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<classpath>
+ <classpathentry kind="src" output="target/classes" path="src/main/java">
+ <attributes>
+ <attribute name="optional" value="true"/>
+ <attribute name="maven.pomderived" value="true"/>
+ </attributes>
+ </classpathentry>
+ <classpathentry kind="src" output="target/test-classes" path="src/test/java">
+ <attributes>
+ <attribute name="test" value="true"/>
+ <attribute name="optional" value="true"/>
+ <attribute name="maven.pomderived" value="true"/>
+ </attributes>
+ </classpathentry>
+ <classpathentry kind="src" path="src/example/java"/>
+ <classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-1.8">
+ <attributes>
+ <attribute name="maven.pomderived" value="true"/>
+ </attributes>
+ </classpathentry>
+ <classpathentry kind="con" path="org.eclipse.m2e.MAVEN2_CLASSPATH_CONTAINER">
+ <attributes>
+ <attribute name="maven.pomderived" value="true"/>
+ </attributes>
+ </classpathentry>
+ <classpathentry kind="output" path="target/classes"/>
+</classpath>
diff --git a/commander/.gitignore b/commander/.gitignore
new file mode 100644
index 0000000..b83d222
--- /dev/null
+++ b/commander/.gitignore
@@ -0,0 +1 @@
+/target/
diff --git a/commander/.project b/commander/.project
new file mode 100644
index 0000000..18dea93
--- /dev/null
+++ b/commander/.project
@@ -0,0 +1,23 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<projectDescription>
+ <name>commander</name>
+ <comment></comment>
+ <projects>
+ </projects>
+ <buildSpec>
+ <buildCommand>
+ <name>org.eclipse.jdt.core.javabuilder</name>
+ <arguments>
+ </arguments>
+ </buildCommand>
+ <buildCommand>
+ <name>org.eclipse.m2e.core.maven2Builder</name>
+ <arguments>
+ </arguments>
+ </buildCommand>
+ </buildSpec>
+ <natures>
+ <nature>org.eclipse.jdt.core.javanature</nature>
+ <nature>org.eclipse.m2e.core.maven2Nature</nature>
+ </natures>
+</projectDescription>
diff --git a/commander/.settings/org.eclipse.core.resources.prefs b/commander/.settings/org.eclipse.core.resources.prefs
new file mode 100644
index 0000000..f9fe345
--- /dev/null
+++ b/commander/.settings/org.eclipse.core.resources.prefs
@@ -0,0 +1,4 @@
+eclipse.preferences.version=1
+encoding//src/main/java=UTF-8
+encoding//src/test/java=UTF-8
+encoding/<project>=UTF-8
diff --git a/commander/.settings/org.eclipse.jdt.core.prefs b/commander/.settings/org.eclipse.jdt.core.prefs
new file mode 100644
index 0000000..2f5cc74
--- /dev/null
+++ b/commander/.settings/org.eclipse.jdt.core.prefs
@@ -0,0 +1,8 @@
+eclipse.preferences.version=1
+org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.8
+org.eclipse.jdt.core.compiler.compliance=1.8
+org.eclipse.jdt.core.compiler.problem.enablePreviewFeatures=disabled
+org.eclipse.jdt.core.compiler.problem.forbiddenReference=warning
+org.eclipse.jdt.core.compiler.problem.reportPreviewFeatures=ignore
+org.eclipse.jdt.core.compiler.release=disabled
+org.eclipse.jdt.core.compiler.source=1.8
diff --git a/commander/.settings/org.eclipse.m2e.core.prefs b/commander/.settings/org.eclipse.m2e.core.prefs
new file mode 100644
index 0000000..f897a7f
--- /dev/null
+++ b/commander/.settings/org.eclipse.m2e.core.prefs
@@ -0,0 +1,4 @@
+activeProfiles=
+eclipse.preferences.version=1
+resolveWorkspaceProjects=true
+version=1
diff --git a/commander/pom.xml b/commander/pom.xml
new file mode 100644
index 0000000..eaba4a6
--- /dev/null
+++ b/commander/pom.xml
@@ -0,0 +1,35 @@
+<?xml version="1.0"?>
+<project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0"
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
+ <modelVersion>4.0.0</modelVersion>
+
+ <parent>
+ <groupId>bjc</groupId>
+ <artifactId>BJCUtils-Parent</artifactId>
+ <version>1.0.0</version>
+ </parent>
+
+ <artifactId>commander</artifactId>
+ <name>commander</name>
+
+ <url>http://maven.apache.org</url>
+
+ <properties>
+ <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
+ </properties>
+
+ <dependencies>
+ <dependency>
+ <groupId>junit</groupId>
+ <artifactId>junit</artifactId>
+ <version>3.8.1</version>
+ <scope>test</scope>
+ </dependency>
+ <dependency>
+ <groupId>bjc</groupId>
+ <artifactId>BJC-Utils2</artifactId>
+ <version>1.0.0</version>
+ <type>jar</type>
+ </dependency>
+ </dependencies>
+</project>
diff --git a/commander/src/example/java/bjc/commander/CommanderCLI.java b/commander/src/example/java/bjc/commander/CommanderCLI.java
new file mode 100644
index 0000000..0f208b6
--- /dev/null
+++ b/commander/src/example/java/bjc/commander/CommanderCLI.java
@@ -0,0 +1,27 @@
+package bjc.commander;
+
+import java.io.*;
+import java.util.*;
+
+/** Runner for the general command-line interface for Commander.
+ *
+ * @author Ben Culkin */
+public class CommanderCLI {
+
+ /* :CLIArgsParsing */
+ /** Run the command line interface
+ *
+ * @param args
+ * Ignored CLI args. */
+ public static void main(String[] args) {
+ /* Create/configure I/O sources. */
+ Commander reader = new Commander();
+
+ reader.ioReaders.put("stdio", new InputStreamReader(System.in));
+
+ Scanner input = new Scanner(System.in);
+ reader.run(input, "console", true);
+ input.close();
+ }
+
+}
diff --git a/commander/src/main/java/bjc/commander/Commander.java b/commander/src/main/java/bjc/commander/Commander.java
new file mode 100644
index 0000000..0c62fd3
--- /dev/null
+++ b/commander/src/main/java/bjc/commander/Commander.java
@@ -0,0 +1,386 @@
+package bjc.commander;
+
+import static bjc.utils.cli.objects.Command.CommandStatus.*;
+
+import java.io.*;
+import java.util.*;
+import java.util.function.*;
+import java.util.logging.*;
+import java.util.regex.*;
+
+import bjc.utils.cli.objects.*;
+import bjc.utils.cli.objects.Command.*;
+import bjc.utils.ioutils.blocks.*;
+
+/** CLI interface for messing around with various java objects.
+ *
+ * @author Ben Culkin */
+public class Commander {
+ /** All of the configured block readers. */
+ public final Map<String, BlockReader> blockReaders;
+ /** All of the configured I/O sources. */
+ public final Map<String, Reader> ioReaders;
+
+ /* Logger. */
+ private final Logger LOGGER = Logger.getLogger(Commander.class.getName());
+
+ /** Create a new CLI for configuring BlockReaders. */
+ public Commander() {
+ ioReaders = new HashMap<>();
+ blockReaders = new HashMap<>();
+ }
+
+ /** Run the CLI on an input source.
+ *
+ * @param input
+ * The place to read input from.
+ *
+ * @param srcName
+ * The name of the place to read input from.
+ *
+ * @param interactive
+ * Whether or not the source is interactive */
+ public void run(Scanner input, String srcName, boolean interactive) {
+ int lno = 0;
+
+ do {
+ /* Print a prompt. */
+ if (interactive) System.out.printf("reader-conf(%d)>", lno);
+
+ /* Read a line. */
+ String ln = input.nextLine();
+ lno += 1;
+
+ /* Parse the command. */
+ Command com = Command.fromString(ln, lno, srcName);
+ /* Ignore blank commands. */
+ if (com == null) continue;
+
+ /* Handle a command. */
+ CommandStatus sts = handleCommand(com, interactive);
+ /* Exit if we finished or encountered a fatal error. */
+ if (sts == FINISH || sts == ERROR) return;
+ } while (input.hasNextLine());
+ }
+
+ /** Handle a command.
+ *
+ * @param com
+ * The command to handle
+ *
+ * @param interactive
+ * Whether the current input source is interactive or not.
+ *
+ * @return The status of the executed command. */
+ public CommandStatus handleCommand(Command com, boolean interactive) {
+ /* Handle each command. */
+ switch (com.name) {
+ case "def-filtered":
+ return defFiltered(com);
+ case "def-layered":
+ return defLayered(com);
+ case "def-pushback":
+ return defPushback(com);
+ case "def-simple":
+ return defSimple(com);
+ case "def-serial":
+ return defSerial(com);
+ case "def-toggled":
+ return defToggled(com);
+ case "}":
+ case "end":
+ case "exit":
+ case "quit":
+ if (interactive)
+ System.out.printf(
+ "Exiting reader-conf, %d readers configured in %d commands\n",
+ blockReaders.size(), com.lno);
+ return FINISH;
+ default:
+ LOGGER.severe(com.error("Unknown command '%s'\n", com.name));
+ return FAIL;
+ }
+ }
+
+ private CommandStatus defFiltered(Command com) {
+ /*
+ * Get the block name.
+ */
+
+ String blockName = com.trimTo(' ');
+ if (blockName == null) {
+ LOGGER.severe(com.error("No name argument for def-filtered.\n"));
+ return FAIL;
+ }
+
+ /*
+ * Check there isn't a reader already bound to this name.
+ */
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader named %s\n", blockName));
+ }
+
+ /*
+ * Get the reader name.
+ */
+
+ String readerName = com.trimTo(' ');
+ if (readerName == null) {
+ LOGGER.severe(com.error("No reader-name argument for def-filtered.\n"));
+ return FAIL;
+ }
+
+ /*
+ * Check there is a reader bound to that name.
+ */
+ if (!blockReaders.containsKey(readerName)) {
+ LOGGER.severe(com.error("No source named %s\n", readerName));
+ return FAIL;
+ }
+
+ /*
+ * Get the pattern.
+ */
+ if (com.remn.equals("")) {
+ LOGGER.severe(com.error("No filter argument for def-filtered\n"));
+ return FAIL;
+ }
+
+ String filter = com.remn;
+
+ try {
+ Pattern pat = Pattern.compile(filter);
+
+ Predicate<Block> pred = block -> {
+ Matcher mat = pat.matcher(block.contents);
+
+ return mat.matches();
+ };
+
+ BlockReader reader
+ = new FilteredBlockReader(blockReaders.get(readerName), pred);
+
+ blockReaders.put(blockName, reader);
+ } catch (PatternSyntaxException psex) {
+ LOGGER.severe(com.error("Invalid regular expression '%s' for filter. (%s)\n",
+ filter, psex.getMessage()));
+ return FAIL;
+ }
+
+ return SUCCESS;
+ }
+
+ private CommandStatus defPushback(Command com) {
+ String[] parts = com.remn.split(" ");
+
+ if (parts.length != 2) {
+ LOGGER.severe(com.error(
+ "Incorrect number of arguments to def-pushback. Requires a block name and a reader name\n"));
+ return FAIL;
+ }
+
+ String blockName = parts[0];
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader %s\n", blockName));
+ return FAIL;
+ }
+
+ String readerName = parts[1];
+ if (!blockReaders.containsKey(readerName)) {
+ LOGGER.severe(com.error("No reader named %s\n", readerName));
+ return FAIL;
+ }
+
+ BlockReader reader = new PushbackBlockReader(blockReaders.get(readerName));
+ blockReaders.put(blockName, reader);
+
+ return SUCCESS;
+ }
+
+ private CommandStatus defToggled(Command com) {
+ String[] parts = com.remn.split(" ");
+
+ if (parts.length != 3) {
+ LOGGER.severe(com.error(
+ "Incorrect number of arguments to def-toggled. Requires a block name and two reader names\n"));
+ return FAIL;
+ }
+
+ /*
+ * Get the block name.
+ */
+ String blockName = parts[0];
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader named %s\n", blockName));
+ }
+
+ /*
+ * Make sure the component readers exist.
+ */
+ if (!blockReaders.containsKey(parts[1])) {
+ LOGGER.severe(com.error("No reader named %s\n", parts[1]));
+ return FAIL;
+ }
+
+ if (!blockReaders.containsKey(parts[2])) {
+ LOGGER.severe(com.error("No reader named %s\n", parts[2]));
+ return FAIL;
+ }
+
+ BlockReader reader = new ToggledBlockReader(blockReaders.get(parts[1]),
+ blockReaders.get(parts[2]));
+ blockReaders.put(blockName, reader);
+
+ return SUCCESS;
+ }
+
+ private CommandStatus defLayered(Command com) {
+ String[] parts = com.remn.split(" ");
+
+ if (parts.length != 3) {
+ LOGGER.severe(com.error(
+ "Incorrect number of arguments to def-layered. Requires a block name and two reader names\n"));
+ return FAIL;
+ }
+
+ /*
+ * Get the block name.
+ */
+ String blockName = parts[0];
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader named %s\n", blockName));
+ }
+
+ /*
+ * Make sure the component readers exist.
+ */
+ if (!blockReaders.containsKey(parts[1])) {
+ LOGGER.severe(com.error("No reader named %s\n", parts[1]));
+ return FAIL;
+ }
+
+ if (!blockReaders.containsKey(parts[2])) {
+ LOGGER.severe(com.error("No reader named %s\n", parts[2]));
+ return FAIL;
+ }
+
+ BlockReader reader = new LayeredBlockReader(blockReaders.get(parts[1]),
+ blockReaders.get(parts[2]));
+ blockReaders.put(blockName, reader);
+
+ return SUCCESS;
+ }
+
+ private CommandStatus defSerial(Command com) {
+ String[] parts = com.remn.split(" ");
+
+ if (parts.length < 2) {
+ LOGGER.severe(com.error(
+ "Not enough arguments to def-serial. Requires at least a block name and at least one reader name\n"));
+ return FAIL;
+ }
+
+ /*
+ * Get the name for this BlockReader.
+ */
+ String blockName = parts[0];
+ /*
+ * Check there isn't a reader already bound to this name.
+ */
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader named %s\n", blockName));
+ }
+
+ /*
+ * Get all of the component readers.
+ */
+ BlockReader[] readerArr = new BlockReader[parts.length - 1];
+ for (int i = 1; i < parts.length; i++) {
+ String readerName = parts[i];
+
+ /*
+ * Check there is a reader bound to that name.
+ */
+ if (!blockReaders.containsKey(readerName)) {
+ LOGGER.severe(com.error("No reader named %s\n", readerName));
+ return FAIL;
+ }
+
+ readerArr[i] = blockReaders.get(readerName);
+ }
+
+ BlockReader reader = new SerialBlockReader(readerArr);
+
+ blockReaders.put(blockName, reader);
+
+ return SUCCESS;
+ }
+
+ private CommandStatus defSimple(Command com) {
+ String remn = com.remn;
+
+ /*
+ * Get the block name.
+ */
+ /* :StringHandling */
+ int idx = remn.indexOf(' ');
+ if (idx == -1) {
+ LOGGER.severe(com.error("No name argument for def-simple.\n"));
+ return FAIL;
+ }
+ String blockName = remn.substring(0, idx).trim();
+ remn = remn.substring(idx).trim();
+
+ /*
+ * Check there isn't a reader already bound to this name.
+ */
+ if (blockReaders.containsKey(blockName)) {
+ LOGGER.warning(com.warn("Shadowing existing reader named %s\n", blockName));
+ }
+
+ /*
+ * Get the source name.
+ */
+ /* :StringHandling */
+ idx = remn.indexOf(' ');
+ if (idx == -1) {
+ LOGGER.severe(com.error("No source-name argument for def-simple.\n"));
+ return FAIL;
+ }
+ String sourceName = remn.substring(0, idx).trim();
+ remn = remn.substring(idx).trim();
+
+ /*
+ * Check there is a source bound to that name.
+ */
+ if (!ioReaders.containsKey(sourceName)) {
+ LOGGER.severe(com.error("No source named %s\n", sourceName));
+ return FAIL;
+ }
+
+ /*
+ * Get the pattern.
+ */
+ if (remn.equals("")) {
+ LOGGER.severe(com.error("No delimiter argument for def-simple\n"));
+ return FAIL;
+ }
+
+ String delim = remn;
+
+ /* Get the delimiter, and create the reader. */
+ try {
+ BlockReader reader
+ = new SimpleBlockReader(delim, ioReaders.get(sourceName));
+
+ blockReaders.put(blockName, reader);
+ } catch (PatternSyntaxException psex) {
+ LOGGER.severe(
+ com.error("Invalid regular expression '%s' for delimiter. (%s)\n",
+ delim, psex.getMessage()));
+ return FAIL;
+ }
+
+ return SUCCESS;
+ }
+}
diff --git a/pom.xml b/pom.xml
index e7e6a80..0aee431 100644
--- a/pom.xml
+++ b/pom.xml
@@ -34,5 +34,6 @@
<modules>
<module>base</module>
<module>clformat</module>
- </modules>
-</project>
+ <module>commander</module>
+ </modules>
+</project> \ No newline at end of file