/*
* Copyright (C) 2011 eXo Platform SAS.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.exoplatform.web.controller.router;
import java.util.LinkedList;
import org.exoplatform.web.controller.regexp.GroupType;
import org.exoplatform.web.controller.regexp.RENode;
import org.exoplatform.web.controller.regexp.REVisitor;
/**
* @author <a href="mailto:julien.viet@exoplatform.com">Julien Viet</a>
* @version $Revision$
*/
class CaptureGroupTransformation extends REVisitor<RuntimeException> {
/** . */
private int depth;
/** Top level group per disjunction. */
private LinkedList<RENode.Group> groups;
CaptureGroupTransformation() {
this.depth = 0;
this.groups = new LinkedList<RENode.Group>();
}
@Override
protected void visit(RENode.Disjunction disjunction) throws RuntimeException {
if (disjunction.hasAlternative()) {
RENode.Alternative alternative = disjunction.getAlternative();
if (alternative != null) {
alternative.accept(this);
//
if (depth == 0) {
if (groups.size() == 1 && groups.get(0).getQuantifier() == null) {
// Do nothing
} else {
// We make all the top level groups non capturing
for (RENode.Group group : groups) {
group.setType(GroupType.NON_CAPTURING_GROUP);
}
// We add a capturing group for the disjunction
RENode.Disjunction disjunction1 = new RENode.Disjunction((RENode.Alternative) null);
RENode.Group group = new RENode.Group(disjunction1, GroupType.CAPTURING_GROUP);
RENode.Alternative alternative1 = new RENode.Alternative(group);
alternative.replaceBy(alternative1);
disjunction1.setAlternative(alternative);
}
//
groups.clear();
}
} else {
if (depth == 0) {
disjunction.setAlternative(new RENode.Alternative(new RENode.Group(new RENode.Disjunction(),
GroupType.CAPTURING_GROUP)));
}
}
}
//
if (disjunction.hasNext()) {
RENode.Disjunction next = disjunction.getNext();
if (next != null) {
next.accept(this);
} else {
if (depth == 0) {
disjunction.setNext(new RENode.Disjunction(new RENode.Alternative(new RENode.Group(
new RENode.Disjunction(), GroupType.CAPTURING_GROUP))));
}
}
}
}
@Override
protected void visit(RENode.Group expr) throws RuntimeException {
if (depth == 0) {
// We collect all capturing top level capturing groups
if (expr.getType() == GroupType.CAPTURING_GROUP) {
groups.add(expr);
}
} else {
// We make nested capturing group as non capturing
if (expr.getType() == GroupType.CAPTURING_GROUP) {
expr.setType(GroupType.NON_CAPTURING_GROUP);
}
}
//
depth++;
super.visit(expr);
depth--;
}
}