/*
* Copyright (C) 2008 Steve Ratcliffe
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*
* Author: Steve Ratcliffe
* Create date: 02-Nov-2008
*/
package uk.me.parabola.mkgmap.osmstyle;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.Reader;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.general.LevelInfo;
import uk.me.parabola.mkgmap.osmstyle.actions.ActionList;
import uk.me.parabola.mkgmap.osmstyle.actions.ActionReader;
import uk.me.parabola.mkgmap.osmstyle.eval.AndOp;
import uk.me.parabola.mkgmap.osmstyle.eval.BinaryOp;
import uk.me.parabola.mkgmap.osmstyle.eval.ExistsOp;
import uk.me.parabola.mkgmap.osmstyle.eval.ExpressionReader;
import uk.me.parabola.mkgmap.osmstyle.eval.LinkedOp;
import uk.me.parabola.mkgmap.osmstyle.eval.Op;
import uk.me.parabola.mkgmap.osmstyle.eval.OrOp;
import uk.me.parabola.mkgmap.osmstyle.eval.ValueOp;
import uk.me.parabola.mkgmap.reader.osm.GType;
import uk.me.parabola.mkgmap.reader.osm.Rule;
import uk.me.parabola.mkgmap.scan.SyntaxException;
import uk.me.parabola.mkgmap.scan.TokenScanner;
import static uk.me.parabola.mkgmap.osmstyle.eval.AbstractOp.*;
/**
* Read a rules file. A rules file contains a list of rules and the
* resulting garmin type, should the rule match.
*
* <pre>
*
* </pre>
* @author Steve Ratcliffe
*/
public class RuleFileReader
{
private static final Logger log =
Logger.
getLogger(RuleFileReader.
class);
private final TypeReader typeReader
;
private final RuleSet rules
;
private TokenScanner scanner
;
public RuleFileReader
(int kind, LevelInfo
[] levels, RuleSet rules
) {
this.
rules = rules
;
typeReader =
new TypeReader
(kind, levels
);
}
/**
* Read a rules file.
* @param loader A file loader.
* @param name The name of the file to open.
* @throws FileNotFoundException If the given file does not exist.
*/
public void load
(StyleFileLoader loader,
String name
) throws FileNotFoundException {
Reader r = loader.
open(name
);
load
(r, name
);
}
void load
(Reader r,
String name
) {
scanner =
new TokenScanner
(name, r
);
scanner.
setExtraWordChars("-:.");
ExpressionReader expressionReader =
new ExpressionReader
(scanner
);
ActionReader actionReader =
new ActionReader
(scanner
);
// Read all the rules in the file.
scanner.
skipSpace();
while (!scanner.
isEndOfFile()) {
Op expr = expressionReader.
readConditions();
ActionList actionList = actionReader.
readActions();
// If there is an action list, then we don't need a type
GType type =
null;
if (scanner.
checkToken("["))
type = typeReader.
readType(scanner
);
else if (actionList ==
null)
throw new SyntaxException
(scanner,
"No type definition given");
saveRule
(expr, actionList, type
);
scanner.
skipSpace();
}
rules.
addUsedTags(expressionReader.
getUsedTags());
rules.
addUsedTags(actionReader.
getUsedTags());
rules.
prepare();
}
/**
* Save the expression as a rule. We need to extract an index such
* as highway=primary first and then add the rest of the expression as
* the condition for it.
*
* So in other words each condition is dropped into a number of different
* baskets based on the first 'tag=value' term. We then only look
* for expressions that are in the correct basket. For each expression
* in a basket we know that the first term is true so we can drop that
* from the expression.
*/
private void saveRule
(Op op, ActionList actions, GType gt
) {
log.
info("EXP", op,
", type=", gt
);
//System.out.println("From: " + op);
Op op2 = rearrangeExpression
(op
);
//System.out.println("TO : " + op2);
if (op2
instanceof BinaryOp
) {
optimiseAndSaveBinaryOp
((BinaryOp
) op2, actions, gt
);
} else {
optimiseAndSaveOtherOp
(op2, actions, gt
);
}
}
/**
* Rearrange the expression so that it is solvable, that is it starts with
* an EQUALS or an EXISTS.
* @param op The expression to be rearranged.
* @return An equivalent expression re-arranged so that it starts with an
* indexable term. If that is not possible then the original expression is
* returned.
*/
private static Op rearrangeExpression
(Op op
) {
if (isFinished
(op
))
return op
;
if (op.
isType(AND
)) {
// Recursively re-arrange the child nodes
rearrangeExpression
(op.
getFirst());
rearrangeExpression
(((BinaryOp
) op
).
getSecond());
swapForSelectivity
((BinaryOp
) op
);
Op op1 = op.
getFirst();
Op op2 =
((BinaryOp
) op
).
getSecond();
// If the first term is an EQUALS or EXISTS then this subtree is
// already solved and we need to do no more.
if (isSolved
(op1
)) {
return rearrangeAnd
((BinaryOp
) op, op1, op2
);
} else if (isSolved
(op2
)) {
return rearrangeAnd
((BinaryOp
) op, op2, op1
);
}
}
return op
;
}
/**
* Swap the terms so that the most selective or fastest term to calculate
* is first.
* @param op A AND operation.
*/
private static void swapForSelectivity
(BinaryOp op
) {
Op first = op.
getFirst();
int sel1 = selectivity
(first
);
Op second = op.
getSecond();
int sel2 = selectivity
(second
);
if (sel1
> sel2
) {
op.
setFirst(second
);
op.
setSecond(first
);
}
}
/**
* Rearrange an AND expression so that it can be executed with indexable
* terms at the front.
* @param top This will be an AndOp.
* @param op1 This is a child of top that is guaranteed to be
* solved already.
* @param op2 This expression is the other child of top.
* @return A re-arranged expression with an indexable term at the beginning
* or several such expressions ORed together.
*/
private static BinaryOp rearrangeAnd
(BinaryOp top, Op op1, Op op2
) {
if (isIndexable
(op1
)) {
top.
setFirst(op1
);
top.
setSecond(op2
);
return top
;
} else if (op1.
isType(AND
)) {
// The first term is AND.
// If its first term is indexable (EQUALS or EXIST) then we
// re-arrange the tree so that that term is first.
Op first = op1.
getFirst();
if (isIndexable
(first
)) {
top.
setFirst(first
);
op1.
setFirst(op2
);
swapForSelectivity
((AndOp
) op1
);
top.
setSecond(op1
);
return top
;
}
} else if (op1.
isType(OR
)) {
// Transform ((first | second) & topSecond)
// into (first & topSecond) | (second & topSecond)
Op first = op1.
getFirst();
OrOp orOp =
new OrOp
();
Op topSecond = top.
getSecond();
AndOp and1 =
new AndOp
();
and1.
setFirst(first
);
and1.
setSecond(topSecond
);
AndOp and2 =
new AndOp
();
Op second = rearrangeExpression
(((OrOp
) op1
).
getSecond());
and2.
setFirst(second
);
and2.
setSecond(topSecond
);
orOp.
setFirst(and1
);
orOp.
setSecond(and2
);
return orOp
;
} else {
// This shouldn't happen
throw new SyntaxException
("X3" + op1.
getType());
}
return top
;
}
/**
* True if this operation can be indexed. It is a plain equality or
* Exists operation.
*/
private static boolean isIndexable
(Op op
) {
return op.
isType(EQUALS
) || op.
isType(EXISTS
);
}
/**
* True if this expression is 'solved'. This means that the first term
* is indexable or it is indexable itself.
*/
private static boolean isSolved
(Op op
) {
return isIndexable
(op
) || isIndexable
(op.
getFirst());
}
/**
* True if there is nothing more that we can do to rearrange this expression.
* It is either solved or it cannot be solved.
*/
private static boolean isFinished
(Op op
) {
// If we can improve the ordering then we are not done just yet
if (op.
isType(AND
) && selectivity
(op.
getFirst()) > selectivity
(((BinaryOp
) op
).
getSecond()))
return false;
if (isSolved
(op
))
return true;
char type = op.
getType();
switch (type
) {
case AND:
return false;
case OR:
return false;
default:
return true;
}
}
/**
* Get a value for how selective this operation is. We try to bring
* EQUALS to the front followed by EXISTS. Without knowing tag
* frequency you can only guess at what the most selective operations
* are, so all we do is arrange EQUALS - EXISTS - everything else.
* Note that you must have an EQUALS or EXISTS first, so you can't
* bring anything else earlier than them.
*
* @return An integer, lower values mean the operation should be earlier
* in the expression than operations with higher values.
*/
private static int selectivity
(Op op
) {
switch (op.
getType()) {
case EQUALS:
return 0;
case EXISTS:
return 1;
case AND:
case OR:
return Math.
min(selectivity
(op.
getFirst()), selectivity
(((BinaryOp
) op
).
getSecond()));
default:
return 1000;
}
}
private void optimiseAndSaveOtherOp
(Op op, ActionList actions, GType gt
) {
if (op.
isType(EXISTS
)) {
// The lookup key for the exists operation is 'tag=*'
createAndSaveRule
(op.
value() +
"=*", op, actions, gt
);
} else {
throw new SyntaxException
(scanner,
"Invalid operation '" + op.
getType() +
"' at top level");
}
}
/**
* Optimise the expression tree, extract the primary key and
* save it as a rule.
* @param op a binary expression
* @param actions list of actions to execute on match
* @param gt the Garmin type of the element
*/
private void optimiseAndSaveBinaryOp
(BinaryOp op, ActionList actions, GType gt
) {
Op first = op.
getFirst();
Op second = op.
getSecond();
log.
debug("binop", op.
getType(), first.
getType());
/*
* We allow the following cases:
* An EQUALS at the top.
* An AND at the top level.
* An OR at the top level.
*/
String keystring
;
if (op.
isType(EQUALS
)) {
if (first.
isType(VALUE
) && second.
isType(VALUE
)) {
keystring = op.
value();
} else {
throw new SyntaxException
(scanner,
"Invalid rule file (expr " + op.
getType() +
')');
}
} else if (op.
isType(AND
)) {
if (first.
isType(EQUALS
)) {
keystring = first.
value();
} else if (first.
isType(EXISTS
)) {
keystring = first.
value() +
"=*";
} else if (first.
isType(NOT_EXISTS
)) {
throw new SyntaxException
(scanner,
"Cannot start rule with tag!=*");
} else {
throw new SyntaxException
(scanner,
"Invalid rule file (expr " + op.
getType() +
')');
}
} else if (op.
isType(OR
)) {
LinkedOp op1 = LinkedOp.
create(first,
true);
saveRule
(op1, actions, gt
);
saveRestOfOr
(actions, gt, second, op1
);
return;
} else {
// We can make every other binary op work by converting to AND(EXISTS, op)
Op existsOp =
new ExistsOp
();
existsOp.
setFirst(new ValueOp
(op.
getFirst().
value()));
AndOp andOp =
new AndOp
();
andOp.
setFirst(existsOp
);
andOp.
setSecond(op
);
optimiseAndSaveBinaryOp
(andOp, actions, gt
);
return;
}
createAndSaveRule
(keystring, op, actions, gt
);
}
private void saveRestOfOr
(ActionList actions, GType gt, Op second, LinkedOp op1
) {
if (second.
isType(OR
)) {
LinkedOp nl = LinkedOp.
create(second.
getFirst(),
false);
op1.
setLink(nl
);
saveRule
(nl, actions, gt
);
saveRestOfOr
(actions, gt,
((BinaryOp
)second
).
getSecond(), op1
);
} else {
LinkedOp op2 = LinkedOp.
create(second,
false);
op1.
setLink(op2
);
saveRule
(op2, actions, gt
);
}
}
private void createAndSaveRule
(String keystring, Op expr, ActionList actions, GType gt
) {
Rule rule
;
if (actions.
isEmpty())
rule =
new ExpressionRule
(expr, gt
);
else
rule =
new ActionRule
(expr, actions.
getList(), gt
);
rules.
add(keystring, rule, actions.
getChangeableTags());
}
public static void main
(String[] args
) throws FileNotFoundException {
if (args.
length > 0) {
Reader r =
new FileReader(args
[0]);
RuleSet rs =
new RuleSet
();
RuleFileReader rr =
new RuleFileReader
(GType.
POLYLINE,
LevelInfo.
createFromString("0:24 1:20 2:18 3:16 4:14"), rs
);
rr.
load(r,
"string");
System.
out.
println("Result: " + rs
);
} else {
System.
err.
println("Usage: RuleFileReader <file>");
}
}
}