mirror of
https://github.com/corda/corda.git
synced 2025-01-23 21:08:48 +00:00
Regex: Implement compiler for regular expression patterns
Originally, this developer wanted to (ab)use the PikeVM with a hand-crafted program and an added "callback" opcode to parse the regular expressions. However, this turned out to be completely unnecessary: there are no ambiguities in regular expression patterns, so there is no need to do anything else than parse the pattern, one character at a time, into a nested expression that then knows how to write itself into a program for the PikeVM. For the moment, we still hardcode the program for the regular expression pattern demonstrating the challenge with the prioritized threads because the compiler cannot yet parse reluctant operators. Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
This commit is contained in:
parent
26c4bf8d8b
commit
04d8955f98
171
test/regex/Compiler.java
Normal file
171
test/regex/Compiler.java
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
/* Copyright (c) 2008-2013, Avian Contributors
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and/or distribute this software
|
||||||
|
for any purpose with or without fee is hereby granted, provided
|
||||||
|
that the above copyright notice and this permission notice appear
|
||||||
|
in all copies.
|
||||||
|
|
||||||
|
There is NO WARRANTY for this software. See license.txt for
|
||||||
|
details. */
|
||||||
|
|
||||||
|
package regex;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Stack;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compiles regular expressions into {@link PikeVM}s.
|
||||||
|
*
|
||||||
|
* @author Johannes Schindelin
|
||||||
|
*/
|
||||||
|
class Compiler implements PikeVMOpcodes {
|
||||||
|
private final static CharacterMatcher regularCharacter =
|
||||||
|
CharacterMatcher.parse("[^\\\\.*+?|\\[\\]{}()^$]");
|
||||||
|
|
||||||
|
private static class Output {
|
||||||
|
private int[] program;
|
||||||
|
private int offset;
|
||||||
|
private int groupCount = -1;
|
||||||
|
|
||||||
|
public Output(Expression expr) {
|
||||||
|
// try-run to determine the code size
|
||||||
|
expr.writeCode(this);
|
||||||
|
program = new int[offset];
|
||||||
|
offset = 0;
|
||||||
|
groupCount = -1;
|
||||||
|
// write it out!
|
||||||
|
expr.writeCode(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void add(int opcode) {
|
||||||
|
if (program != null) {
|
||||||
|
program[offset] = opcode;
|
||||||
|
}
|
||||||
|
offset++;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int markJump() {
|
||||||
|
return offset++;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setJump(int mark) {
|
||||||
|
if (program != null) {
|
||||||
|
program[mark] = offset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public PikeVM toVM() {
|
||||||
|
return new PikeVM(program, groupCount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private abstract class Expression {
|
||||||
|
protected abstract void writeCode(Output output);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class QuestionMark extends Expression {
|
||||||
|
private Expression expr;
|
||||||
|
|
||||||
|
public QuestionMark(Expression expr) {
|
||||||
|
this.expr = expr;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void writeCode(Output output) {
|
||||||
|
output.add(SPLIT);
|
||||||
|
int jump = output.markJump();
|
||||||
|
expr.writeCode(output);
|
||||||
|
output.setJump(jump);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Group extends Expression {
|
||||||
|
private ArrayList<Expression> list = new ArrayList<Expression>();
|
||||||
|
|
||||||
|
public void push(Expression expr) {
|
||||||
|
list.add(expr);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void push(final int c) {
|
||||||
|
push(new Expression() {
|
||||||
|
public void writeCode(Output output) {
|
||||||
|
output.add(c);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public Expression pop() {
|
||||||
|
Expression result = list.remove(list.size() - 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void writeCode(Output output) {
|
||||||
|
int groupIndex = ++ output.groupCount;
|
||||||
|
output.add(SAVE_OFFSET);
|
||||||
|
output.add(2 * groupIndex);
|
||||||
|
for (Expression expr : list) {
|
||||||
|
expr.writeCode(output);
|
||||||
|
}
|
||||||
|
output.add(SAVE_OFFSET);
|
||||||
|
output.add(2 * groupIndex + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Group0 extends Expression {
|
||||||
|
private final Group group;
|
||||||
|
|
||||||
|
public Group0() {
|
||||||
|
group = new Group();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void writeCode(Output output) {
|
||||||
|
group.writeCode(output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Group0 root;
|
||||||
|
private Stack<Group> groups;
|
||||||
|
|
||||||
|
public Compiler() {
|
||||||
|
root = new Group0();
|
||||||
|
groups = new Stack<Group>();
|
||||||
|
groups.add(root.group);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pattern compile(String regex) {
|
||||||
|
char[] array = regex.toCharArray();
|
||||||
|
for (int index = 0; index < array.length; ++ index) {
|
||||||
|
char c = array[index];
|
||||||
|
Group current = groups.peek();
|
||||||
|
if (regularCharacter.matches(c)) {
|
||||||
|
current.push(c);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
switch (c) {
|
||||||
|
case '?':
|
||||||
|
current.push(new QuestionMark(current.pop()));
|
||||||
|
break;
|
||||||
|
case '(':
|
||||||
|
if (index + 1 < array.length && array[index + 1] == '?') {
|
||||||
|
throw new UnsupportedOperationException("Not yet supported: "
|
||||||
|
+ regex.substring(index));
|
||||||
|
}
|
||||||
|
current.push(groups.push(new Group()));
|
||||||
|
continue;
|
||||||
|
case ')':
|
||||||
|
if (groups.size() < 2) {
|
||||||
|
throw new RuntimeException("Invalid group close @" + index + ": "
|
||||||
|
+ regex);
|
||||||
|
}
|
||||||
|
groups.pop();
|
||||||
|
continue;
|
||||||
|
default:
|
||||||
|
throw new RuntimeException("Parse error @" + index + ": " + regex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (groups.size() != 1) {
|
||||||
|
throw new IllegalArgumentException("Unclosed groups: ("
|
||||||
|
+ (groups.size() - 1) + "): " + regex);
|
||||||
|
}
|
||||||
|
return new RegexPattern(regex, 0, new Output(root).toVM());
|
||||||
|
}
|
||||||
|
}
|
@ -49,19 +49,7 @@ public abstract class Pattern implements PikeVMOpcodes {
|
|||||||
if (flags != 0) {
|
if (flags != 0) {
|
||||||
throw new UnsupportedOperationException("TODO");
|
throw new UnsupportedOperationException("TODO");
|
||||||
}
|
}
|
||||||
if ("a(bb)?a".equals(regex)) {
|
if ("a(a*?)(a?)(a??)(a+)(a*)a".equals(regex)) {
|
||||||
return new RegexPattern(regex, flags, new PikeVM(new int[] {
|
|
||||||
SAVE_OFFSET, 0,
|
|
||||||
'a',
|
|
||||||
SPLIT, 11,
|
|
||||||
SAVE_OFFSET, 2,
|
|
||||||
'b',
|
|
||||||
'b',
|
|
||||||
SAVE_OFFSET, 3,
|
|
||||||
/* 11 */ 'a',
|
|
||||||
SAVE_OFFSET, 1
|
|
||||||
}, 1));
|
|
||||||
} else if ("a(a*?)(a?)(a??)(a+)(a*)a".equals(regex)) {
|
|
||||||
return new RegexPattern(regex, flags, new PikeVM(new int[] {
|
return new RegexPattern(regex, flags, new PikeVM(new int[] {
|
||||||
SAVE_OFFSET, 0,
|
SAVE_OFFSET, 0,
|
||||||
'a',
|
'a',
|
||||||
@ -91,7 +79,7 @@ public abstract class Pattern implements PikeVMOpcodes {
|
|||||||
SAVE_OFFSET, 1
|
SAVE_OFFSET, 1
|
||||||
}, 5));
|
}, 5));
|
||||||
}
|
}
|
||||||
throw new UnsupportedOperationException("Cannot handle regex " + regex);
|
return new Compiler().compile(regex);
|
||||||
}
|
}
|
||||||
|
|
||||||
public int flags() {
|
public int flags() {
|
||||||
|
Loading…
Reference in New Issue
Block a user