/*
* See the NOTICE file distributed with this work for additional
* information regarding copyright ownership.
*
* 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.xwiki.query.internal;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xwiki.component.manager.ComponentLookupException;
import org.xwiki.component.manager.ComponentManager;
import org.xwiki.query.Query;
import org.xwiki.query.QueryException;
import org.xwiki.query.QueryFilter;
import org.xwiki.query.QueryManager;
import org.xwiki.query.QueryParameter;
import org.xwiki.query.SecureQuery;
import java.util.List;
import java.util.Map;
/**
* Query wrapper that allows to set filter from the filter component hint.
*
* @version $Id: 415e89e4afd1c7aea31db2d6028bc751bb5ce588 $
* @since 4.0RC1
*/
public class ScriptQuery implements SecureQuery
{
/**
* Used to log possible warnings.
*/
private static final Logger LOGGER = LoggerFactory.getLogger(ScriptQuery.class);
/**
* Used to retrieve {@link org.xwiki.query.QueryFilter} implementations.
*/
private ComponentManager componentManager;
/**
* The wrapped {@link Query}.
*/
private Query query;
/**
* Constructor.
*
* @param query the query object to wrap.
* @param cm the xwiki component manager.
*/
public ScriptQuery(Query query, ComponentManager cm)
{
this.query = query;
this.componentManager = cm;
}
/**
* Set a filter in the wrapped query from the filter component hint.
*
* @param filter the hint of the component filter to set in the wrapped query.
* @return this query object.
*/
public Query addFilter(String filter)
{
if (!StringUtils.isBlank(filter)) {
try {
QueryFilter queryFilter = this.componentManager.getInstance(QueryFilter.class, filter);
addFilter(queryFilter);
} catch (ComponentLookupException e) {
// We need to avoid throwing exceptions in the wiki if the filter does not exist.
LOGGER.warn("Failed to load QueryFilter with component hint [{}]", filter);
}
}
return this;
}
/**
* Allow to retrieve the total count of items for the given query instead of the actual results. This method will
* only work for queries selecting document full names, see {@link CountDocumentFilter} for more information.
*
* @return the total number of results for this query.
*/
public long count()
{
long result = -1;
try {
// Create a copy of the wrapped query.
QueryManager queryManager = (QueryManager) this.componentManager.getInstance(QueryManager.class);
Query countQuery = queryManager.createQuery(getStatement(), getLanguage());
countQuery.setWiki(getWiki());
for (Map.Entry<Integer, Object> entry : getPositionalParameters().entrySet()) {
countQuery.bindValue(entry.getKey(), entry.getValue());
}
for (Map.Entry<String, Object> entry : getNamedParameters().entrySet()) {
countQuery.bindValue(entry.getKey(), entry.getValue());
}
for (QueryFilter filter : getFilters()) {
countQuery.addFilter(filter);
}
// Add the count filter to it.
countQuery.addFilter(this.componentManager.<QueryFilter>getInstance(QueryFilter.class, "count"));
// Execute and retrieve the count result.
List<Long> results = countQuery.execute();
result = results.get(0);
} catch (Exception e) {
LOGGER.warn("Failed to create count query for query [{}]", getStatement());
e.printStackTrace();
}
return result;
}
@Override
public String getStatement()
{
return this.query.getStatement();
}
@Override
public String getLanguage()
{
return this.query.getLanguage();
}
@Override
public boolean isNamed()
{
return this.query.isNamed();
}
@Override
public Query setWiki(String wiki)
{
this.query.setWiki(wiki);
return this;
}
@Override
public String getWiki()
{
return this.query.getWiki();
}
@Override
public Query bindValue(String var, Object val)
{
this.query.bindValue(var, val);
return this;
}
@Override
public Query bindValue(int index, Object val)
{
this.query.bindValue(index, val);
return this;
}
@Override
public Query bindValues(List<Object> values)
{
this.query.bindValues(values);
return this;
}
@Override
public QueryParameter bindValue(String var)
{
QueryParameter parameter = this.query.bindValue(var);
return new ScriptQueryParameter(this, parameter);
}
@Override
public Map<String, Object> getNamedParameters()
{
return this.query.getNamedParameters();
}
@Override
public Map<Integer, Object> getPositionalParameters()
{
return this.query.getPositionalParameters();
}
@Override
public Query addFilter(QueryFilter filter)
{
this.query.addFilter(filter);
return this;
}
@Override
public List<QueryFilter> getFilters()
{
return this.query.getFilters();
}
@Override
public Query setLimit(int limit)
{
this.query.setLimit(limit);
return this;
}
@Override
public Query setOffset(int offset)
{
this.query.setOffset(offset);
return this;
}
@Override
public int getLimit()
{
return this.query.getLimit();
}
@Override
public int getOffset()
{
return this.query.getOffset();
}
@Override
public <T> List<T> execute() throws QueryException
{
return this.query.execute();
}
@Override
public boolean isCurrentAuthorChecked()
{
return this.query instanceof SecureQuery ? ((SecureQuery) this.query).isCurrentAuthorChecked() : true;
}
@Override
public SecureQuery checkCurrentAuthor(boolean checkCurrentAuthor)
{
// Always check current author for scripts
return this;
}
@Override
public boolean isCurrentUserChecked()
{
return this.query instanceof SecureQuery ? ((SecureQuery) this.query).isCurrentAuthorChecked() : false;
}
@Override
public SecureQuery checkCurrentUser(boolean checkCurrentUser)
{
if (this.query instanceof SecureQuery) {
((SecureQuery) this.query).isCurrentAuthorChecked();
}
return this;
}
}