/* Copyright (2007-2012) Schibsted ASA
* This file is part of Possom.
*
* Possom 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 3 of the License, or
* (at your option) any later version.
*
* Possom 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 Possom. If not, see <http://www.gnu.org/licenses/>.
* FullnameAlternation.java
*
*
*/
package no.sesat.search.query.parser.alt;
import java.util.LinkedList;
import java.util.Set;
import no.sesat.search.query.Clause;
import no.sesat.search.query.DefaultOperatorClause;
import no.sesat.search.query.BinaryClause;
import no.sesat.search.query.OrClause;
import no.sesat.search.query.PhraseClause;
import no.sesat.search.query.XorClause;
import no.sesat.search.query.XorClause.Hint;
import no.sesat.search.query.XorClause.Hint;
import no.sesat.search.query.finder.Counter;
import no.sesat.search.query.finder.PredicateFinder;
import no.sesat.search.query.parser.alt.AbstractAlternation;
import no.sesat.search.query.token.Categories;
import no.sesat.search.query.token.TokenPredicate;
import org.apache.log4j.Logger;
/** SEARCH-597 Forbedringer av proximity på navn.
* The results of this alternation depend on the skin's FULLNAME list.
*
*
* @version <tt>$Id$</tt>
*/
public final class FullnameAlternation extends AbstractAlternation {
// Constants -----------------------------------------------------
private static final Logger LOG = Logger.getLogger(FullnameAlternation.class);
// Attributes ----------------------------------------------------
// Static --------------------------------------------------------
// Constructors --------------------------------------------------
/** Creates a new instance of FullnameAlternation
* @param cxt
*/
public FullnameAlternation(final Context cxt) {
super(cxt);
}
/** {@inherit} **/
public Clause alternate(final Clause clause) {
final PredicateFinder finder = new PredicateFinder();
final Set<Clause> fullnames = finder.findClauses(
clause,
Categories.FULLNAME,
context.getTokenEvaluationEngine());
Clause result = clause;
for(Clause fullname : fullnames){
if( fullname instanceof DefaultOperatorClause ){
// find the right-leaning equilivalent of this fullname
// since the presumption is that the surname is on the right hand side
DefaultOperatorClause rightLeaningFullname = null;
for(Clause fn : fullnames){
if( fn.getTerm().replaceAll("\\(|\\)", "").equals(fullname.getTerm().replaceAll("\\(|\\)", "")) ){
rightLeaningFullname = (DefaultOperatorClause) fn;
//break;
}
}
final DefaultOperatorClause doFullname = (DefaultOperatorClause)fullname;
// determine which children make up da lastname (we'll take da largest & furthest right fullname found)
// final Set<Clause> lastnames = finder.findClauses(
// rightLeaningFullname,
// TokenPredicate.LASTNAME,
// context.getTokenEvaluationEngine());
Clause surname = finder.findFirstClause(
rightLeaningFullname,
Categories.LASTNAME,
context.getTokenEvaluationEngine());
// for(Clause lastname : lastnames){
// if( null != surname ){
// surname = new Counter().getTermCount(lastname) == new Counter().getTermCount(surname)
// ? lastname
// : surname;
// }else{
// surname = lastname;
// }
// }
if(null == surname){
// default to second clause
surname = rightLeaningFullname.getSecondClause();
}
final DefaultOperatorClause surnameParent
= (DefaultOperatorClause)context.getParentFinder().getParent(rightLeaningFullname, surname);
if(rightLeaningFullname.getSecondClause() != surname){
// surname must be on the right side
surname = rightLeaningFullname.getSecondClause();
}
LOG.debug("surname detected as " + surname);
// and the clause that comprises all givennames
final Clause givennames = rightLeaningFullname.getFirstClause();
LOG.debug("givennames detected as " + givennames);
// create fullname phrase clause
final PhraseClause fullnamePhrased = context
.createPhraseClause('\"' + givennames.getTerm() + ' ' + surname.getTerm() + '\"', null);
LOG.debug("fullname phraseClause created " + fullnamePhrased);
// create reversed fullname phrase clause
final PhraseClause reversedPhrased = context
.createPhraseClause('\"' + surname.getTerm() + ' ' + givennames.getTerm() + '\"', null);
LOG.debug("reversed phraseClause created " + reversedPhrased);
// create the OR emcompassing both fullname versions
final OrClause orPhrasedClause = context.createOrClause(fullnamePhrased, reversedPhrased);
LOG.debug("orClause created " + orPhrasedClause);
// create the OR emcompassing the fullname versions and the original unphrased fullnames
final OrClause orClause = context.createOrClause(orPhrasedClause, doFullname);
LOG.debug("orClause created " + orClause);
// create xorClause
final LinkedList<Clause> list = new LinkedList<Clause>();
list.add(doFullname);
list.add(orClause);
final XorClause xorClause = createXorClause(list);
LOG.debug("XorClause created " + xorClause);
// replace fullname with the new xorClause
final BinaryClause fullnameParent = result == fullname
? doFullname
: (BinaryClause) context.getParentFinder().getParent(result, fullname);
result = result == fullname
? xorClause
: replaceDescendant((BinaryClause)result, xorClause, doFullname, fullnameParent);
LOG.debug("Updated root to " + result);
}
}
return result;
}
// Public --------------------------------------------------------
// Package protected ---------------------------------------------
// Protected -----------------------------------------------------
/** {@inherit} **/
@Override
protected Hint getAlternationHint() {
return Hint.FULLNAME_ON_LEFT;
}
// Private -------------------------------------------------------
// Inner classes -------------------------------------------------
}