blob: 3c23c141e915b56b76a3878569690ef1fe739db9 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.freemarker.core.model.impl;
import org.apache.freemarker.core.TemplateException;
import org.apache.freemarker.core.model.*;
import java.io.Serializable;
import java.util.Iterator;
import java.util.List;
/**
* A simple implementation of {@link TemplateIterableModel}.
* It's able to wrap {@link Iterable}-s and {@link Iterator}-s.
* If you wrap an {@link Iterator}, the variable can be <#list>-ed only once!
*
* <p>Consider using {@link SimpleSequence} instead of this class if you want to wrap {@link Iterator}-s.
* {@link SimpleSequence} will read all elements of the {@link Iterator}, and store them in a {@link List}
* (this may cause too high resource consumption in some applications), so you can list the variable
* for unlimited times. Also, if you want to wrap {@link Iterable}-s, and then list the resulting
* variable for many times, {@link SimpleSequence} may gives better performance, as the
* wrapping of non-{@link TemplateModel} objects happens only once.
*
* <p>This class is thread-safe. The returned {@link TemplateModelIterator}-s
* are <em>not</em> thread-safe.
*/
public class SimpleIterable extends WrappingTemplateModel
implements TemplateIterableModel, Serializable {
private boolean iteratorOwned;
private final Iterator iterator;
private final Iterable iterable;
public SimpleIterable(Iterator iterator, ObjectWrapper wrapper) {
super(wrapper);
this.iterator = iterator;
iterable = null;
}
public SimpleIterable(Iterable iterable, ObjectWrapper wrapper) {
super(wrapper);
this.iterable = iterable;
iterator = null;
}
/**
* Retrieves a template model iterator that is used to iterate over the elements in this iterable.
*
* <p>When you wrap an {@code Iterator} and you get {@code TemplateModelIterator} for multiple times,
* only one of the returned {@code TemplateModelIterator} instances can be really used. When you have called a
* method of a {@code TemplateModelIterator} instance, all other instance will throw a
* {@link TemplateException} when you try to call their methods, since the wrapped {@code Iterator}
* can't return the first element anymore.
*/
@Override
public TemplateModelIterator iterator() {
return iterator != null
? new SimpleTemplateModelIterator(iterator, false)
: new SimpleTemplateModelIterator(iterable.iterator(), true);
}
/**
* Wraps an {@link Iterator}; not thread-safe. The encapsulated {@link Iterator} may be accessible from multiple
* threads (as multiple {@link SimpleTemplateModelIterator} instance can wrap the same {@link Iterator} instance),
* but if the {@link Iterator} was marked in the constructor as shared, the first thread which uses the
* {@link Iterator} will monopolize that.
*/
private class SimpleTemplateModelIterator implements TemplateModelIterator {
private final Iterator iterator;
private boolean iteratorOwnedByMe;
SimpleTemplateModelIterator(Iterator iterator, boolean iteratorOwnedByMe) {
this.iterator = iterator;
this.iteratorOwnedByMe = iteratorOwnedByMe;
}
@Override
public TemplateModel next() throws TemplateException {
if (!iteratorOwnedByMe) {
synchronized (SimpleIterable.this) {
checkIteratorNotOwned();
iteratorOwned = true;
iteratorOwnedByMe = true;
}
}
Object value = iterator.next();
return value instanceof TemplateModel ? (TemplateModel) value : wrap(value);
}
@Override
public boolean hasNext() throws TemplateException {
// Calling hasNext may looks safe, but I have met sync. problems.
if (!iteratorOwnedByMe) {
synchronized (SimpleIterable.this) {
checkIteratorNotOwned();
}
}
return iterator.hasNext();
}
private void checkIteratorNotOwned() throws TemplateException {
if (iteratorOwned) {
throw new TemplateException(
"This value wraps a java.util.Iterator, thus it can be listed only once.");
}
}
}
}