/*
* Copyright (C) 2011 Benoit GUEROUT <bguerout at gmail dot com> and Yves AMSELLEM <amsellem dot yves at gmail dot com>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jongo;
import com.mongodb.AggregationOptions;
import com.mongodb.DBCollection;
import com.mongodb.DBObject;
import org.jongo.marshall.Unmarshaller;
import org.jongo.query.QueryFactory;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.concurrent.atomic.AtomicReference;
import static org.jongo.ResultHandlerFactory.newResultHandler;
public class Aggregate {
private final Unmarshaller unmarshaller;
private final QueryFactory queryFactory;
private final List<DBObject> pipeline;
private final AtomicReference<AggregationOptions> options;
private final DBCollection collection;
Aggregate(DBCollection collection, Unmarshaller unmarshaller, QueryFactory queryFactory) {
this.unmarshaller = unmarshaller;
this.queryFactory = queryFactory;
this.pipeline = new ArrayList<DBObject>();
this.options = new AtomicReference<AggregationOptions>();
this.collection = collection;
}
public Aggregate and(String pipelineOperator, Object... parameters) {
DBObject dbQuery = queryFactory.createQuery(pipelineOperator, parameters).toDBObject();
pipeline.add(dbQuery);
return this;
}
public <T> ResultsIterator<T> as(final Class<T> clazz) {
return map(newResultHandler(clazz, unmarshaller));
}
public Aggregate options(AggregationOptions options) {
this.options.set(options);
return this;
}
public <T> ResultsIterator<T> map(ResultHandler<T> resultHandler) {
Iterator<DBObject> results;
AggregationOptions options = this.options.get();
if (options != null) {
results = collection.aggregate(pipeline, options);
} else {
results = collection.aggregate(pipeline).results().iterator();
}
return new ResultsIterator<T>(results, resultHandler);
}
public static class ResultsIterator<E> implements Iterator<E>, Iterable<E> {
private Iterator<DBObject> results;
private ResultHandler<E> resultHandler;
private ResultsIterator(Iterator<DBObject> results, ResultHandler<E> resultHandler) {
this.resultHandler = resultHandler;
this.results = results;
}
public Iterator<E> iterator() {
return this;
}
public boolean hasNext() {
return results.hasNext();
}
public E next() {
if (!hasNext())
throw new NoSuchElementException();
DBObject dbObject = results.next();
return resultHandler.map(dbObject);
}
public void remove() {
throw new UnsupportedOperationException("remove() method is not supported");
}
}
}